本文复制、参考自文章:iOS多线程编程之NSThread的使用  ,主要为了加强个人对知识的理解和记忆,不做他用。原作者声明:

著作权声明:本文由http://blog.csdn.net/totogo2010/原创,欢迎转载分享。请尊重作者劳动,转载时保留该声明和作者博客链接,谢谢!

这里对原作者的辛勤工作表示感谢!

1. 简介

1.1 iOS的多线程编程技术分类

(1)NSThread

(2) Cocoa NSOperation

(3) GCD (Grand Central Dispatch)

这三种方式从上到下,抽象层次逐渐增高,使用也越来越简单。

1.2 三种方式的优缺点

  优点   缺点
NSThread 轻量 需要自己管理线程的生命周期,线程同步。线程同步加锁时,会有一定的系统开销。
NSOperation   无需关心线程管理,数据同步,可以把精力放在自己需要的执行操作上  
GCD iOS4.0后出现,以替代NSThread,NSOperation等技术的,很高效、强大  

2. NSThread的使用

2.1 创建方式

(1) 实例方法创建

- (id) initWithTarget:(id)target selector:(SEL)selector object:(id) argument
示例:
NSThread* myThread = [[NSThread alloc] initWithTarget:self
selector:@selector(doSomething:)
object:nil];
[myThread start];

(2) NSThread 类方法创建

+ (void)detachNewThreadSelector:(SEL) aSelector toTarget:(id)aTarget withObject:(id)anArgument

示例:
[NSThread detachNewThreadSelector:@selector(doSomething:) toTarget:self withObject:nil];

(3)NSObject 非显示方法创建

+ (void) performSelectorInBackground:(SEL)aSelector withObject:(id)anArgument
示例
[Obj performSelectorInBackground:@selector(doSomething:)withObject:nil];

selector: 线程执行的方法,这个selector只能有一个参数,而且不能有返回值;

target:    selector消息发送的对象

argument:传输给selector的唯一参数,也可以是nil

第一种方式会直接创建线程并且开始运行线程,第二种方法是先创建线程对象,然后再运行线程操作,在运行线程操作前可以设置线程的优先级等线程信息。

2.2 使用示例1: 异步加载图片

(1)在viewDidLoad中创建子线程:

- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
NSThread* thread = [[NSThread alloc] initWithTarget:self selector:@selector(downloadImage:) object:kURL]; // 子线程执行方法 downloadImage
[thread start]; // 启用子线程
}

(2)在线程执行方法 downloadImage 中执行加载数据操作,并使用performSelectorOnMainThread方法 通知主线程进行渲染操作。(线程调回主线程,并传递了数据)

- (void)downloadImage:(NSString *) url
{
NSData* data = [[NSData alloc] initWithContentsOfURL:[NSURL URLWithString:url]];
UIImage* image = [[UIImage alloc] initWithData:data];
if(image == nil)
{
NSLog(@"image load failed...");
}
else
{
[self performSelectorOnMainThread:@selector(updateUI:) withObject:image waitUntilDone:YES]; // 与主线程通信,传递了image数据
}
}

同样,还可以使用:

- (void)performSelector:(SEL)aSelector  onThread:(NSThread *)thread   withObject:(id)arg    waitUntilDone:(BOOL)wait    modes:(NSArray *)array

方法来于其他线程通信。

2.3 使用示例2: 多线程卖票

(1) 定义public的数据结构,AppDelegate.h中

@interface AppDelegate : UIResponder <UIApplicationDelegate>
{
int tickets;
int count;
NSThread* ticketsThreadOne;
NSThread* ticketsThreadTwo;
NSCondition* ticketsCondition;
NSLock* theLock;
}

这里测试使用了NSCondition和NSLock两种加锁方法。

(2)初始化, AppDelegate.m中

    tickets = ;
count = ;
theLock = [[NSLock alloc] init];
// 锁对象
ticketsCondition = [[NSCondition alloc] init];
ticketsThreadOne = [[NSThread alloc] initWithTarget:self selector:@selector(run) object:nil];
[ticketsThreadOne setName:@"Thread-1"];
[ticketsThreadOne start]; ticketsThreadTwo = [[NSThread alloc] initWithTarget:self selector:@selector(run) object:nil];
[ticketsThreadTwo setName:@"Thread-2"];
[ticketsThreadTwo start];

(3)执行操作中,加锁,防止资源抢占

- (void)run
{
while (TRUE)
{
// 上锁
// [theLock lock];
[ticketsCondition lock];
if (tickets >= )
{
[NSThread sleepForTimeInterval:0.09];
count = -tickets;
NSLog(@"当前票数是:%d, 售出:%d, 线程名:%@", tickets, count, [[NSThread currentThread] name]);
tickets--;
}
else
{
break;
}
[ticketsCondition unlock];
// [theLock unlock];
}
}

NSCondition 可以使用 [ticketsCondition wait];等待,并由其他线程使用[ticketsCondition signal];唤起等待。

2.4 使用@synchoronized

我们还可以使用@synchoronized来简化NSLock的使用,这样就不必显示创建和调用NSLock对象,而自动创建了一个互斥锁(mutex lock),防止对资源的抢占。 如下

    obj1 = [[NSObject alloc]init]; // 两个线程,分别传递两个对象用于原操作标识
