iOS Socket

大纲

一.Socket简介

二.BSD Socket编程准备

1.地址

2.端口

3.网络字节序

4.半相关与全相关

5.网络编程模型

三.socket接口编程示例

四.使用select

五.使用kqueue

六.使用流

一.Socket简介

在UNIX系统中,万物皆文件(Everything is a file)。所有的IO操作都可以看作对文件的IO操作,都遵循着这样的操作模式:打开 -> 读/写 -> 关闭,打开操作(如open函数)获取“文件”使用权,返回文件描述符,后继的操作都通过这个文件描述符来进行。很多系统调用都依赖于文件描述符,它是一个无符号整数,每一个用户进程都对应着一个文件描述符表,通过文件描述符就可以找到对应文件的信息。 在类UNIX平台上,对于控制台的标准输入输出以及标准错误输出都有对应的文件描述符,分别为0,1,2。它们定义在 unistd.h中

#define  STDIN_FILENO  0  /* standard input file descriptor */

#define STDOUT_FILENO  1  /* standard output file descriptor */

#define STDERR_FILENO  2  /* standard error file descriptor */

UNIX内核加入TCP/IP协议的时候,便在系统中引入了一种新的IO操作,只不过由于网络连接的不可靠性,所以网络IO比本地设备的IO复杂很多。这一系列的接口叫做BSD Socket API,当初由伯克利大学研发,最终成为网络开发接口的标准。 网络通信从本质上讲也是进程间通信,只是这两个进程一般在网络中不同计算机上。当然Socket API其实也提供了专门用于本地IPC的使用方式:UNIX Domain Socket,这个这里就不细说了。本文所讲的Socket如无例外,均是说的Internet Socket。

在本地的进程中,每一个进程都可以通过PID来标识,对于网络上的一个计算机中的进程如何标识呢?网络中的计算机可以通过一个IP地址进行标识,一个计算机中的某个进程则可以通过一个无符号整数(端口号)来标识,所以一个网络中的进程可以通过IP地址+端口号的方式进行标识。

二.BSD Socket编程准备

1.地址

在程序中,我们如何保存一个地址呢?在中的sockaddr便是描述socket地址的结构体类型.

/*

* [XSI] Structure used by kernel to store most addresses.

*/

struct sockaddr {

__uint8_t  sa_len;    /* total length */

sa_family_t sa_family;  /* [XSI] address family */

char        sa_data[14];    /* [XSI] addr value (actually larger) */

};

为了方便设置用语网络通信的socket地址,引入了sockaddr_in结构体(对于UNIX Domain Socket则对应sockaddr_un)

/*

* Socket address, internet style.

*/

struct sockaddr_in {

__uint8_t  sin_len;

sa_family_t sin_family;

in_port_t  sin_port;//得是网络字节序

struct  in_addr sin_addr;//in_addr存在的原因则是历史原因,其实质是代表一个IP地址的32位整数

char        sin_zero[8];//bzero之,纯粹是为了兼容sockaddr

};

在实际编程的时候,经常需要将sockaddr_in强制转换成sockaddr类型。

2.端口

说到端口我们经常会联想到硬件,在网络编程中的端口其实是一个标识而已,或者说是系统的资源而已。系统提供了端口分配和管理的机制。

3.网络字节序

谈网络字节序(Endianness)之前我们先说说什么是字节序。字节序又叫端序,就是指计算机中存放 多字节数据的字节的顺序。典型的就是数据存放在内存中或者网络传输时的字节的顺序。常用的字节序有大端序(big-endian),小端序(litle-endian,另还有不常见的混合序middle-endian)。不同的CPU可能会使用不同的字节序,如X86,PDP-11等处理器为小端序,Motorola 6800,PowerPC 970等使用的是大端序。小端序是指低字节位存放在内存地址的低端,高端序是指高位字节存放在内存的低端。 举个例子来说明什么是大端序和小端序: 比如一个4字节的整数 16进制形式为 0x12345678,最左边是高位。

大端序

低位

高位

12

34

56

78

小端序

低位

高位

78

56

34

12

TCP/IP 各层协议将字节序使用的是大端序,我们把TCP/IP协议中使用的字节序称之为网络字节序。 编程的时候可以使用定义在sys/_endian.h中的相关的接口进行本地字节序和网络字节序的互转。

#define ntohs(x)    __DARWIN_OSSwapInt16(x) // 16位整数 网络字节序转主机字节序

#define htons(x)    __DARWIN_OSSwapInt16(x) // 16位整数 主机字节序转网络字节序

