iphone开发之多线程NSThread和NSInvocationOperation

简介:

以下是开发初期收集整理的一点资料

多线程之NSInvocationOperation 
多线程编程是防止主线程堵塞,增加运行效率等等的最佳方法。而原始的多线程方法存在很多的毛病,包括线程锁死等。在Cocoa中,Apple提供了NSOperation这个类,提供了一个优秀的多线程编程方法。 
本次介绍NSOperation的子集,简易方法的NSInvocationOperation: 
@implementation MyCustomClass 
- (void)launchTaskWithData:(id)data 

    //创建一个NSInvocationOperation对象,并初始化到方法 
    //在这里,selector参数后的值是你想在另外一个线程中运行的方法(函数,Method) 
    //在这里,object后的值是想传递给前面方法的数据 
    NSInvocationOperation* theOp = [[NSInvocationOperation alloc] initWithTarget:self 
                    selector:@selector(myTaskMethod:) object:data]; 
  
    // 下面将我们建立的操作“Operation”加入到本地程序的共享队列中(加入后方法就会立刻被执行) 
    // 更多的时候是由我们自己建立“操作”队列 
    [[MyAppDelegate sharedOperationQueue] addOperation:theOp]; 

  
// 这个是真正运行在另外一个线程的“方法” 
- (void)myTaskMethod:(id)data 

    // Perform the task. 

  
@end一个NSOperationQueue 操作队列,就相当于一个线程管理器,而非一个线程。因为你可以设置这个线程管理器内可以并行运行的的线程数量等等。下面是建立并初始化一个操作队列:

@interface MyViewController : UIViewController { 
  
    NSOperationQueue *operationQueue; 
    //在头文件中声明该队列 

@end 
  
@implementation MyViewController 
  
- (id)init  

    self = [super init]; 
    if (self) { 
        operationQueue = [[NSOperationQueue alloc] init]; //初始化操作队列 
        [operationQueue setMaxConcurrentOperationCount:1]; 
        //在这里限定了该队列只同时运行一个线程 
        //这个队列已经可以使用了 
    } 
    return self; 

  
- (void)dealloc 

    [operationQueue release]; 
    //正如Alan经常说的,我们是程序的好公民,需要释放内存! 
    [super dealloc]; 

  
@end简单介绍之后,其实可以发现这种方法是非常简单的。很多的时候我们使用多线程仅仅是为了防止主线程堵塞,而NSInvocationOperation就是最简单的多线程编程,在iPhone编程中是经常被用到的。



/////////////////////////////////////////////////////////////////////////////////////////////////// 
1 在主线程里加入一个loading画面…… 
2 { 
3 [window addSubview:view_loading]; 
4 [NSThread detachNewThreadSelector:@selector(init_backup:) toTarget:self withObject:nil]; 
5 } 
可以通过performSelectorOhMainThread更新UI元素,比如设置进度条等等。最后消除loading画面,载入主View。 
7 - (void)init_backup:(id)sender 
8 { 
9 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init]; 
10 
11 // ... 
12 int i = status; 
13 [self performSelectorOnMainThread:@selector(show_loading:) withObject:[NSNumber numberWithInt:i] waitUntil Done:NO]; 
14 
15 [view_loading removeFromSuperview]; 
16 [window addSubview:tabcontroller_main.view]; 
17 [pool release]; 
18 }

 

///////////////////////////////////////////////////////

利用iphone的多线程实现和线程同步 

从接口的定义中可以知道,NSThread和大多数iphone的接口对象一样,有两种方式可以初始化:

一种使用initWithTarget :(id)target selector:(SEL)selector object:(id)argument,但需要负责在对象的retain count为0时调用对象的release方法清理对象。

另一种则使用所谓的convenient method,这个方便接口就是detachNewThreadSelector,这个方法可以直接生成一个线程并启动它,而且无需为线程的清理负责。

#import <UIKit/UIKit.h> 
@interface SellTicketsAppDelegate : NSObject <UIApplicationDelegate> { 
    int tickets; 
    int count; 
    NSThread* ticketsThreadone; 
    NSThread* ticketsThreadtwo; 
    NSCondition* ticketsCondition; 
    UIWindow *window; 
}

@property (nonatomic, retain) IBOutlet UIWindow *window; 
@end

 

