一个进程中,线程并非单独存在,往往需要与其他线程进行通讯以执行特定的任务。接下来就用一个简单的例子来实现线程之间最简单的通讯,并借此探究一下UI控件下得常见设置
需求:从网络上下载一张图片在屏幕上显示,图片可以滚动,可以捏合缩放大小
效果如下:
开始图片没有显示是因为正在从网上下载
项目开搞
新建一个新项目。
因为视图有滚动的需求,所以需要添加一个UIScrollView以及一个显示图片的UIImageView
@interface ViewController () <UIScrollViewDelegate>
@property (nonatomic,strong) UIScrollView *scrollView;
@property (weak,nonatomic) UIImageView *imgV;
@end
在loadView方法里面初始化UI控件,将scrollView设为根视图
- (void)loadView
{
_scrollView = [[UIScrollView alloc] init];
self.view = _scrollView;
self.view.backgroundColor = [UIColor brownColor];
// 初始化UIImageView
UIImageView *imgV = [[UIImageView alloc] init];
[self.view addSubview:imgV];
self.imgV = imgV;
}
设置scrollView的代理,图片的缩放操作将由代理的方法来实现,在后台开启一条线程执行下载图片的耗时操作:
- (void)viewDidLoad {
[super viewDidLoad];
_scrollView.delegate = self;
// 最小缩放比例
_scrollView.minimumZoomScale = 0.5;
// 最大缩放比例
_scrollView.maximumZoomScale = 2.0;
// 在后台开启一条线程执行下载图片的耗时操作
[self performSelectorInBackground:@selector(downloadImage) withObject:nil];
// 打印所在线程
NSLog(@"viewDidLoad - %@",[NSThread currentThread]);
}
将下载图片抽取为一个方法downloadImage,该方法中实现了线程间的通信 ———— 让主线程更新UI,通知主线程区执行setImage:方法,通讯对象是从网络上下载的image图片对象
- (void)downloadImage
{
NSURL *url = [NSURL URLWithString:@"http://g.hiphotos.baidu.com/image/pic/item/4afbfbedab64034f06bd8359acc379310a551d78.jpg"];
NSData *data = [NSData dataWithContentsOfURL:url];
UIImage *image = [UIImage imageWithData:data];
// 更新UI都要在主线程执行
[self performSelectorOnMainThread:@selector(setImg:) withObject:image waitUntilDone:NO];
NSLog(@"downloadImage - %@",[NSThread currentThread]);
}
在setImg:方法里面让图片显示
- (void)setImg:(UIImage *)img
{
self.imgV.image = img;
// 设置UIImageView根据image自动调节frame的大小
[self.imgV sizeToFit];
// 设置scrollView滚动范围
self.scrollView.contentSize = img.size;
NSLog(@"setImg - %@",[NSThread currentThread]);
}
代理方法:
/**
* 该方法返回需要缩放的view
*/
- (UIView *)viewForZoomingInScrollView:(UIScrollView *)scrollView
{
return self.imgV;
}
如此这般就简单实现了线程间的通讯,程序会从网络上下载图片显示,控制台打印的结果如下
number = 1 代表在主线程上执行。
number = 2(只要number!=1)代表在子线程上执行。
耗时操作在后台子线程执行,更新UI在主线程执行。
为何在后台子线程执行,如果在主线程执行耗时操作,那么该操作执行期间,主线程一直被占用,用户的操作就无法响应,这样的后果可想而知。
为什么更新UI要在主线程执行?在子线程中是不能进行UI 更新的,而可以更新的结果只是一个幻像:因为子线程代码执行完毕了,又自动进入到了主线程,执行了子线程中的UI更新的函数栈,这中间的时间非常的短,就让大家误以为分线程可以更新UI。如果子线程一直在运行,则子线程中的UI更新的函数栈 主线程无法获知,即无法更新。只有极少数的UI能更新,因为开辟线程时会获取当前环境,如点击某个按钮,这个按钮响应的方法是开辟一个子线程,在子线程中对该按钮进行UI
更新是能及时的,如换标题,换背景图,但这没有任何意义。