ios 网络编程总结-多线程

一:确认网络环境3G/WIFI  
 
    1. 添加源文件和framework  
      
    开发Web等网络应用程序的时候,需要确认网络环境,连接情况等信息。如果没有处理它们,是不会通过Apple的审查的。  
    Apple 的 例程 Reachability 中介绍了取得/检测网络状态的方法。要在应用程序程序中使用Reachability,首先要完成如下两部:  
      
    1.1. 添加源文件:  
    在你的程序中使用 Reachability 只须将该例程中的 Reachability.h 和 Reachability.m 拷贝到你的工程中。
    1.2.添加framework:  
    将SystemConfiguration.framework 添加进工程。
    2. 网络状态  
    Reachability.h中定义了三种网络状态:  
    typedef enum {  
        NotReachable = 0,            //无连接  
        ReachableViaWiFi,            //使用3G/GPRS网络  
        ReachableViaWWAN            //使用WiFi网络  
    } NetworkStatus;  
      
    因此可以这样检查网络状态:  
 
    Reachability *r = [Reachability reachabilityWithHostName:@“ www.apple.com ”];  
    switch ([r currentReachabilityStatus]) {  
            case NotReachable:  
                    // 没有网络连接  
                    break;  
            case ReachableViaWWAN:  
                    // 使用3G网络  
                    break;  
            case ReachableViaWiFi:  
                    // 使用WiFi网络  
                    break;  
    }  
      
    3.检查当前网络环境  
    程序启动时,如果想检测可用的网络环境,可以像这样  
    // 是否wifi  
    + (BOOL) IsEnableWIFI {  
        return ([[Reachability reachabilityForLocalWiFi] currentReachabilityStatus] != NotReachable);  
    }  
 
    // 是否3G  
    + (BOOL) IsEnable3G {  
        return ([[Reachability reachabilityForInternetConnection] currentReachabilityStatus] != NotReachable);  

    } 

在视图即将呈现的时候检查网络环境。

    例子:  
    - (void)viewWillAppear:(BOOL)animated {      
    if (([Reachability reachabilityForInternetConnection].currentReachabilityStatus == NotReachable) &&   
            ([Reachability reachabilityForLocalWiFi].currentReachabilityStatus == NotReachable)) {  
            self.navigationItem.hidesBackButton = YES;  
            [self.navigationItem setLeftBarButtonItem:nil animated:NO];  
        }  
    }  
 
    4. 链接状态的实时通知  
    网络连接状态的实时检查,通知在网络应用中也是十分必要的。接续状态发生变化时,需要及时地通知用户:  
      
    Reachability 1.5版本  
    // My.AppDelegate.h  
    #import "Reachability.h"  
 
    @interface MyAppDelegate : NSObject <UIApplicationDelegate> {  
        NetworkStatus remoteHostStatus;  
    }  
 
    @property NetworkStatus remoteHostStatus;  
 
    @end  
 
    // My.AppDelegate.m  
    #import "MyAppDelegate.h"  
 
    @implementation MyAppDelegate  
    @synthesize remoteHostStatus;  
 
    // 更新网络状态  
    - (void)updateStatus {  
        self.remoteHostStatus = [[Reachability sharedReachability] remoteHostStatus];  
    }  
 
    // 通知网络状态  
    - (void)reachabilityChanged:(NSNotification *)note {  
        [self updateStatus];  
        if (self.remoteHostStatus == NotReachable) {  
            UIAlertView *alert = [[UIAlertView alloc] initWithTitle:NSLocalizedString(@"AppName", nil)  
                         message:NSLocalizedString (@"NotReachable", nil)  
                        delegate:nil cancelButtonTitle:@"OK" otherButtonTitles: nil];  
            [alert show];  
            [alert release];  
        }  
    }  
 
    // 程序启动器,启动网络监视  
    - (void)applicationDidFinishLaunching:(UIApplication *)application {  
      
        // 设置网络检测的站点  
        [[Reachability sharedReachability] setHostName:@"www.apple.com"];  
        [[Reachability sharedReachability] setNetworkStatusNotificationsEnabled:YES];  
        // 设置网络状态变化时的通知函数  
        [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(reachabilityChanged:)  
                                                 name:@"kNetworkReachabilityChangedNotification" object:nil];  
        [self updateStatus];  
    }  
 
    - (void)dealloc {  
        // 删除通知对象  
        [[NSNotificationCenter defaultCenter] removeObserver:self];  
        [window release];  
        [super dealloc];  
    }   
      
    Reachability 2.0版本  
      
 
    // MyAppDelegate.h  
    @class Reachability;  
 
        @interface MyAppDelegate : NSObject <UIApplicationDelegate> {  
            Reachability  *hostReach;  
        }  
 
    @end  
 
    // MyAppDelegate.m  
    - (void)reachabilityChanged:(NSNotification *)note {  
        Reachability* curReach = [note object];  
        NSParameterAssert([curReach isKindOfClass: [Reachability class]]);  
        NetworkStatus status = [curReach currentReachabilityStatus];  
      
        if (status == NotReachable) {  
            UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"AppName""  
                              message:@"NotReachable"  
                              delegate:nil  
                              cancelButtonTitle:@"YES" otherButtonTitles:nil];  
                              [alert show];  
                              [alert release];  
        }  
    }  
                                
    - (void)applicationDidFinishLaunching:(UIApplication *)application {  
        // ...  
                    
        // 监测网络情况  
        [[NSNotificationCenter defaultCenter] addObserver:self  
                              selector:@selector(reachabilityChanged:)  
                              name: kReachabilityChangedNotification  
                              object: nil];  
        hostReach = [[Reachability reachabilityWithHostName:@"www.google.com"] retain];  
        hostReach startNotifer];  
        // ...  
    }  
 
 
