iOS开发多线程篇 - NSOperation(下)

简介: iOS开发多线程篇 - NSOperation

代码示例:

//
//  ViewController.m
//  TestNSOperationQueue
//
//  Created by taobaichi on 2017/3/21.
//  Copyright © 2017年 MaChao. All rights reserved.
//
#import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    //创建NSInvocationOperation对象,封装操作
    NSInvocationOperation * operation1 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(testOperation1) object:nil];
    NSInvocationOperation * operation2 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(testOperation2) object:nil];
    //创建对象,封装操作
    NSBlockOperation * operation3 = [NSBlockOperation blockOperationWithBlock:^{
        NSLog(@"-----NSBlockOperation--3--1---- %@",[NSThread currentThread]);
    }];
    [operation3 addExecutionBlock:^{
        NSLog(@"-----NSBlockOperation--3--2---- %@",[NSThread currentThread]);
    }];
    //创建NSOperationQueue
    NSOperationQueue * queue = [[NSOperationQueue alloc]init];
    //把操作添加到队列中
    [queue addOperation:operation1];
    [queue addOperation:operation2];
    [queue addOperation:operation3];
}
-(void)testOperation1
{
    NSLog(@"-----NSInvocationOperation--1---- %@",[NSThread currentThread]);
}
-(void)testOperation2
{
    NSLog(@"-----NSInvocationOperation--2---- %@",[NSThread currentThread]);
}
@end


打印效果:

2017-03-21 13:36:39.594 TestNSOperationQueue[5185:159392] -----NSInvocationOperation--2---- <NSThread: 0x60800007b480>{number = 4, name = (null)}
2017-03-21 13:36:39.594 TestNSOperationQueue[5185:159391] -----NSInvocationOperation--1---- <NSThread: 0x600000269f40>{number = 3, name = (null)}
2017-03-21 13:36:39.594 TestNSOperationQueue[5185:159394] -----NSBlockOperation--3--1---- <NSThread: 0x60000026a0c0>{number = 5, name = (null)}
2017-03-21 13:36:39.594 TestNSOperationQueue[5185:159414] -----NSBlockOperation--3--2---- <NSThread: 0x608000266240>{number = 6, name = (null)}


注意:系统自动将NSOperationQueue中NSOperation对象取出,将其封装的操作放到一条新的线程中执行,上面的代码示例一共有四个任务,operation1operation2分别有一个任务,operation3有2个任务。一共四个任务,开启了四条线程,通过任务执行的时间全部是2017-03-21 13:36:39.594可以看出,这些任务是并行执行的


提示:队列的取出是有顺序的,与打印结果并不矛盾。这就好比,选手A,B,C虽然起跑的顺序是先A,后B,然后C,但是到达终点的顺序却不一定是A,B在前,C在后。


下面使用for循环打印,可以更明显的看出任务是并行执行的


代码示例:

//
//  ViewController.m
//  TestNSOperationQueue
//
//  Created by taobaichi on 2017/3/21.
//  Copyright © 2017年 MaChao. All rights reserved.
//
#import "ViewController.h"
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    //创建NSInvocationOperation对象,封装操作
    NSInvocationOperation * operation1 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(testOperation1) object:nil];
    NSInvocationOperation * operation2 = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(testOperation2) object:nil];
    //创建对象,封装操作
    NSBlockOperation * operation3 = [NSBlockOperation blockOperationWithBlock:^{
        for (int i = 0; i < 5; i++) {
        NSLog(@"-----NSBlockOperation----3--1---- %@",[NSThread currentThread]);
        }
    }];
    [operation3 addExecutionBlock:^{
        for (int i = 0; i < 5; i++) {
        NSLog(@"-----NSBlockOperation----3--2---- %@",[NSThread currentThread]);
        }
    }];
    //创建NSOperationQueue
    NSOperationQueue * queue = [[NSOperationQueue alloc]init];
    //把操作添加到队列中
    [queue addOperation:operation1];
    [queue addOperation:operation2];
    [queue addOperation:operation3];
}
-(void)testOperation1
{
    for (int i = 0; i < 5; i++) {
    NSLog(@"-----NSInvocationOperation--1---- %@",[NSThread currentThread]);
    }
}
-(void)testOperation2
{
    for (int i = 0; i < 5; i++) {
     NSLog(@"-----NSInvocationOperation--2---- %@",[NSThread currentThread]);
    }
}
@end


打印结果:

