程序员人生 网站导航

iOS网络编程

栏目:综合技术时间:2015-03-31 08:34:24

         ios网络编程(http、socket)

分类: OS-X/iOS开发 7369人浏览 评论(3) 收藏 举报

目录(?)[+]

http编程综述:亦可称为soap编程。通常情况下,http编程要比socket编程相对要简单易用很多。所以用的最广广泛。

1、http编程其实就是http要求。http要求最长用的方法是 get 和 post 方法。
==》get方法和post方法相比理解起来比较简单,get方法可以直接要求1个url,也能够url后面拼接上参数作为1个新的url地址进行要求。get方法后面的value要经过unicode编码。form的enctype属性默许为application/x-www-form-urlencoded。不能发送2进制文件。
==》post方法相对要复杂1些。首先post方法要设置key和value ,所有的key和value都会拼接成 key1=value1&key2=value2的样式的字符串,然后这个字符串转化为2进制放到 http要求的body中。当要求发送的时候,也就跟随body1起传给服务器。http要求Content-Type设置为:application/x-www-form-urlencoded。这里讲的只是简单的post要求,1般发送文件不会选择这类方式(从技术方面斟酌也能够发送文件,就是把文件以 key 和 value的方式放入)。下面我们再讨论1下post发送2进制文件更加普遍的方法。

2、HTTP协议是甚么?
简单来讲,就是1个基于利用层的通讯规范:双方要进行通讯,大家都要遵照1个规范,这个规范就是HTTP协议。
HTTP协议能做甚么?
很多人首先1定会想到:阅读网页。没错,阅读网页是HTTP的主要利用,但是这其实不代表HTTP就只能利用于网页的阅读。HTTP是1种协议,只要通讯的双方都遵照这个协议,HTTP就可以有用武之地。比如我们经常使用的QQ,迅雷这些软件,都会使用HTTP协议(还包括其他的协议)。
HTTP协议如何工作?
大家都知道1般的通讯流程:首先客户端发送1个要求(request)给服务器服务器在接收到这个要求后将生成1个响应(response)返回给客户端。
在这个通讯的进程中HTTP协议在以下4个方面做了规定:
1.         Request和Response的格式()
2.         建立连接的方式(1、非持久连接 2、持久连接)
3.         缓存的机制
4.         响应授权激起机制
(利用场合)
5.        基于HTTP的利用(1、 HTTP代理 2、多线程下载 3、 HTTPS传输协议原理 4、开发web程序经常用的Request Methods 5、用户与服务器的交互)


经常使用cocoa内部类: 
1,Reachability.h 苹果demo支持网络连接诊断
2,NSConnection连接、NSMutableURLRequest URL网址的要求封装包
3,NSXMLParser XML解析

经常使用第3方库:
1,ASIHttprequest 库

操作步骤:
1:检查网络环境(3G/WIFI)
2:发起NSConnection要求
3:处理返回xml数据包(NSXMLParser解析xml文件),或返回文件png、pdf之类
4:若返回数据包中含有待下载的图片下载地址。则重复2、3步骤下载下来图片。只不过3中返回的是文件。

socket编程综述:
经常使用cocoa内部类:
经常使用第3方库:1,Asyncsocket库
操作步骤:



http网络编程实例   

1:确认网络环境3G/WIFI 
 
    1. 添加源文件和framework 
     
    开发Web等网络利用程序的时候,需要确认网络环境,连接情况等信息。如果没有处理它们,是不会通过Apple的审查的。 
    Apple 的 例程 Reachability 中介绍了获得/检测网络状态的方法。要在利用程序程序中使用Reachability,首先要完成以下两部: 
     
    1.1. 添加源文件: 
    在你的程序中使用 Reachability 只须将该例程中的 Reachability.h 和 Reachability.m 拷贝到你的工程中。以下图: 
 
     
     
    1.2.添加framework: 
    将SystemConfiguration.framework 添加进工程。以下图: 
     
     
    2. 网络状态 
     
    Reachability.h中定义了3种网络状态: 
    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. 链接状态的实时通知 
    网络连接状态的实时检查,通知在网络利用中也是10分必要的。接续状态产生变化时,需要及时地通知用户: 
     
    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] setNetworkStatusNotificati*****Enabled: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]; 
        // ... 
    } 
 
 