二:使用NSConnection下载数据  
      
    1.创建NSConnection对象,设置委托对象  
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:[self urlString]]];  
    [NSURLConnection connectionWithRequest:request delegate:self];  
      
    2. NSURLConnection delegate委托方法  
        - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response;    
        - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error;    
        - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data;    
        - (void)connectionDidFinishLoading:(NSURLConnection *)connection;    
 
    3. 实现委托方法  
    - (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response {  
        // store data  
        [self.receivedData setLength:0];            //通常在这里先清空接受数据的缓存  
    }  
      
    - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {  
           /* appends the new data to the received data */  
        [self.receivedData appendData:data];        //可能多次收到数据,把新的数据添加在现有数据最后  
    }  
 
    - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {  
        // 错误处理  
    }  
 
    - (void)connectionDidFinishLoading:(NSURLConnection *)connection {  
        // disconnect  
        [UIApplication sharedApplication].networkActivityIndicatorVisible = NO;     
        NSString *returnString = [[NSString alloc] initWithData:self.receivedData encoding:NSUTF8StringEncoding];  
        NSLog(returnString);  
        [self urlLoaded:[self urlString] data:self.receivedData];  
        firstTimeDownloaded = YES;  
    }  
 
三:使用NSXMLParser解析xml文件  
 
    1. 设置委托对象,开始解析  
    NSXMLParser *parser = [[NSXMLParser alloc] initWithData:data];   //或者也可以使用initWithContentsOfURL直接下载文件,但是有一个原因不这么做:  
    // It's also possible to have NSXMLParser download the data, by passing it a URL, but this is not desirable  
    // because it gives less control over the network, particularly in responding to connection errors.  
    [parser setDelegate:self];  
    [parser parse];  
 
    2. 常用的委托方法  
    - (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName   
                                namespaceURI:(NSString *)namespaceURI  
                                qualifiedName:(NSString *)qName   
                                attributes:(NSDictionary *)attributeDict;  
    - (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName   
                                namespaceURI:(NSString *)namespaceURI   
                                qualifiedName:(NSString *)qName;  
    - (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string;  
    - (void)parser:(NSXMLParser *)parser parseErrorOccurred:(NSError *)parseError;  
 
    static NSString *feedURLString = @"http://www.yifeiyang.net/test/test.xml";  
 
    3.  应用举例  
    - (void)parseXMLFileAtURL:(NSURL *)URL parseError:(NSError **)error  
    {  
        NSXMLParser *parser = [[NSXMLParser alloc] initWithContentsOfURL:URL];  
        [parser setDelegate:self];  
        [parser setShouldProcessNamespaces:NO];  
        [parser setShouldReportNamespacePrefixes:NO];  
        [parser setShouldResolveExternalEntities:NO];  
        [parser parse];  
        NSError *parseError = [parser parserError];  
        if (parseError && error) {  
            *error = parseError;  
        }  
        [parser release];  
    }  
 
    - (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI   
                                        qualifiedName:(NSString*)qName attributes:(NSDictionary *)attributeDict{  
        // 元素开始句柄  
        if (qName) {  
            elementName = qName;  
        }  
        if ([elementName isEqualToString:@"user"]) {  
            // 输出属性值  
            NSLog(@"Name is %@ , Age is %@", [attributeDict objectForKey:@"name"], [attributeDict objectForKey:@"age"]);  
        }  
    }  
 
    - (void)parser:(NSXMLParser *)parser didEndElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI   
                                        qualifiedName:(NSString *)qName  
    {  
        // 元素终了句柄  
        if (qName) {  
               elementName = qName;  
        }  
    }  
 
    - (void)parser:(NSXMLParser *)parser foundCharacters:(NSString *)string  
    {  
        // 取得元素的text  
    }  
 
    NSError *parseError = nil;  

    [self parseXMLFileAtURL:[NSURL URLWithString:feedURLString] parseError:&parseError];


使用NSOperation和NSOperationQueue启动多线程
在app store中的很多应用程序非常的笨重,他们有好的界面,但操作性很差,比如说当程序从网上或本地载入数据的时候,界面被冻结了,用户只能等程序完全载入数据之后才能进行操作。 
当打开一个应用程序时,iphone会产生一个包含main方法的线程,所用程序中的界面都是运行在这个线程之中的(table views, tab bars, alerts…),有时候我们会用数据填充这些view,现在问        题是如何有效的载入数据,并且用户还能自如的操作程序。方法是启动新的线程,专门用于数据的下载,而主线程不会因为下载数据被阻塞。 
不管使用任何编程语言,在实现多线程时都是一件很麻烦的事情。更糟糕的是,一旦出错,这种错误通常相当糟糕。然而,幸运的是apple从os x10.5在这方面做了很多的改进,NSThread的引入,使得开发多线程应用程序容易多了。除此之外,它们还引入了两个全新的类,NSOperation和NSOperationQueue。 
接下来我们通过一个实例来剖析如何使用这两个类实现多线程。这里指示展示这两个类的基本用法,当然这不是使用他们的唯一办法。 
如果你熟悉java或者它的别的变种语言的话 ,你会发现NSOperation对象很像java.lang.Runnable接口,就像java.lang.Runnable接口那样,NSOperation类也被设计为可扩展的,而且只有一个需要重写的方法。它就是-(void)main。使用NSOperation的最简单的方式就是把一个NSOperation对象加入到NSOperationQueue队列中,一旦这个对象被加入到队列,队列就开始处理这个对象,直到这个对象的所有操作完成。然后它被队列释放。 
下面的例子中,使用一个获取网页,并对其解析程NSXMLDocument,最后将解析得到的NSXMLDocument返回给主线程。 
     
PageLoadOperation.h@interface PageLoadOperation : NSOperation { 
    NSURL *targetURL;} 
@property(retain) NSURL *targetURL; 
- (id)initWithURL:(NSURL*)url;@end 
 
PageLoadOperation.m 
#import "PageLoadOperation.h"#import "AppDelegate.h"@implementation PageLoadOperation@synthesize targetURL;- (id)initWithURL:(NSURL*)url;{ 
    if (![super init]) return nil; 
    [self setTargetURL:url]; 
    return self;}- (void)dealloc { 
    [targetURL release], targetURL = nil; 
    [super dealloc]; 

- (void)main  

    NSString *webpageString = [[[NSString alloc] 
    initWithContentsOfURL:[self targetURL]] autorelease]; 
    NSError *error = nil; 
    NSXMLDocument *document = [[NSXMLDocument alloc] 
    initWithXMLString:webpageString  
    options:NSXMLDocumentTidyHTML error:&error]; 
    if (!document) { 
        NSLog(@"%s Error loading document (%@): %@",  
        _cmd, [[self targetURL] absoluteString], error); 
         return; 
    } 
    [[AppDelegate shared] 
    performSelectorOnMainThread:@selector(pageLoaded:) 
         withObject:document waitUntilDone:YES]; 
    [document release]; 

@end 
    正如我们所看到的那样,这个类相当的简单,在它的init方法中接受一个url并保存起来,当main函数被调用的时候,它使用这个保存的url创建一个字符串,并将这个字符串传递给NSXMLDocumentinit方法。如果加载的xml数据没有出错,数据会被传递给AppDelegate,它处于主线程中。到此,这个线程的任务就完成了。在主线程中注销操作队列的时候,会将这个NSOperation对象释放。 
AppDelegate.h 
@interface AppDelegate : NSObject { 
    NSOperationQueue *queue; 
}
+ (id)shared;
- (void)pageLoaded:(NSXMLDocument*)document;
@end
AppDelegate.m        
#import "AppDelegate.h"
#import "PageLoadOperation.h"
@implementation AppDelegate 
static AppDelegate *shared; 
static NSArray *urlArray; 
- (id)init 

    if (shared) 
    { 
        [self autorelease]; 
        return shared; 
    } 
    if (![super init]) return nil;   
 NSMutableArray *array = [[NSMutableArray alloc] init];
[array addObject:@"http://www.google.com"];
[array addObject:@"http://www.apple.com"];
[array addObject:@"http://www.yahoo.com"];
[array addObject:@"http://www.zarrastudios.com"];
[array addObject:@"http://www.macosxhints.com"];
urlArray = array;    
queue = [[NSOperationQueue alloc] init];
shared = self;return self; 
    } 
    •    (void)applicationDidFinishLaunching: 
    (NSNotification *)aNotification 

        for (NSString *urlString in urlArray)  
        { 
        NSURL *url =  
        [NSURL URLWithString:urlString];        PageLoadOperation *plo =  
        [[PageLoadOperation alloc] initWithURL:url]; 
        [queue addOperation:plo]; 
        [plo release]; 
        } 

- (void)dealloc 

        [queue release], queue = nil; 
        [super dealloc]; 

+ (id)shared; 

        if (!shared) { 
            [[AppDelegate alloc] init]; 
        } 
        return shared; 

- (void)pageLoaded:(NSXMLDocument*)document; 

        NSLog(@"%s Do something with the XMLDocument: %@", 
             _cmd, document); 

@end 
 
NSOperationQueue的并行控制(NSOperationQueue Concurrency) 
        在上面这个简单的例子中,我们很难看出这些操作是并行运行的,然而,如果你你的操作花费的时间远远比这里的要长,你将会发现,队列是同时执行这些操作的。幸运的是,如果你想要为队列限制同时只能运行几个操作,你可以使用NSOperationQueue的setMaxConcurrentOperationCount:方法。例如,[queue setMaxConcurrentOperationCount:2]; 


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
网络编程中,多线程编程是一种常用的技术,可以提高程序的并发性和性能。下面是一些关于多线程编程的常用方法和注意事项: 1. NSThread:NSThread是iOS中最底层的线程类,它可以通过类方法或实例方法来创建线程。使用NSThread可以设置线程的名称、优先级,以及控制线程的睡眠和退出等操作。 2. 线程调度:在多线程编程中,多个线程会并发运行,但线程的执行顺序是由CPU调度器决定的,程序员无法控制。多个线程会同时竞争CPU资源,谁先抢到资源谁就先执行,所以多线程的执行顺序是随机的。 3. 多线程的创建:在iOS开发中,常用的多线程编程方式有三种:NSThread、GCD和NSOperation。NSThread是最底层的线程类,可以直接操作线程的各种属性和方法。GCD(Grand Central Dispatch)提供了一种高效的并发编程模型,可以通过队列来管理任务的执行。NSOperation是基于GCD的更高层次的封装,提供了更多的控制和管理线程的功能。 4. 线程的创建顺序:在多线程编程中,并不能保证哪个线程会先运行,即无法确定新创建的线程或调用线程哪个会先执行。新创建的线程可以访问进程的地址空间,并继承调用线程的浮点环境和信号屏蔽字,但挂起信号集会被清除。 总结来说,多线程编程是一种提高程序并发性和性能的技术,在网络编程中尤为重要。通过使用NSThread、GCD或NSOperation等方法,可以实现多线程的创建和管理。然而,程序员无法控制线程的执行顺序,因为线程的调度是由CPU调度器决定的。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* [IOS多线程基础(OC)](https://blog.csdn.net/yong_19930826/article/details/105857055)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *3* [UNIX环境高级编程笔记](https://blog.csdn.net/w_x_myself/article/details/128613534)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值