2017-03-21 13:51:44.091 TestNSOperationQueue[5519:170579] -----NSBlockOperation----3--1---- <NSThread: 0x608000074f80>{number = 5, name = (null)}
2017-03-21 13:51:44.091 TestNSOperationQueue[5519:170582] -----NSInvocationOperation--2---- <NSThread: 0x608000075140>{number = 4, name = (null)}
2017-03-21 13:51:44.090 TestNSOperationQueue[5519:170580] -----NSInvocationOperation--1---- <NSThread: 0x60800006f200>{number = 3, name = (null)}
2017-03-21 13:51:44.091 TestNSOperationQueue[5519:170644] -----NSBlockOperation----3--2---- <NSThread: 0x6080000752c0>{number = 6, name = (null)}
2017-03-21 13:51:44.094 TestNSOperationQueue[5519:170579] -----NSBlockOperation----3--1---- <NSThread: 0x608000074f80>{number = 5, name = (null)}
2017-03-21 13:51:44.096 TestNSOperationQueue[5519:170580] -----NSInvocationOperation--1---- <NSThread: 0x60800006f200>{number = 3, name = (null)}
2017-03-21 13:51:44.096 TestNSOperationQueue[5519:170582] -----NSInvocationOperation--2---- <NSThread: 0x608000075140>{number = 4, name = (null)}
2017-03-21 13:51:44.099 TestNSOperationQueue[5519:170644] -----NSBlockOperation----3--2---- <NSThread: 0x6080000752c0>{number = 6, name = (null)}
2017-03-21 13:51:44.100 TestNSOperationQueue[5519:170579] -----NSBlockOperation----3--1---- <NSThread: 0x608000074f80>{number = 5, name = (null)}
2017-03-21 13:51:44.101 TestNSOperationQueue[5519:170580] -----NSInvocationOperation--1---- <NSThread: 0x60800006f200>{number = 3, name = (null)}
2017-03-21 13:51:44.103 TestNSOperationQueue[5519:170582] -----NSInvocationOperation--2---- <NSThread: 0x608000075140>{number = 4, name = (null)}
2017-03-21 13:51:44.103 TestNSOperationQueue[5519:170644] -----NSBlockOperation----3--2---- <NSThread: 0x6080000752c0>{number = 6, name = (null)}
2017-03-21 13:51:44.105 TestNSOperationQueue[5519:170579] -----NSBlockOperation----3--1---- <NSThread: 0x608000074f80>{number = 5, name = (null)}
2017-03-21 13:51:44.105 TestNSOperationQueue[5519:170580] -----NSInvocationOperation--1---- <NSThread: 0x60800006f200>{number = 3, name = (null)}
2017-03-21 13:51:44.106 TestNSOperationQueue[5519:170582] -----NSInvocationOperation--2---- <NSThread: 0x608000075140>{number = 4, name = (null)}
2017-03-21 13:51:44.108 TestNSOperationQueue[5519:170644] -----NSBlockOperation----3--2---- <NSThread: 0x6080000752c0>{number = 6, name = (null)}
2017-03-21 13:51:44.112 TestNSOperationQueue[5519:170579] -----NSBlockOperation----3--1---- <NSThread: 0x608000074f80>{number = 5, name = (null)}
2017-03-21 13:51:44.113 TestNSOperationQueue[5519:170580] -----NSInvocationOperation--1---- <NSThread: 0x60800006f200>{number = 3, name = (null)}
2017-03-21 13:51:44.117 TestNSOperationQueue[5519:170582] -----NSInvocationOperation--2---- <NSThread: 0x608000075140>{number = 4, name = (null)}
2017-03-21 13:51:44.119 TestNSOperationQueue[5519:170644] -----NSBlockOperation----3--2---- <NSThread: 0x6080000752c0>{number = 6, name = (null)}


实战: 实现一个下载图片的例子

//
//  ViewController.m
//  TestNSOperationQueue
//
//  Created by taobaichi on 2017/3/21.
//  Copyright © 2017年 MaChao. All rights reserved.
//
#import "ViewController.h"
#define kURL @"http://avatar.csdn.net/2/C/D/1_totogo2010.jpg"
@interface ViewController ()
@property (nonatomic, strong) UIImageView * imagView;
@end
@implementation ViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    self.imagView = [[UIImageView alloc]initWithFrame:CGRectMake(self.view.frame.size.width/2 - 50, self.view.frame.size.height/2 - 50, 100, 100)];
    self.imagView.backgroundColor = [UIColor lightGrayColor];
    [self.view addSubview:self.imagView];
    //创建一个后台线程,后台线程执行downloadImage方法
    NSInvocationOperation * operation = [[NSInvocationOperation alloc]initWithTarget:self selector:@selector(downloadImage:) object:kURL];
    //创建NSOperationQueue
    NSOperationQueue * queue = [[NSOperationQueue alloc]init];
    //把创建的后台线程放到NSOperationQueue中
    [queue addOperation:operation];
}
-(void)downloadImage:(NSString *)url
{
    NSLog(@"url: %@",url);
    NSURL * nsUrl = [NSURL URLWithString:url];
    NSData * data = [[NSData alloc]initWithContentsOfURL:nsUrl];
    UIImage * image = [[UIImage alloc]initWithData:data];
    //下载完成后执行主线程updateUI方法
    [self performSelectorOnMainThread:@selector(updateUI:) withObject:image waitUntilDone:YES];
}
-(void)updateUI:(UIImage *)image
{
    self.imagView.image = image;
}
@end


