iphone包含了很多框架和库,从底层的套接字到不同层次的封装,可以方便地给程序添加网络功能。
(1)BSD套接字。最底层的套接字,这是Unix网络开发常用的API。如果从其他系统移植程序,而程序用的是BSD套接字,那么网络部分可以继续使用这些API。
(2)CFNetwork framework 。CFNetwork 也是比较底层的, 是对BSD套接字的一个扩展 。它是一个C语言的库,它是基于BSD套接字,提供了对网络协议的抽象。这些抽象使得用户更容易地操作套接字、处理网络的各种连接。。它集成了run-loop,因此使用CFNetwork不用自己去实现事件循环。CFNetwork 还包括了一些网络协议(如HTTP、FTP)的实现,可以在不了解这些协议细节的情况下直接使用。
(3) Foundation framework是基于Objectice-C语言的库,它为CFNetwork API 提供了面向对象的抽象。
(4)CFURL(C)/NSURL(Objective-C)是比较高层的API,它们提供了从Web和FTP服务器下载文件或其他资源的一种简单的方法。其中,CFURL是CFNetwork的一部分,NSURL是Foundation的一部分。
(5)CFNetServices/NSNetServices提供了使用Bonjour注册和发现网络服务的方法。
总之,CFNetwork framework 和 Foundation framework是最为强大的两个库,它们是比较底层、高效,提供的接口也非常丰富。
1.CFSocket
CFSocket是对BSD套接字的一个抽象封装,它提供了BSD套接字的几乎全部功能,还集成了 run-loop。CFSocket可以处理任何类型的套接字,而不仅仅限于流式套接字。
(1)创建CFSocket
常用的方法是CFSocketCreate 和CFSocketCreateWithNative。
CFSocketCreate 方法声明如下:
CFSocketRef CFSocketCreate (
CFAllocatorRef allocator, //指定创建新套接字的内存分配器类型,传入NULL或kCFAllocatorDefault可以使用默认的分配器。一般默认即可。
SInt32 protocolFamily, //指定套接字的协议族。默认使用PF_INET,也就是平时说的IPV4。指定PF_INET6以使用IPV6协议。
SInt32 socketType, //套接字类型,SOCK_STREAM或SOCK_DGRAM。
SInt32 protocol, //套接字使用的协议,IPPROTO_TCP或IPPROTO_UDP。此项需要与套接字类型一致,设为0取默认值(如果socketType为SOCK_STREAM,此项默认值为IPPROTO_TCP,否则为IPPROTO_UDP)。
CFOptionFlags callBackTypes, //CFSocket提供的run-loop可以在特定件发生时回调指定的函数。这个参数使用了位掩码,因此可以使用按位或运算指定多个类型。苹果公司将一些类型定义为枚举值如下所示:
CFSocketCallBack callout,//指定回调函数,当指定的事件类型中的一个发生时函数被调用。这样,不用自己写循环等待连接、发送数据了。
const CFSocketContext *context //存储与套接字相关信息的数据结构,里面可以包含自定义数据。函数会将里面的内容拷出来,因此函数调用完后参数指向的内存不必再保留。可以为NULL。
);
CFSocketCreateWithNative方法,可以使用一个已经存在的BSD套接字来创建CRSocket,
CFSocketRef CFSocketCreateWithNative
{
CFAllocatorRef allocator,
CFSocketNativeHandle sock,
CFOptionFlags callBackTypes,
CFSocketCallBack callout,
const CFSocketContext *context
};
2.套接字函数
创建好CFSocket后,就可以使用它提供的一系列函数了。通过CFSocketNative函数,还可以操作更底层的BSD套接字。以下几个函数式CFSocket中常用的。
(1)CFSocketGetNative 返回系统套接字,通常是int类型的。获得这个套接字,可以使用Unix系统原生的套接字操作。调用setsockopt函数。
(2) CFSocketConnectToAddress 用于将套接字连接到一个正在监听的套接字(服务器)。
(3)CFSocketCopyAddress 返回CFSocket的地址,可以知道套接字正在哪个IP地址上监听。
(4)CFSocketCopyPeerAddress 获得CFSocket连接到的远程套接字的地址。
(5)CFSocketCreateRunLoopSource 为CFSocket创建一个CFRunLoop。
(6)CFSocketSendData 发送数据,需要传入CFSocket、地址(NULL则发送到套接字已经连接到的地址)、要发送的数据(CFDataRef类型)、超时时间。返回值有kCFSocketSuccess、kCFSocketError和kCFSocketTimeout 3种。
3.回调函数
每个回调函数可以获得以下信息,根据类型不同,一些信息也会不一样。
(1)CFSocketRefs 事件对应的CFSocket,这样可以区分不同套接字产生的事件。
(2)CFSocketCallBackType callbackType 标识哪一种事件发生了。
(3)CFDataRef address 包含底层sockaddr信息的CFData指针。可以通过这个参数获得的套接字连接到的远程地址。只有在kCFSocketAcceptCallBack和kCFSocketDataCallBack事件发生时才会提供这个参数。
(4)const void *data 根据回调类型指向不同的数据。如果是kCFSocketDataCallBack类型,这个参数为CFDataRef类型,包含接收到的数据;如果是kCFSocketAcceptCallBack类型,这个参数为指向CFSocketNativeHandle的指针;如果为kCFSocketConnectCallBack,连接在后台失败了,这个参数是指向SInt32类型错误代码的一个指针。其他情况这个参数为NULL。
(5)void *info CFSocketContext结构体中的info成员,是自定义数据。CFSocketContext是在创建CFSocket时被指定的。
4.CFSocketContext
它是包含自定义数据及一些回调的结构体。 因为CFSocket通过run-loop异步通知发生的事件,当有很多连接的时候,如何保持对每个连接相关数据的跟踪变得困难起来。CFSocketContext允许为套接字绑定任何类型的数据,每个回调函数都可以得到这个数据。