ios多线程操作(三)—— 线程通讯

本文介绍如何在iOS应用中实现线程间通讯,并通过代理方法控制UIScrollView和UIImageView的操作,确保图片下载与显示过程流畅无阻塞。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

     一个进程中,线程并非单独存在,往往需要与其他线程进行通讯以执行特定的任务。接下来就用一个简单的例子来实现线程之间最简单的通讯,并借此探究一下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 更新是能及时的,如换标题,换背景图,但这没有任何意义。













评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值