#define ntohl(x)    __DARWIN_OSSwapInt32(x)  //32位整数 网络字节序转主机字节序

#define htonl(x)    __DARWIN_OSSwapInt32(x) //32位整数 主机字节序转网络字节序

以上声明中 n代表netwrok, h代表host ,s代表short,l代表long

如果数据是单字节的话,则其没有字节序的说法了。

4.半相关与全相关

半相关(half-association)是指一个三元组 (协议,本地IP地址,本地端口),通过这个三元组就可以唯一标识一个网络中的进程,一般用于listening socket。但是实际进行通信的过程,至少需要两个进程,且它们所使用的协议必须一致,所以一个完成的网络通信至少需要一个五元组表示(协议,本地地址,本地端口,远端地址,远端端口),这样的五元组叫做全相关。

5.网络编程模型

网络存在的本质其实就是网络中个体之间的在某个领域的信息存在不对等性,所以一般情况下总有一些个体为另一些个体提供服务。提供服务器的我们把它叫做服务器,接受服务的叫做客户端。所以在网络编程中,也存在服务器端和客户端之分。

三.BSD Socket编程详解

下面的例子是一个简单的一对一聊天的程序,分服务器和客户端,且发送消息和接受消息次序固定。

Server端代码

#include

#include

#include

#include

#include

int main (int argc, const char * argv[])

{

struct sockaddr_in server_addr;

server_addr.sin_len = sizeof(struct sockaddr_in);

server_addr.sin_family = AF_INET;//Address families AF_INET互联网地址簇

server_addr.sin_port = htons(11332);

server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");

bzero(&(server_addr.sin_zero),8);

//创建socket

int server_socket = socket(AF_INET, SOCK_STREAM, 0);//SOCK_STREAM 有连接

if (server_socket == -1) {

perror("socket error");

return 1;

}

//绑定socket:将创建的socket绑定到本地的IP地址和端口,此socket是半相关的,只是负责侦听客户端的连接请求,并不能用于和客户端通信

int bind_result = bind(server_socket, (struct sockaddr *)&server_addr, sizeof(server_addr));

if (bind_result == -1) {

perror("bind error");

return 1;

}

//listen侦听 第一个参数是套接字,第二个参数为等待接受的连接的队列的大小,在connect请求过来的时候,完成三次握手后先将连接放到这个队列中,直到被accept处理。如果这个队列满了,且有新的连接的时候,对方可能会收到出错信息。

if (listen(server_socket, 5) == -1) {

perror("listen error");

return 1;

}

struct sockaddr_in client_address;

socklen_t address_len;

int client_socket = accept(server_socket, (struct sockaddr *)&client_address, &address_len);

//返回的client_socket为一个全相关的socket,其中包含client的地址和端口信息,通过client_socket可以和客户端进行通信。

if (client_socket == -1) {

perror("accept error");

return -1;

}

char recv_msg[1024];

char reply_msg[1024];

while (1) {

bzero(recv_msg, 1024);

bzero(reply_msg, 1024);

printf("reply:");

scanf("%s",reply_msg);

send(client_socket, reply_msg, 1024, 0);

long byte_num = recv(client_socket,recv_msg,1024,0);

recv_msg[byte_num] = '\0';

printf("client said:%s\n",recv_msg);

}

return 0;

}

Client端代码

#include

#include

#include

#include

#include

int main (int argc, const char * argv[])

{

struct sockaddr_in server_addr;

server_addr.sin_len = sizeof(struct sockaddr_in);

server_addr.sin_family = AF_INET;

server_addr.sin_port = htons(11332);

server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");

bzero(&(server_addr.sin_zero),8);

int server_socket = socket(AF_INET, SOCK_STREAM, 0);

if (server_socket == -1) {

perror("socket error");

return 1;

}

char recv_msg[1024];

char reply_msg[1024];

if (connect(server_socket, (struct sockaddr *)&server_addr, sizeof(struct sockaddr_in))==0)    {

//connect 成功之后,其实系统将你创建的socket绑定到一个系统分配的端口上,且其为全相关,包含服务器端的信息,可以用来和服务器端进行通信。

while (1) {

bzero(recv_msg, 1024);

bzero(reply_msg, 1024);

long byte_num = recv(server_socket,recv_msg,1024,0);

recv_msg[byte_num] = '\0';

printf("server said:%s\n",recv_msg);

printf("reply:");

scanf("%s",reply_msg);

if (send(server_socket, reply_msg, 1024, 0) == -1) {

perror("send error");

}

}

}

// insert code here...

printf("Hello, World!\n");

return 0;

}