代码注释:


  1. viewDidLoad方法里可以看到我们用NSInvocationOperation建了一个后台线程,并且放到NSOperationQueue中。后台线程执行downloadImage方法。


  1. downloadImage 方法处理下载图片的逻辑。下载完成后用performSelectorOnMainThread执行主线程updateUI方法。


  1. updateUI 并把下载的图片显示到图片控件中。


运行后可以看到下载图片显示在界面上


image.png


Simulator Screen Shot 2017年3月21日 14.29.48.png



相关文章
|
13天前
|
人工智能 程序员 API
iOS|记一名 iOS 开发新手的前两次 App 审核经历
啥,这玩意也有新手保护期?
20 0
|
24天前
|
JavaScript 搜索推荐 Android开发
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
51 8
【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
|
2月前
|
iOS开发 开发者
uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
163 67
uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
|
3月前
|
存储 监控 API
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
app开发之安卓Android+苹果ios打包所有权限对应解释列表【长期更新】-以及默认打包自动添加权限列表和简化后的基本打包权限列表以uniapp为例-优雅草央千澈
122 11
|
3月前
|
iOS开发 开发者 MacOS
深入探索iOS开发中的SwiftUI框架
【10月更文挑战第21天】 本文将带领读者深入了解Apple最新推出的SwiftUI框架,这一革命性的用户界面构建工具为iOS开发者提供了一种声明式、高效且直观的方式来创建复杂的用户界面。通过分析SwiftUI的核心概念、主要特性以及在实际项目中的应用示例,我们将展示如何利用SwiftUI简化UI代码,提高开发效率,并保持应用程序的高性能和响应性。无论你是iOS开发的新手还是有经验的开发者,本文都将为你提供宝贵的见解和实用的指导。
159 66
|
3月前
|
开发框架 Android开发 iOS开发
安卓与iOS开发中的跨平台策略:一次编码,多平台部署
在移动应用开发的广阔天地中,安卓和iOS两大阵营各占一方。随着技术的发展,跨平台开发框架应运而生,它们承诺着“一次编码,到处运行”的便捷。本文将深入探讨跨平台开发的现状、挑战以及未来趋势,同时通过代码示例揭示跨平台工具的实际运用。
190 3
|
3月前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
4月前
|
存储 前端开发 Swift
探索iOS开发:从新手到专家的旅程
本文将带您领略iOS开发的奇妙之旅,从基础概念的理解到高级技巧的掌握,逐步深入iOS的世界。文章不仅分享技术知识,还鼓励读者在编程之路上保持好奇心和创新精神,实现个人成长与技术突破。
|
4月前
|
API Android开发 iOS开发
深入探索Android与iOS的多线程编程差异
在移动应用开发领域,多线程编程是提高应用性能和响应性的关键。本文将对比分析Android和iOS两大平台在多线程处理上的不同实现机制,探讨它们各自的优势与局限性,并通过实例展示如何在这两个平台上进行有效的多线程编程。通过深入了解这些差异,开发者可以更好地选择适合自己项目需求的技术和策略,从而优化应用的性能和用户体验。
|
4月前
|
安全 IDE Swift
探索iOS开发之旅:从初学者到专家
在这篇文章中,我们将一起踏上iOS开发的旅程,从基础概念的理解到深入掌握核心技术。无论你是编程新手还是希望提升技能的开发者,这里都有你需要的指南和启示。我们将通过实际案例和代码示例,展示如何构建一个功能齐全的iOS应用。准备好了吗?让我们一起开始吧!

热门文章

最新文章

  • 1
    iOS|解决 setBrightness 调节屏幕亮度不生效的问题
    113
  • 2
    iOS|记一名 iOS 开发新手的前两次 App 审核经历
    20
  • 3
    iOS各个证书生成细节
    29
  • 4
    【01】噩梦终结flutter配安卓android鸿蒙harmonyOS 以及next调试环境配鸿蒙和ios真机调试环境-flutter项目安卓环境配置-gradle-agp-ndkVersion模拟器运行真机测试环境-本地环境搭建-如何快速搭建android本地运行环境-优雅草卓伊凡-很多人在这步就被难倒了
    151
  • 5
    Cellebrite UFED 4PC 7.71 (Windows) - Android 和 iOS 移动设备取证软件
    50
  • 6
    【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    66
  • 7
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    52
  • 8
    【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
    51
  • 9
    uniapp开发ios打包Error code = -5000 Error message: Error: certificate file(p12) import failed!报错问题如何解决
    163
  • 10
    【05】2025年1月首发完整版-篇幅较长-苹果app如何上架到app store完整流程·不借助第三方上架工具的情况下无需花钱但需仔细学习-优雅草央千澈详解关于APP签名以及分发-们最关心的一篇来了-IOS上架app
    352