然后在实现中添加如下代码: 
//  SellTicketsAppDelegate.m 
//  SellTickets 
// 
//  Created by sun dfsun2009 on 09-11-10. 
//  Copyright __MyCompanyName__ 2009. All rights reserved. 
// 
#import "SellTicketsAppDelegate.h" 
@implementation SellTicketsAppDelegate 
@synthesize window; 
- (void)applicationDidFinishLaunching:(UIApplication *)application { 
    tickets = 100; 
    count = 0; 
    // 锁对象 
    ticketCondition = [[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]; 
    //[NSThread detachNewThreadSelector:@selector(run) toTarget:self withObject:nil]; 
    // Override point for customization after application launch 
    [window makeKeyAndVisible]; 

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

-         (void)dealloc { 
    [ticketsThreadone release]; 
    [ticketsThreadtwo release]; 
    [ticketsCondition release]; 
    [window release]; 
    [super dealloc]; 

@end

------------------------------------------------------------------------------------- 
// 定义 
#import <UIKit/UIKit.h>

@interface ThreadSyncSampleViewController : UIViewController { 
int _threadCount; 
NSCondition *_myCondition; 
}

@end

//实现文件如下:

#import "ThreadSyncSampleViewController.h"

@implementation ThreadSyncSampleViewController

 

/* 
// The designated initializer. Override to perform setup that is required before the view is loaded. 
- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil { 
    if (self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil]) { 
        // Custom initialization 
    } 
    return self; 

*/

/* 
// Implement loadView to create a view hierarchy programmatically, without using a nib. 
- (void)loadView { 

*/

 

// Implement viewDidLoad to do additional setup after loading the view, typically from a nib. 
- (void)viewDidLoad { 
    [super viewDidLoad]; 
// 
//_myCondition = nil; 
// 
_myCondition = [[NSCondition alloc] init]; 
// 
NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:30 
             target:self 
              selector:@selector(threadTester) 
              userInfo:nil 
             repeats:YES]; 
[timer fire]; 
  
}


- (void)threadTester{ 
[_myCondition lock]; 
  
_threadCount = -2; 
//如果有n个要等待的thread,这里置成 -n 
[_myCondition unlock]; 
// 
NSLog(@""); 
NSLog(@"------------------------------------------------------------------------------"); 
[NSThread detachNewThreadSelector:@selector(threadOne) toTarget:self withObject:nil]; 
[NSThread detachNewThreadSelector:@selector(threadTwo) toTarget:self withObject:nil]; 
[NSThread detachNewThreadSelector:@selector(threadThree) toTarget:self withObject:nil]; 
return; 
}

- (void)threadOne{ 
NSLog(@"@@@ In thread 111111 start."); 
[_myCondition lock]; 
  
int n = rand()%5 + 1; 
NSLog(@"@@@ Thread 111111 Will sleep %d seconds ,now _threadCount is : %d",n,_threadCount); 
sleep(n); 
//[NSThread sleepForTimeInterval:n]; 
_threadCount ++ ; 
NSLog(@"@@@ Thread 111111 has sleep %d seconds ,now _threadCount is : %d",n,_threadCount); 
[_myCondition signal]; 
NSLog(@"@@@ Thread 1111111 has signaled ,now _threadCount is : %d",_threadCount); 
[_myCondition unlock]; 
NSLog(@"@@@ In thread one complete."); 
[NSThread exit]; 
return; 
}

- (void)threadTwo{ 
NSLog(@"### In thread 2222222 start."); 
[_myCondition lock]; 
  
int n = rand()%5 + 1; 
NSLog(@"### Thread 2222222 Will sleep %d seconds ,now _threadCount is : %d",n,_threadCount); 
sleep(n); 
//   [NSThread sleepForTimeInterval:n]; 
_threadCount ++ ; 
NSLog(@"### Thread 2222222 has sleep %d seconds ,now _threadCount is : %d",n,_threadCount); 
[_myCondition signal]; 
NSLog(@"### Thread 2222222 has signaled ,now _threadCount is : %d",_threadCount); 
[_myCondition unlock]; 
//_threadCount ++ ; 
NSLog(@"### In thread 2222222 complete."); 
[NSThread exit]; 
return; 
}

- (void)threadThree{ 
NSLog(@"&lt;&lt;&lt; In thread 333333 start."); 
[_myCondition lock]; 
while (_threadCount &lt; 0) { 
  [_myCondition wait]; 

NSLog(@"&lt;&lt;&lt; In thread 333333 ,_threadCount now is %d ,will start work.",_threadCount); 
[_myCondition unlock]; 
NSLog(@"&lt;&lt;&lt; In thread 333333 complete."); 
[NSThread exit]; 
return; 
}

/* 
// Override to allow orientations other than the default portrait orientation. 
- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation { 
    // Return YES for supported orientations 
    return (interfaceOrientation == UIInterfaceOrientationPortrait); 

*/

- (void)didReceiveMemoryWarning { 
// Releases the view if it doesn't have a superview. 
    [super didReceiveMemoryWarning]; 
  
// Release any cached data, images, etc that aren't in use. 
}

- (void)viewDidUnload { 
// Release any retained subviews of the main view. 
// e.g. self.myOutlet = nil; 
}


- (void)dealloc { 
[_myCondition release]; 
    [super dealloc]; 
}

@end










本文转自 arthurchen 51CTO博客,原文链接:http://blog.51cto.com/arthurchen/577907,如需转载请自行联系原作者
目录
相关文章
|
19天前
|
编解码 测试技术 iOS开发
iPhone 屏幕尺寸和开发适配
【10月更文挑战第23天】iPhone 的屏幕尺寸变化给开发者带来了一定的挑战,但也为创新提供了机遇。通过深入了解不同屏幕尺寸的特点,遵循适配原则和策略,运用合适的技巧和方法,我们能够为用户提供在不同 iPhone 机型上都具有良好体验的应用。在未来,随着技术的不断进步,我们还需要持续学习和适应,以满足用户对优质应用体验的不断追求。
|
19天前
|
编解码 iOS开发 UED
响应式设计在 iPhone 开发适配中的具体应用
【10月更文挑战第23天】响应式设计在 iPhone 开发适配中扮演着至关重要的角色,它能够帮助我们打造出适应不同屏幕尺寸和用户需求的高质量应用。通过合理运用响应式设计的原则和方法,我们可以在提供良好用户体验的同时,提高开发效率和应用的可维护性。
|
6月前
|
安全 Java 编译器
深入理解PHP 8.0新特性及其对开发的影响Java中的多线程编程:从理论到实践
【5月更文挑战第27天】在这篇文章中,我们将详细探讨PHP 8.0的新特性以及它们如何影响开发者的工作流程。我们将深入研究这些新特性,包括JIT编译器,联合类型,命名参数,以及更多的错误处理机制。我们还将讨论这些新特性如何提高代码的可读性和性能,以及它们如何改变我们编写和维护PHP应用程序的方式。 【5月更文挑战第27天】在现代计算机科学中,多线程编程是一个重要的概念,它允许多个线程在同一时间内运行,从而提高了程序的效率和性能。本文将深入探讨Java中的多线程编程,包括其理论基础,实现方法,以及一些常见的问题和解决方案。我们将通过实例来理解如何创建和管理线程,以及如何使用Java的并发工具来
|
2月前
|
API Android开发 iOS开发
安卓与iOS开发中的线程管理对比
【9月更文挑战第12天】在移动应用的世界中,安卓和iOS平台各自拥有庞大的用户群体。开发者们在这两个平台上构建应用时,线程管理是他们必须面对的关键挑战之一。本文将深入探讨两大平台在线程管理方面的异同,通过直观的代码示例,揭示它们各自的设计理念和实现方式,帮助读者更好地理解如何在安卓与iOS开发中高效地处理多线程任务。
|
3月前
|
安全 测试技术 调度
iOS开发-多线程编程
【8月更文挑战第12天】在iOS开发中,属性的内存管理至关重要,直接影响应用性能与稳定性。主要策略包括:`strong`(强引用),保持对象不被释放;`weak`(弱引用),不保持对象,有助于避免循环引用;`assign`(赋值),适用于基本数据类型及非指针对象类型;`copy`(复制),复制对象而非引用,确保不变性。内存管理基于引用计数,利用自动引用计数(ARC)自动管理对象生命周期。此外,需注意避免循环引用,特别是在block中。最佳实践包括理解各策略、避免不必要的强引用、及时释放不再使用的对象、注意block中的内存管理,并使用工具进行内存分析。正确管理内存能显著提升应用质量。
|
3月前
|
安全 网络安全 数据安全/隐私保护
网络安全与信息安全:关于网络安全漏洞、加密技术、安全意识等方面的知识分享安卓与iOS开发中的线程管理比较
【8月更文挑战第30天】本文将探讨网络安全与信息安全的重要性,并分享关于网络安全漏洞、加密技术和安全意识的知识。我们将了解常见的网络攻击类型和防御策略,以及如何通过加密技术和提高安全意识来保护个人和组织的信息安全。
|
3月前
|
数据采集 iOS开发 Python
Chatgpt教你开发iPhone风格计算器,Python代码实现
Chatgpt教你开发iPhone风格计算器,Python代码实现
|
4月前
|
缓存 编译器 Go
开发与运维线程问题之Go语言的goroutine基于线程模型实现如何解决
开发与运维线程问题之Go语言的goroutine基于线程模型实现如何解决
55 3
|
4月前
|
算法 编译器 C++
开发与运维线程问题之在C++的原子操作中memory_order如何解决
开发与运维线程问题之在C++的原子操作中memory_order如何解决
43 2
|
4月前
|
Java 运维
开发与运维命令问题之使用jstack命令查看Java进程的线程栈如何解决
开发与运维命令问题之使用jstack命令查看Java进程的线程栈如何解决
65 2