上面的服务器端和客户端连接成功之后打开的端口的情况是怎么样的呢? * 服务器端 ,存在一个用于listen的半相关的socket,一个用于和客户端进行通信的全相关的socket

服务器端进程打开文件

* 客户端 存在一个用于和服务器端进行通信的全相关的socket

客户端进程打开文件

由于accept只运行了一次,所以服务器端一次只能和一个客户端进行通信,且使用的send和recv方法都是阻塞的,所以上面这个例子存在一个问题就是服务器端客户端连接成功之后,发送,接受,发送,接受的次序就被固定了。比如服务器端发送消息之后就等客户端发送消息了,没有接受到客户端的消息之前服务器端是没有办法发送消息的。使用select这个这个系统调用可以解决上面的问题。

四.使用select select这个系统调用,是一种多路复用IO方案,可以同时对多个文件描述符进行监控,从而知道哪些文件描述符可读,可写或者出错,不过select方法是阻塞的,可以设定超时时间。 select使用的步骤如下:

1.创建一个fd_set变量(fd_set实为包含了一个整数数组的结构体),用来存放所有的待检查的文件描述符

2.清空fd_set变量,并将需要检查的所有文件描述符加入fd_set

3.调用select。若返回-1,则说明出错;返回0,则说明超时,返回正数,则为发生状态变化的文件描述符的个数

4.若select返回大于0,则依次查看哪些文件描述符变的可读,并对它们进行处理

5.返回步骤2,开始新一轮的检测 若上面的聊天程序使用select进行改写,则是下面这样的

服务器端

#include

#include

#include

#include

#include

#include

#include

#define BACKLOG 5 //完成三次握手但没有accept的队列的长度

#define CONCURRENT_MAX 8 //应用层同时可以处理的连接

#define SERVER_PORT 11332

#define BUFFER_SIZE 1024

#define QUIT_CMD ".quit"

int client_fds[CONCURRENT_MAX];

int main (int argc, const char * argv[])

{

char input_msg[BUFFER_SIZE];

char recv_msg[BUFFER_SIZE];

//本地地址

struct sockaddr_in server_addr;

server_addr.sin_len = sizeof(struct sockaddr_in);

server_addr.sin_family = AF_INET;

server_addr.sin_port = htons(SERVER_PORT);

server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");

bzero(&(server_addr.sin_zero),8);

//创建socket

int server_sock_fd = socket(AF_INET, SOCK_STREAM, 0);

if (server_sock_fd == -1) {

perror("socket error");

return 1;

}

//绑定socket

int bind_result = bind(server_sock_fd, (struct sockaddr *)&server_addr, sizeof(server_addr));

if (bind_result == -1) {

perror("bind error");

return 1;

}

//listen

if (listen(server_sock_fd, BACKLOG) == -1) {

perror("listen error");

return 1;

}

//fd_set

fd_set server_fd_set;

int max_fd = -1;

struct timeval tv;

tv.tv_sec = 20;

tv.tv_usec = 0;

while (1) {

FD_ZERO(&server_fd_set);

//标准输入

FD_SET(STDIN_FILENO, &server_fd_set);

if (max_fd < STDIN_FILENO) {

max_fd = STDIN_FILENO;

}

//服务器端socket

FD_SET(server_sock_fd, &server_fd_set);

if (max_fd < server_sock_fd) {

max_fd = server_sock_fd;

}

//客户端连接

for (int i = 0; i < CONCURRENT_MAX; i++) {

if (client_fds[i]!=0) {

FD_SET(client_fds[i], &server_fd_set);

if (max_fd < client_fds[i]) {

max_fd = client_fds[i];

}

}

}

int ret = select(max_fd+1, &server_fd_set, NULL, NULL, &tv);

if (ret < 0) {

perror("select 出错\n");

continue;

}else if(ret == 0){

printf("select 超时\n");

continue;

}else{

//ret为未状态发生变化的文件描述符的个数

if (FD_ISSET(STDIN_FILENO, &server_fd_set)) {

//标准输入

bzero(input_msg, BUFFER_SIZE);

fgets(input_msg, BUFFER_SIZE, stdin);

//输入 ".quit" 则退出服务器

if (strcmp(input_msg, QUIT_CMD) == 0) {

exit(0);

}

for (int i=0; i

if (client_fds[i]!=0) {

send(client_fds[i], input_msg, BUFFER_SIZE, 0);

}

}

}

if (FD_ISSET(server_sock_fd, &server_fd_set)) {

//有新的连接请求

struct sockaddr_in client_address;

socklen_t address_len;

int client_socket_fd = accept(server_sock_fd, (struct sockaddr *)&client_address, &address_len);

if (client_socket_fd > 0) {

int index = -1;

for (int i = 0; i < CONCURRENT_MAX; i++) {

if (client_fds[i] == 0) {

index = i;

client_fds[i] = client_socket_fd;

break;

}

}

if (index >= 0) {

printf("新客户端(%d)加入成功 %s:%d \n",index,inet_ntoa(client_address.sin_addr),ntohs(client_address.sin_port));

}else{

bzero(input_msg, BUFFER_SIZE);

strcpy(input_msg, "服务器加入的客户端数达到最大值,无法加入!\n");

send(client_socket_fd, input_msg, BUFFER_SIZE, 0);

printf("客户端连接数达到最大值,新客户端加入失败 %s:%d \n",inet_ntoa(client_address.sin_addr),ntohs(client_address.sin_port));

}

}

}

for (int i = 0; i

if (client_fds[i]!=0) {

if (FD_ISSET(client_fds[i], &server_fd_set)) {

//处理某个客户端过来的消息

bzero(recv_msg, BUFFER_SIZE);

long byte_num = recv(client_fds[i],recv_msg,BUFFER_SIZE,0);

if (byte_num > 0) {

if (byte_num > BUFFER_SIZE) {

byte_num = BUFFER_SIZE;

}

recv_msg[byte_num] = '\0';

printf("客户端(%d):%s\n",i,recv_msg);

}else if(byte_num < 0){

printf("从客户端(%d)接受消息出错.\n",i);

}else{

FD_CLR(client_fds[i], &server_fd_set);

client_fds[i] = 0;

printf("客户端(%d)退出了\n",i);

}

}

}

}

}

}

return 0;

}