obj2 = [[NSObject alloc]init]; ticketsThreadOne = [[NSThread alloc] initWithTarget:self selector:@selector(run:) object:obj1];
[ticketsThreadOne setName:@"Thread-1"];
[ticketsThreadOne start]; ticketsThreadTwo = [[NSThread alloc] initWithTarget:self selector:@selector(run:) object:obj2];
[ticketsThreadTwo setName:@"Thread-2"];
[ticketsThreadTwo start];
- (void)run:(NSObject*) obj
{
@synchronized(obj)
{
while (TRUE)
{
// 上锁
// [theLock lock];
// [ticketsCondition lock];
int currentTicketNum = [ticketCounter currentTicketNum];
if ( currentTicketNum >= )
{
[NSThread sleepForTimeInterval:0.09];
NSLog(@"当前票数是:%d, 售出:%d, 线程名:%@", currentTicketNum, ticketCounter.initTicketNum - currentTicketNum, [[NSThread currentThread] name]);
ticketCounter.currentTicketNum--;
}
else
{
break;
}
// [ticketsCondition unlock];
// [theLock unlock];
}
}
}

iOS 多线程学习笔记 —— NSThread的更多相关文章

  1. iOS 多线程学习笔记 —— GCD

    本文复制.参考自文章:iOS多线程编程之Grand Central Dispatch(GCD)介绍和使用 ,主要为了加强个人对知识的理解和记忆,不做他用.原作者声明: 著作权声明:本文由http:// ...

  2. iOS 多线程学习笔记 —— NSOperation

    本文复制.参考自文章:iOS多线程编程之NSOperation和NSOperationQueue的使用 ,主要为了加强个人对知识的理解和记忆,不做他用.原作者声明: 著作权声明:本文由http://b ...

  3. java多线程学习笔记——详细

    一.线程类  1.新建状态(New):新创建了一个线程对象.        2.就绪状态(Runnable):线程对象创建后,其他线程调用了该对象的start()方法.该状态的线程位于可运行线程池中, ...

  4. JAVA多线程学习笔记(1)

    JAVA多线程学习笔记(1) 由于笔者使用markdown格式书写,后续copy到blog可能存在格式不美观的问题,本文的.mk文件已经上传到个人的github,会进行同步更新.github传送门 一 ...

  5. 多线程学习笔记九之ThreadLocal

    目录 多线程学习笔记九之ThreadLocal 简介 类结构 源码分析 ThreadLocalMap set(T value) get() remove() 为什么ThreadLocalMap的键是W ...

  6. java进阶-多线程学习笔记

    多线程学习笔记 1.什么是线程 操作系统中 打开一个程序就是一个进程 一个进程可以创建多个线程 现在系统中 系统调度的最小单元是线程 2.多线程有什么用? 发挥多核CPU的优势 如果使用多线程 将计算 ...

  7. Java多线程学习笔记(一)——多线程实现和安全问题

    1. 线程.进程.多线程: 进程是正在执行的程序,线程是进程中的代码执行,多线程就是在一个进程中有多个线程同时执行不同的任务,就像QQ,既可以开视频,又可以同时打字聊天. 2.线程的特点: 1.运行任 ...

  8. iOS多线程编程之NSThread的使用

      目录(?)[-] 简介 iOS有三种多线程编程的技术分别是 三种方式的有缺点介绍 NSThread的使用 NSThread 有两种直接创建方式 参数的意义 PS不显式创建线程的方法 下载图片的例子 ...

  9. iOS多线程编程之NSThread的使用(转)

    本文由http://blog.csdn.net/totogo2010/原创 1.简介: 1.1 iOS有三种多线程编程的技术,分别是: 1..NSThread 2.Cocoa NSOperation  ...

随机推荐

  1. mvc5 + ef6 + autofac搭建项目(repository+uow)(二)

    续上篇: DBContext 在上篇 图一类库根目录创建的 DbContextBase /// <summary> /// 数据库上下文基类 /// </summary> // ...

  2. C#调用cmd程序,读取结果

    示例,调用cmd执行PING命令,读取结果,代码如下: using System; using System.Collections.Generic; using System.Linq; using ...

  3. 6 关于 Oracle NULL栏位和PL./SQL执行实验

    今日有针对NULL值有了相关实验. 对NULL 值插入的讨论. 1, PL/SQL 中可以执行插入''或者NULL 的操作, 前提是栏位允许为空. 2, 可以对NULL进行一系列数据库运算. 如:   ...

  4. C语言中short的意思

    short和int等一样,是C或C++的一种内部数据类型.用于表示有符号整数.不同的是,他们在内存中所占的空间大小不同,short通常为int所占一半,也有一些实现为和int一样,但不会比int大.所 ...

  5. QT UI 使一个QWidget里面的元素自动填充满本QWidget

    使一个QWidget里面的元素自动填充满本QWidget: 对象查看器,右键点击本QWidget,选择"布局",为此QWidget增加一个布局. 如果该QWidget只有一个对象, ...

  6. C与C++的区别

    C++与C的区别 1. 动态分配内存 1)C语言 a. malloc函数:在内存的动态存储区中分配一个长度为size的连续空间:       void *malloc(unsigned int siz ...

  7. 读书笔记之 - javascript 设计模式 - 适配器模式

    适配器模式可以用来在现在接口和不兼容的类之间进行适配. 使用这种模式的对象又叫包装器,因为他们是在用一个新接口包装另一个对象. 在设计类的时候往往遇到有些接口不能与现有api一同使用的情况,借助于适配 ...

  8. PHP使用DES进行加密解密

    DES是一种对称加密算法,也就是通过密文和合法的密钥能够将明文还原出来,在程序开发过程中有些 接口可能需要获取原始数据,而发送的数据又比较敏感(比如用户的密码等信息),这时可以选择DES加密算法,DE ...

  9. Android2.2 API —— ImageView

    注意 请查看本文后期更新完整版: http://www.cnblogs.com/over140/archive/2011/06/08/2075054.html 来源: 农民伯伯: http://www ...

  10. xadmin的插件机制

    xadmin的视图方法中如果加了@filter_hook 标记的都可以作为插件的钩子函数. 例如在ListAdminView类中有许多加了上述标记的方法, @filter_hook def get_c ...