2:使用NSConnection下载数据 
     
    1.创建NSConnection对象,设置拜托对象 
     
    NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:[self urlString]]]; 
    [NSURLConnection connectionWithRequest:request delegate:self]; 
     
    2. NSURLConnection delegate拜托方法 
        - (void)connection:(NSURLConnection *)connection didReceiveResp*****e:(NSURLResp*****e *)resp*****e;   
        - (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error;   
        - (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data;   
        - (void)connectionDidFinishLoading:(NSURLConnection *)connection;   
 
    3. 实现拜托方法 
    - (void)connection:(NSURLConnection *)connection didReceiveResp*****e:(NSURLResp*****e *)resp*****e { 
        // 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; 
    } 
 
3:使用NSXMLParser解析xml文件 
 
    1. 设置拜托对象,开始解析 
    NSXMLParser *parser = [[NSXMLParser alloc] initWithData:data];   //或也能够使用initWithContentsOfURL直接下载文件,但是有1个缘由不这么做: 
    // 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]; 

//实例
//
//  NLViewController.m
//  NetWorkTest
//
//  Created by Nono on 12⑸⑴6.
//  Copyright (c) 2012年 NonoWithLilith. All rights reserved.
//
#import "NLViewController.h"

@interface NLViewController ()

@end

@implementation NLViewController
@synthesize label = _label;
@synthesize data = _data;
@synthesize connection = _connection;
- (void)dealloc{
    [self.label release];
    [self.data release];
    [super dealloc];
}
- (void)viewDidLoad
{
    [super viewDidLoad];
    UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(10.0, 10.0, 300.0, 400)];
    self.label = label;
    label.textAlignment = UITextAlignmentCenter;
    [label setNumberOfLines:0];
    label.lineBreakMode = UILineBreakModeWordWrap; 
    self.label.text = @"正在在要求数据";
    [self.view addSubview:label];
    [label release];
    //step 1:要求地址
    NSString *urlString = @"http://www.google.com";
    NSURL *url = [NSURL URLWithString:urlString];
    //step 2:实例化1个request
    NSURLRequest *requrst = [NSURLRequest requestWithURL:url cachePolicy:NSURLRequestUseProtocolCachePolicy timeoutInterval:30.0];
    //step 3:创建链接
    self.connection = [[NSURLConnection alloc] initWithRequest:requrst delegate:self];
    if ( self.connection) {
        NSLog(@"链接成功");
    }else {
        NSLog(@"链接失败");
    }
    
    [url release];
    [urlString release];
    [requrst release];
// Do any additional setup after loading the view, typically from a nib.
}

- (void)viewDidUnload
{
    self.label = nil;
    self.data = nil;
    [super viewDidUnload];
    // Release any retained subviews of the main view.
}

- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
{
    return (interfaceOrientation != UIInterfaceOrientationPortraitUpsideDown);
}

#pragma mark-
#pragma NSUrlConnectionDelegate methods
- (void)connection:(NSURLConnection *)connection didReceiveResponse:(NSURLResponse *)response
{
    //接受1个服务端回话,再次1般初始化接受数据的对象
   
    NSLog(@"返回数据类型:%@",[response textEncodingName]); 
    NSMutableData *d = [[NSMutableData alloc] init];
     self.data = d;
    [d release];
}

- (void)connection:(NSURLConnection *)connection didReceiveData:(NSData *)data
{
    //接受返回数据,这个方法可能会被调用屡次,因此将屡次返回数据加起来
    
    NSUInteger datalength = [data length];
    NSLog(@"返回数据量:%d",datalength);
    [self.data appendData:data];
}

- (void)connectionDidFinishLoading:(NSURLConnection *)connection
{
    //连接结束
    
    NSLog(@"%d:",[self.data length]);
    NSStringEncoding enc = CFStringConvertEncodingToNSStringEncoding(kCFStringEncodingGB_18030_2000);
    NSString *mystr = [[NSString alloc] initWithData:_data encoding:enc];
   // string i
    NSLog(@"最后的结果:%@",mystr);
    self.label.text = mystr;
    [mystr release];
    [self.connection release];
}

- (void)connection:(NSURLConnection *)connection didFailWithError:(NSError *)error
{
    //链接毛病
}

@end
------分隔线----------------------------
------分隔线----------------------------

最新技术推荐