客户端

#include

#include

#include

#include

#include

#include

#include

#define BUFFER_SIZE 1024

int main (int argc, const char * argv[])

{

struct sockaddr_in server_addr;

server_addr.sin_len = sizeof(struct sockaddr_in);

server_addr.sin_family = AF_INET;

server_addr.sin_port = htons(11332);

server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");

bzero(&(server_addr.sin_zero),8);

int server_sock_fd = socket(AF_INET, SOCK_STREAM, 0);

if (server_sock_fd == -1) {

perror("socket error");

return 1;

}

char recv_msg[BUFFER_SIZE];

char input_msg[BUFFER_SIZE];

if (connect(server_sock_fd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr_in))==0) {

fd_set client_fd_set;

struct timeval tv;

tv.tv_sec = 20;

tv.tv_usec = 0;

while (1) {

FD_ZERO(&client_fd_set);

FD_SET(STDIN_FILENO, &client_fd_set);

FD_SET(server_sock_fd, &client_fd_set);

int ret = select(server_sock_fd + 1, &client_fd_set, NULL, NULL, &tv);

if (ret < 0 ) {

printf("select 出错!\n");

continue;

}else if(ret ==0){

printf("select 超时!\n");

continue;

}else{

if (FD_ISSET(STDIN_FILENO, &client_fd_set)) {

bzero(input_msg, BUFFER_SIZE);

fgets(input_msg, BUFFER_SIZE, stdin);

if (send(server_sock_fd, input_msg, BUFFER_SIZE, 0) == -1) {

perror("发送消息出错!\n");

}

}

if (FD_ISSET(server_sock_fd, &client_fd_set)) {

bzero(recv_msg, BUFFER_SIZE);

long byte_num = recv(server_sock_fd,recv_msg,BUFFER_SIZE,0);

if (byte_num > 0) {

if (byte_num > BUFFER_SIZE) {

byte_num = BUFFER_SIZE;

}

recv_msg[byte_num] = '\0';

printf("服务器:%s\n",recv_msg);

}else if(byte_num < 0){

printf("接受消息出错!\n");

}else{

printf("服务器端退出!\n");

exit(0);

}

}

}

}

}

return 0;

}

当然select也有其局限性。当fd_set中的文件描述符较少,或者大都数文件描述符都比较活跃的时候,select的效率还是不错的。Mac系统中已经定义了fd_set 最大可以容纳的文件描述符的个数为1024

//sys/_structs.h

#define __DARWIN_FD_SETSIZE 1024

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

//Kernel.framework sys/select.h

#define FD_SETSIZE  __DARWIN_FD_SETSIZE

