安哥网络 发表于 2013-10-6 22:28:18

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 = www.apple.com”];
switch () {
      case NotReachable:
          // 没有网络连接
          break;
      case ReachableViaWWAN:
          // 使用3G网络
          break;
      case ReachableViaWiFi:
          // 使用WiFi网络
          break;
}

3.检查当前网络环境
程序启动时,如果想检测可用的网络环境,可以像这样
// 是否wifi
+ (BOOL) IsEnableWIFI {
    return ([ currentReachabilityStatus] != NotReachable);
}

// 是否3G
+ (BOOL) IsEnable3G {
    return ([ currentReachabilityStatus] != NotReachable);
}
例子:
- (void)viewWillAppear:(BOOL)animated {
if ((.currentReachabilityStatus == NotReachable) &&
      (.currentReachabilityStatus == NotReachable)) {
      self.navigationItem.hidesBackButton = YES;
      ;
    }
}

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 = [ remoteHostStatus];
}

// 通知网络状态
- (void)reachabilityChanged:(NSNotification *)note {
    ;
    if (self.remoteHostStatus == NotReachable) {
      UIAlertView *alert = [ initWithTitle:NSLocalizedString(@"AppName", nil)
             message:NSLocalizedString (@"NotReachable", nil)
            delegate:nil cancelButtonTitle:@"OK" otherButtonTitles: nil];
      ;
      ;
    }
}

// 程序启动器,启动网络监视
- (void)applicationDidFinishLaunching:(UIApplication *)application {

    // 设置网络检测的站点
    [ setHostName:@"www.apple.com"];
    [ setNetworkStatusNotificationsEnabled:YES];
    // 设置网络状态变化时的通知函数
    [ addObserver:self selector:@selector(reachabilityChanged:)
                         name:@"kNetworkReachabilityChangedNotification" object:nil];
    ;
}

- (void)dealloc {
    // 删除通知对象
    [ removeObserver:self];
    ;
    ;
}

Reachability 2.0版本


// MyAppDelegate.h
@class Reachability;

    @interface MyAppDelegate : NSObject <UIApplicationDelegate> {
      Reachability *hostReach;
    }

@end

// MyAppDelegate.m
- (void)reachabilityChanged:(NSNotification *)note {
    Reachability* curReach = ;
    NSParameterAssert(]);
    NetworkStatus status = ;

    if (status == NotReachable) {
      UIAlertView *alert = [ initWithTitle:@"AppName""
               message:@"NotReachable"
               delegate:nil
               cancelButtonTitle:@"YES" otherButtonTitles:nil];
               ;
               ;
    }
}

- (void)applicationDidFinishLaunching:(UIApplication *)application {
    // ...

    // 监测网络情况
    [ addObserver:self
               selector:@selector(reachabilityChanged:)
               name: kReachabilityChangedNotification
               object: nil];
    hostReach = [ retain];
    hostReach startNotifer];
    // ...
}


二:使用NSConnection下载数据

1.创建NSConnection对象,设置委托对象

NSMutableURLRequest *request = ]];
;

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
    ;      //通常在这里先清空接受数据的缓存
}

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data {

    ;    //可能多次收到数据,把新的数据添加在现有数据最后
}

- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error {
    // 错误处理
}

- (void)connectionDidFinishLoading:(NSURLConnection *)connection {
    // disconnect
    .networkActivityIndicatorVisible = NO;
    NSString *returnString = [ initWithData:self.receivedData encoding:NSUTF8StringEncoding];
    NSLog(returnString);
    data:self.receivedData];
    firstTimeDownloaded = YES;
}

三:使用NSXMLParser解析xml文件

1. 设置委托对象,开始解析
NSXMLParser *parser = [ 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.
;
;

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 = [ initWithContentsOfURL:URL];
    ;
    ;
    ;
    ;
    ;
    NSError *parseError = ;
    if (parseError && error) {
      *error = parseError;
    }
    ;
}

- (void)parser:(NSXMLParser *)parser didStartElement:(NSString *)elementName namespaceURI:(NSString *)namespaceURI
                  qualifiedName:(NSString*)qName attributes:(NSDictionary *)attributeDict{
    // 元素开始句柄
    if (qName) {
      elementName = qName;
    }
    if () {
      // 输出属性值
      NSLog(@"Name is %@ , Age is %@", , );
    }
}

- (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;
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 (!) return nil;
;
return self;}- (void)dealloc {
, targetURL = nil;
;
}
- (void)main
{
NSString *webpageString = [[
initWithContentsOfURL:] autorelease];
NSError *error = nil;
NSXMLDocument *document = [
initWithXMLString:webpageString
options:NSXMLDocumentTidyHTML error:&error];
if (!document) {
    NSLog(@"%s Error loading document (%@): %@",
    _cmd, [ absoluteString], error);
   return;
}
[
performSelectorOnMainThread:@selector(pageLoaded:)
   withObject:document waitUntilDone:YES];
;
}
@end
正如我们所看到的那样,这个类相当的简单,在它的init方法中接受一个url并保存起来,当main函数被调用的时候,它使用这个保存的url创建一个字符串,并将这个字符串传递给NSXMLDocumentinit方法。如果加载的xml数据没有出错,数据会被传递给AppDelegate,它处于主线程中。到此,这个线程的任务就完成了。在主线程中注销操作队列的时候,会将这个NSOperation对象释放。
AppDelegate.h
@interface AppDelegate : NSObject {
NSOperationQueue *queue;
}+ (id)shared;- (void)pageLoaded:(NSXMLDocument*)document;@endAppDelegate.m    #import "AppDelegate.h"#import "PageLoadOperation.h"@implementation AppDelegate
static AppDelegate *shared;
static NSArray *urlArray;
- (id)init
{
if (shared)
{
    ;
    return shared;
}
if (!) return nil;NSMutableArray *array = [ init];;;;;;urlArray = array;queue = [ init];shared = self;return self;
}
•(void)applicationDidFinishLaunching:
(NSNotification *)aNotification
{
    for (NSString *urlString in urlArray)
    {
    NSURL *url =
    ;    PageLoadOperation *plo =
    [ initWithURL:url];
    ;
    ;
    }
}
- (void)dealloc
{
    , queue = nil;
    ;
}
+ (id)shared;
{
    if (!shared) {
      [ init];
    }
    return shared;
}
- (void)pageLoaded:(NSXMLDocument*)document;
{
    NSLog(@"%s Do something with the XMLDocument: %@",
       _cmd, document);
}
@end

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

本文摘自:http://flandycheng.blog.51cto.com/855176/789625


页: [1]
查看完整版本: IOS 网络编程总结 (转)