每一次select 调用的时候,都涉及到user space和kernel space的内存拷贝,且会对fd_set中的所有文件描述符进行遍历,如果所有的文件描述符均不满足,且没有超时,则当前进程便开始睡眠,直到超时或者有文件描述符状态发生变化。当文件描述符数量较大的时候,将耗费大量的CPU时间。所以后来有新的方案出现了,如windows2000引入的IOCP,Linux Kernel 2.6中成熟的epoll,FreeBSD4.x引入的kqueue。

五.使用kqueue Mac是基于BSD的内核

所使用的是kqueue(kernel event notification mechanism,详细内容可以Mac中man 2 kqueue),kqueue比select先进的地方就在于使用事件触发的机制,且其调用无需每次对所有的文件描述符进行遍历,返回的时候只返回需要处理的事件,而不像select中需要自己去一个个通过FD_ISSET检查。 kqueue默认的触发方式是level 水平触发,可以通过设置event的flag为EV_CLEAR使得这个事件变为边沿触发,可能epoll的触发方式无法细化到单个event,需要查证。 kqueue中涉及两个系统调用,kqueue()和kevent()

kqueue() 创建kernel级别的事件队列,并返回队列的文件描述符

kevent() 往事件队列中加入订阅事件,或者返回相关的事件数组 kqueue使用的流程一般如下:

创建kqueue

创建struct kevent变量(注意这里的kevent是结构体类型名),可以通过EV_SET这个宏提供的快捷方式进行创建

通过kevent系统调用将创建好的kevent结构体变量加入到kqueue队列中,完成对指定文件描述符的事件的订阅

通过kevent系统调用获取满足条件的事件队列,并对每一个事件进行处理

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#define BACKLOG 5 //完成三次握手但没有accept的队列的长度

#define CONCURRENT_MAX 8 //应用层同时可以处理的连接

#define SERVER_PORT 11332

#define BUFFER_SIZE 1024

#define QUIT_CMD ".quit"

int client_fds[CONCURRENT_MAX];

struct kevent events[10];//CONCURRENT_MAX + 2

int main (int argc, const char * argv[])

{

char input_msg[BUFFER_SIZE];

char recv_msg[BUFFER_SIZE];

//本地地址

struct sockaddr_in server_addr;

server_addr.sin_len = sizeof(struct sockaddr_in);

server_addr.sin_family = AF_INET;

server_addr.sin_port = htons(SERVER_PORT);

server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");

bzero(&(server_addr.sin_zero),8);

//创建socket

int server_sock_fd = socket(AF_INET, SOCK_STREAM, 0);

if (server_sock_fd == -1) {

perror("socket error");

return 1;

}

//绑定socket

int bind_result = bind(server_sock_fd, (struct sockaddr *)&server_addr, sizeof(server_addr));

if (bind_result == -1) {

perror("bind error");

return 1;

}

//listen

if (listen(server_sock_fd, BACKLOG) == -1) {

perror("listen error");

return 1;

}

struct timespec timeout = {10,0};

//kqueue

int kq = kqueue();

if (kq == -1) {

perror("创建kqueue出错!\n");

exit(1);

}

struct kevent event_change;

EV_SET(&event_change, STDIN_FILENO, EVFILT_READ, EV_ADD, 0, 0, NULL);

kevent(kq, &event_change, 1, NULL, 0, NULL);

EV_SET(&event_change, server_sock_fd, EVFILT_READ, EV_ADD, 0, 0, NULL);

kevent(kq, &event_change, 1, NULL, 0, NULL);

while (1) {

int ret = kevent(kq, NULL, 0, events, 10, &timeout);

if (ret < 0) {

printf("kevent 出错!\n");

continue;

}else if(ret == 0){

printf("kenvent 超时!\n");

continue;

}else{

//ret > 0 返回事件放在events中

for (int i = 0; i < ret; i++) {

struct kevent current_event = events[i];

//kevent中的ident就是文件描述符

if (current_event.ident == STDIN_FILENO) {

//标准输入

bzero(input_msg, BUFFER_SIZE);

fgets(input_msg, BUFFER_SIZE, stdin);

//输入 ".quit" 则退出服务器

if (strcmp(input_msg, QUIT_CMD) == 0) {

exit(0);

}

for (int i=0; i

if (client_fds[i]!=0) {

send(client_fds[i], input_msg, BUFFER_SIZE, 0);

}

}

}else if(current_event.ident == server_sock_fd){

//有新的连接请求

struct sockaddr_in client_address;

socklen_t address_len;

int client_socket_fd = accept(server_sock_fd, (struct sockaddr *)&client_address, &address_len);

if (client_socket_fd > 0) {

int index = -1;

for (int i = 0; i < CONCURRENT_MAX; i++) {

if (client_fds[i] == 0) {

index = i;

client_fds[i] = client_socket_fd;

break;

}

}

if (index >= 0) {

EV_SET(&event_change, client_socket_fd, EVFILT_READ, EV_ADD, 0, 0, NULL);

kevent(kq, &event_change, 1, NULL, 0, NULL);

printf("新客户端(fd = %d)加入成功 %s:%d \n",client_socket_fd,inet_ntoa(client_address.sin_addr),ntohs(client_address.sin_port));

}else{

bzero(input_msg, BUFFER_SIZE);

strcpy(input_msg, "服务器加入的客户端数达到最大值,无法加入!\n");

send(client_socket_fd, input_msg, BUFFER_SIZE, 0);

printf("客户端连接数达到最大值,新客户端加入失败 %s:%d \n",inet_ntoa(client_address.sin_addr),ntohs(client_address.sin_port));

}

}

}else{

//处理某个客户端过来的消息

bzero(recv_msg, BUFFER_SIZE);

long byte_num = recv((int)current_event.ident,recv_msg,BUFFER_SIZE,0);

if (byte_num > 0) {

if (byte_num > BUFFER_SIZE) {

byte_num = BUFFER_SIZE;

}

recv_msg[byte_num] = '\0';

printf("客户端(fd = %d):%s\n",(int)current_event.ident,recv_msg);

}else if(byte_num < 0){

printf("从客户端(fd = %d)接受消息出错.\n",(int)current_event.ident);

}else{

EV_SET(&event_change, current_event.ident, EVFILT_READ, EV_DELETE, 0, 0, NULL);

kevent(kq, &event_change, 1, NULL, 0, NULL);

close((int)current_event.ident);

for (int i = 0; i < CONCURRENT_MAX; i++) {

if (client_fds[i] == (int)current_event.ident) {

client_fds[i] = 0;

break;

}

}

printf("客户端(fd = %d)退出了\n",(int)current_event.ident);

}

}

}

}

}

return 0;

}

其实kqueue的应用场景非常的广阔,可以监控文件系统中文件的变化(对文件变化的事件可以粒度非常的细,具体可以查看kqueue的手册),监控系统进程的生命周期。GCD的事件处理便是建立在kqueue之上的。

六.使用Streams

使用Objective-C的一大优点便是面向对象编程,使得逻辑抽象得更加优美,更加符合人类思维。 一开始说过,无论是对于文件的操作或者对于网络的操作,本质上都是IO操作,无非写数据和读数据,可以对这种输入输出进行抽象,抽象成输入流和输出流,从输入流中读取数据,往输出流中写数据。 Cocoa中的NSInputStream和NSOutputStream便是输入流和输出流的抽象,它们的实现分别基于CoreFoundation中的CFReadStream和CFWriteStream。 输入输出流对runloop有很好的支持。 NSInputStream和CFReadStream以及NSOutputStream和CFWriteStream之间可以通过 "toll-free bridging"实现无缝的类型转换。 CoreFoundation中的CFStream提供了输入输出流和CFSocket绑定的函数。 这样便可以通过输入输出流和远端进行通信了。 首先通过XCode创建一个Foundation(C的也行,但是你得将main.c改成main.m)的命令行项目. 创建一个ChatServer的类,包含一个run的方法。在Cocoa的程序中有一点是和C语言不同的,你无需自己去写一个死循环充当runloop,框架本身就对runloop进行了支持,需要做的就是将事件源加入到当前线程的runloop中,然后启动runloop。 所以在run方法中,创建好用于侦听连接请求的socket,socket有对应的处理连接accept的回调函数,以及把它封装成runloop的输入源,加入到当前runloop。 我们还得从标准输入获取需要发送消息,所以使用了CFFileDescriptor,它是文件描述符的objc的封装,加入了runloop的支持,通过它可以将标准输入以输入源的方法加入到当前runloop,当标准输入缓冲区有数据可读的时候,设置好的回调函数便会被调用。 最后启动runloop。

ChatServer中的run方法

- (BOOL)run:(NSError **)error{

BOOL successful = YES;

CFSocketContext socketCtxt = {0, self, NULL, NULL, NULL};

_socket = CFSocketCreate(kCFAllocatorDefault, PF_INET, SOCK_STREAM,

IPPROTO_TCP,

kCFSocketAcceptCallBack,

(CFSocketCallBack)&SocketConnectionAcceptedCallBack,

&socketCtxt);

if (NULL == _socket) {

if (nil != error) {

*error = [[NSError alloc]

initWithDomain:ServerErrorDomain

code:kServerNoSocketsAvailable

userInfo:nil];

}

successful = NO;

}

if(YES == successful) {

// enable address reuse

int yes = 1;

setsockopt(CFSocketGetNative(_socket),

SOL_SOCKET, SO_REUSEADDR,

(void *)&yes, sizeof(yes));

uint8_t packetSize = 128;

setsockopt(CFSocketGetNative(_socket),

SOL_SOCKET, SO_SNDBUF,

(void *)&packetSize, sizeof(packetSize));

setsockopt(CFSocketGetNative(_socket),

SOL_SOCKET, SO_RCVBUF,

(void *)&packetSize, sizeof(packetSize));

struct sockaddr_in addr4;

memset(&addr4, 0, sizeof(addr4));

addr4.sin_len = sizeof(addr4);

addr4.sin_family = AF_INET;

addr4.sin_port = htons(CHAT_SERVER_PORT);

addr4.sin_addr.s_addr = htonl(INADDR_ANY);

NSData *address4 = [NSData dataWithBytes:&addr4 length:sizeof(addr4)];

if (kCFSocketSuccess != CFSocketSetAddress(_socket, (CFDataRef)address4)) {

if (error) *error = [[NSError alloc]

initWithDomain:ServerErrorDomain

code:kServerCouldNotBindToIPv4Address

userInfo:nil];

if (_socket) CFRelease(_socket);

_socket = NULL;

successful = NO;

} else {

// now that the binding was successful, we get the port number

NSData *addr = [(NSData *)CFSocketCopyAddress(_socket) autorelease];

memcpy(&addr4, [addr bytes], [addr length]);

self.port = ntohs(addr4.sin_port);

// 将socket 输入源加入到当前的runloop

CFRunLoopRef cfrl = CFRunLoopGetCurrent();

CFRunLoopSourceRef source4 = CFSocketCreateRunLoopSource(kCFAllocatorDefault, _socket, 0);

CFRunLoopAddSource(cfrl, source4, kCFRunLoopDefaultMode);

CFRelease(source4);

//标准输入,当在命令行中输入时,回调函数便会被调用

CFFileDescriptorContext context = {0,self,NULL,NULL,NULL};

CFFileDescriptorRef stdinFDRef = CFFileDescriptorCreate(kCFAllocatorDefault, STDIN_FILENO, true, FileDescriptorCallBack, &context);

CFFileDescriptorEnableCallBacks(stdinFDRef,kCFFileDescriptorReadCallBack);

CFRunLoopSourceRef stdinSource = CFFileDescriptorCreateRunLoopSource(kCFAllocatorDefault, stdinFDRef, 0);

CFRunLoopAddSource(cfrl, stdinSource, kCFRunLoopDefaultMode);

CFRelease(stdinSource);

CFRelease(stdinFDRef);

CFRunLoopRun();

}

}

return successful;

}

当有客户端连接请求过来时, SocketConnectionAcceptedCallBack这个回调函数会被调用,根据新的全相关的socket,生成输入输出流,并设置输入输出流的delegate方法,将其添加到当前的runloop,这样流中有数据过来的时候,delegate方法会被调用。SocketConnectionAcceptedCallBack函数

static void SocketConnectionAcceptedCallBack(CFSocketRef socket,

CFSocketCallBackType type,

CFDataRef address,

const void *data, void *info) {

ChatServer *theChatServer = (ChatServer *)info;

if (kCFSocketAcceptCallBack == type) {

// 摘自kCFSocketAcceptCallBack的文档,New connections will be automatically accepted and the callback is called with the data argument being a pointer to a CFSocketNativeHandle of the child socket. This callback is usable only with listening sockets.

CFSocketNativeHandle nativeSocketHandle = *(CFSocketNativeHandle *)data;

// create the read and write streams for the connection to the other process

CFReadStreamRef readStream = NULL;

CFWriteStreamRef writeStream = NULL;

CFStreamCreatePairWithSocket(kCFAllocatorDefault, nativeSocketHandle,

&readStream, &writeStream);

if(NULL != readStream && NULL != writeStream) {

CFReadStreamSetProperty(readStream,

kCFStreamPropertyShouldCloseNativeSocket,

kCFBooleanTrue);

CFWriteStreamSetProperty(writeStream,

kCFStreamPropertyShouldCloseNativeSocket,

kCFBooleanTrue);

NSInputStream *inputStream = (NSInputStream *)readStream;//toll-free bridging

NSOutputStream *outputStream = (NSOutputStream *)writeStream;//toll-free bridging

inputStream.delegate = theChatServer;

[inputStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];

[inputStream open];

outputStream.delegate = theChatServer;

[outputStream scheduleInRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];

[outputStream open];

Client *aClient = [[Client alloc] init];

aClient.inputStream = inputStream;

aClient.outputStream = outputStream;

aClient.sock_fd = nativeSocketHandle;

[theChatServer.clients setValue:aClient

forKey:[NSString stringWithFormat:@"%d",inputStream]];

NSLog(@"有新客户端(sock_fd=%d)加入",nativeSocketHandle);

} else {

close(nativeSocketHandle);

}

if (readStream) CFRelease(readStream);

if (writeStream) CFRelease(writeStream);

}

}

当客户端有数据传过来时,相应的NSInputStream的delegate方法被调用

- (void) stream:(NSStream*)stream handleEvent:(NSStreamEvent)eventCode {

switch (eventCode) {

case NSStreamEventOpenCompleted: {

break;

}

case NSStreamEventHasBytesAvailable: {

Client *client = [self.clients objectForKey:[NSString stringWithFormat:@"%d",stream]];

NSMutableData *data = [NSMutableData data];

uint8_t *buf = calloc(128, sizeof(uint8_t));

NSUInteger len = 0;

while([(NSInputStream*)stream hasBytesAvailable]) {

len = [(NSInputStream*)stream read:buf maxLength:128];

if(len > 0) {

[data appendBytes:buf length:len];

}

}

free(buf);

if ([data length] == 0) {

//客户端退出

NSLog(@"客户端(sock_fd=%d)退出",client.sock_fd);

[self.clients removeObjectForKey:[NSString stringWithFormat:@"%d",stream]];

close(client.sock_fd);

}else{

NSLog(@"收到客户端(sock_fd=%d)消息:%@",client.sock_fd,[[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding] autorelease]);

}

break;

}

case NSStreamEventHasSpaceAvailable: {

break;

}

case NSStreamEventEndEncountered: {

break;

}

case NSStreamEventErrorOccurred: {

break;

}

default:

break;

}

}

当在debug窗口中输入内容并回车时,标准输入缓冲区中便有数据了,这个时候回调函数FileDescriptorCallBack将被调用,处理标准输入。

static void FileDescriptorCallBack(CFFileDescriptorRef f,

CFOptionFlags callBackTypes,

void *info){

int fd = CFFileDescriptorGetNativeDescriptor(f);

ChatServer *theChatServer = (ChatServer *)info;

if (fd == STDIN_FILENO) {

NSData *inputData = [[NSFileHandle fileHandleWithStandardInput] availableData];

NSString *inputString = [[[NSString alloc] initWithData:inputData encoding:NSUTF8StringEncoding] autorelease];

NSLog(@"准备发送消息:%@",inputString);

for (Client *client in [theChatServer.clients allValues]) {

[client.outputStream write:[inputData bytes] maxLength:[inputData length]];

}

//处理完数据之后必须重新Enable 回调函数

CFFileDescriptorEnableCallBacks(f,kCFFileDescriptorReadCallBack);

}

}

服务器端

创建Socket

将Socket和本地的地址端口绑定

开始进行侦听

握手成功,接受请求,得到一个新的Socket,通过它可以和客户端进行通信

客户端

创建一个Socket和服务器的地址并通过它们向服务器发送连接请求

连接成功,客户端的Socket会绑定到系统分配的一个端口上,并可以通过它和服务器端进行通信

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • 大纲 一.Socket简介 二.BSD Socket编程准备 1.地址 2.端口 3.网络字节序 4.半相关与全相...
    y角阅读 2,398评论 2 11
  • 研究IPv6 socket编程原因: Supporting IPv6 in iOS 9 WWDC2015苹果宣布在...
    li大鹏阅读 7,315评论 7 15
  • BSD socket API 一、客户端操作 1、绑定地址和端口操作 2、接受客户端的链接 3、从服务端接受消息 ...
    806349745123阅读 826评论 0 0
  • 一 、Socket 网络上的两个程序通过一个双向的通信连接实现数据的交换,这个连接的一端称为一个 Socket。S...
    空白Null阅读 1,752评论 1 9
  • 谁哼着摇篮曲,引我入梦。 那伴奏的声音可是小提琴? 白鸽飞过的广场下, 玄色西装的男孩, 与一袭白衣的少女, 在艳...
    塘月花影阅读 243评论 9 7