一析孽、Socket連接
1.連接過程
建立Socket連接至少需要一對(duì)套接字,其中一個(gè)運(yùn)行于客戶端,稱為ClientSocket围苫,另一個(gè)運(yùn)行于服務(wù)器端退盯,稱為ServerSocket彼乌。
套接字之間的連接過程分為三個(gè)步驟:服務(wù)器監(jiān)聽,客戶端請(qǐng)求渊迁,連接確認(rèn)慰照。
- 1)服務(wù)器監(jiān)聽:服務(wù)器端套接字并不定位具體的客戶端套接字,而是處于等待連接的狀態(tài)琉朽,實(shí)時(shí)監(jiān)控網(wǎng)絡(luò)狀態(tài)焚挠,等待客戶端的連接請(qǐng)求。
- 2)客戶端請(qǐng)求:指客戶端的套接字提出連接請(qǐng)求漓骚,要連接的目標(biāo)是服務(wù)器端的套接字蝌衔。為此,客戶端的套接字必須首先描述它要連接的服務(wù)器的套接字蝌蹂,指出服務(wù)器端套接字的地址和端口號(hào)噩斟,然后就向服務(wù)器端套接字提出連接請(qǐng)求。
- 3)連接確認(rèn):當(dāng)服務(wù)器端套接字監(jiān)聽到或者說接收到客戶端套接字的連接請(qǐng)求時(shí)孤个,就響應(yīng)客戶端套接字的請(qǐng)求剃允,建立一個(gè)新的線程,把服務(wù)器端套接字的描述發(fā)給客戶端,一旦客戶端確認(rèn)了此描述斥废,雙方就正式建立連接椒楣。而服務(wù)器端套接字繼續(xù)處于監(jiān)聽狀態(tài),繼續(xù)接收其他客戶端套接字的連接請(qǐng)求牡肉。
- 4)斷開連接.
2.TCP Socket
常用的Socket類型有兩種:流式Socket(SOCK_STREAM)和數(shù)據(jù)報(bào)式Socket(SOCK_DGRAM)捧灰。流式是一種面向連接的Socket,針對(duì)于面向連接的TCP服務(wù)應(yīng)用统锤;數(shù)據(jù)報(bào)式Socket是一種無連接的Socket毛俏,對(duì)應(yīng)于無連接的UDP服務(wù)應(yīng)用。
2.1 TCP Socket客戶端
客戶端的工作流程:首先調(diào)用socket函數(shù)創(chuàng)建一個(gè)Socket饲窿,然后指定服務(wù)端的IP地址和端口號(hào)煌寇,就可以調(diào)用sendto將字符串傳送給服務(wù)器端,并可以調(diào)用recvfrom接收服務(wù)器端返回的字符串逾雄,最后關(guān)閉該socket阀溶。
第一步:創(chuàng)建socket并配置socket
第二步:調(diào)用bind綁定監(jiān)聽ip和端口號(hào)
第三步:調(diào)用connect連接服務(wù)器
第四步:調(diào)用getsockname獲取套接字信息
第五步:調(diào)用send發(fā)送消息到服務(wù)器端
第六步:調(diào)用close關(guān)閉socket
這里沒有寫接收來自服務(wù)器端的消息,大家可以自行添加鸦泳。
客戶端的代碼實(shí)現(xiàn):
- (void)tcpClient {
// 第一步:創(chuàng)建soket
// TCP是基于數(shù)據(jù)流的银锻,因此參數(shù)二使用SOCK_STREAM
int error = -1;
int clientSocketId = socket(AF_INET, SOCK_STREAM, 0);
BOOL success = (clientSocketId != -1);
struct sockaddr_in addr;
// 第二步:綁定端口號(hào)
if (success) {
NSLog(@"client socket create success");
// 初始化
memset(&addr, 0, sizeof(addr));
addr.sin_len = sizeof(addr);
// 指定協(xié)議簇為AF_INET,比如TCP/UDP等
addr.sin_family = AF_INET;
// 監(jiān)聽任何ip地址
addr.sin_addr.s_addr = INADDR_ANY;
error = bind(clientSocketId, (const struct sockaddr *)&addr, sizeof(addr));
success = (error == 0);
}
if (success) {
// p2p
struct sockaddr_in peerAddr;
memset(&peerAddr, 0, sizeof(peerAddr));
peerAddr.sin_len = sizeof(peerAddr);
peerAddr.sin_family = AF_INET;
peerAddr.sin_port = htons(1024);
// 指定服務(wù)端的ip地址辽故,測(cè)試時(shí)徒仓,修改成對(duì)應(yīng)自己服務(wù)器的ip
peerAddr.sin_addr.s_addr = inet_addr("192.168.1.107");
socklen_t addrLen;
addrLen = sizeof(peerAddr);
NSLog(@"will be connecting");
// 第三步:連接服務(wù)器
error = connect(clientSocketId, (struct sockaddr *)&peerAddr, addrLen);
success = (error == 0);
if (success) {
// 第四步:獲取套接字信息
error = getsockname(clientSocketId, (struct sockaddr *)&addr, &addrLen);
success = (error == 0);
if (success) {
NSLog(@"client connect success, local address:%s,port:%d",
inet_ntoa(addr.sin_addr),
ntohs(addr.sin_port));
// 這里只發(fā)送10次
int count = 10;
do {
// 第五步:發(fā)送消息到服務(wù)端
send(clientSocketId, "哈哈,server您好誊垢!", 1024, 0);
count--;
// 告訴server掉弛,客戶端退出了
if (count == 0) {
send(clientSocketId, "exit", 1024, 0);
}
} while (count >= 1);
// 第六步:關(guān)閉套接字
close(clientSocketId);
}
} else {
NSLog(@"connect failed");
// 第六步:關(guān)閉套接字
close(clientSocketId);
}
}
}
客戶端的打印日志
2015-12-06 18:35:00.385 iOS-Socket-C-Version-Client[9726:4256295] client socket create success
2015-12-06 18:35:00.386 iOS-Socket-C-Version-Client[9726:4256295] will be connecting
2015-12-06 18:35:00.507 iOS-Socket-C-Version-Client[9726:4256295] client connect success, local address:192.168.1.100,port:50311
說明連接服務(wù)器成功,然后發(fā)送了消息到服務(wù)器端喂走。
2.2 TCP Socket服務(wù)器端
服務(wù)器端的工作流程:首先調(diào)用socket函數(shù)創(chuàng)建一個(gè)套接字殃饿,然后調(diào)用bind函數(shù)將其與本機(jī)地址以及一個(gè)本地端口號(hào)綁定,接收到一個(gè)客戶端時(shí)芋肠,服務(wù)器顯示該客戶端的IP地址乎芳,并將字串返回給客戶端。
第一步:創(chuàng)建socket并配置socket
第二步:調(diào)用bind綁定服務(wù)器本機(jī)ip及端口號(hào)
第三步:調(diào)用listen監(jiān)聽客戶端的連接帖池,并指定同時(shí)最多可讓accept的數(shù)量
第四步:調(diào)用accept等待客戶端的連接
第五步:調(diào)用recvfrom接收來自客戶端的消息
第六步:調(diào)用close關(guān)閉socket
服務(wù)器端代碼實(shí)現(xiàn)
- (void)tcpServer {
// 第一步:創(chuàng)建socket
int error = -1;
// 創(chuàng)建socket套接字
int serverSocketId = socket(AF_INET, SOCK_STREAM, 0);
// 判斷創(chuàng)建socket是否成功
BOOL success = (serverSocketId != -1);
// 第二步:綁定端口號(hào)
if (success) {
NSLog(@"server socket create success");
// Socket address
struct sockaddr_in addr;
// 初始化全置為0
memset(&addr, 0, sizeof(addr));
// 指定socket地址長(zhǎng)度
addr.sin_len = sizeof(addr);
// 指定網(wǎng)絡(luò)協(xié)議奈惑,比如這里使用的是TCP/UDP則指定為AF_INET
addr.sin_family = AF_INET;
// 指定端口號(hào)
addr.sin_port = htons(1024);
// 指定監(jiān)聽的ip,指定為INADDR_ANY時(shí)睡汹,表示監(jiān)聽所有的ip
addr.sin_addr.s_addr = INADDR_ANY;
// 綁定套接字
error = bind(serverSocketId, (const struct sockaddr *)&addr, sizeof(addr));
success = (error == 0);
}
// 第三步:監(jiān)聽
if (success) {
NSLog(@"bind server socket success");
error = listen(serverSocketId, 5);
success = (error == 0);
}
if (success) {
NSLog(@"listen server socket success");
while (true) {
// p2p
struct sockaddr_in peerAddr;
int peerSocketId;
socklen_t addrLen = sizeof(peerAddr);
// 第四步:等待客戶端連接
// 服務(wù)器端等待從編號(hào)為serverSocketId的Socket上接收客戶連接請(qǐng)求
peerSocketId = accept(serverSocketId, (struct sockaddr *)&peerAddr, &addrLen);
success = (peerSocketId != -1);
if (success) {
NSLog(@"accept server socket success,remote address:%s,port:%d",
inet_ntoa(peerAddr.sin_addr),
ntohs(peerAddr.sin_port));
char buf[1024];
size_t len = sizeof(buf);
// 第五步:接收來自客戶端的信息
// 當(dāng)客戶端輸入exit時(shí)才退出
do {
// 接收來自客戶端的信息
recv(peerSocketId, buf, len, 0);
if (strlen(buf) != 0) {
NSString *str = [NSString stringWithCString:buf encoding:NSUTF8StringEncoding];
if (str.length >= 1) {
NSLog(@"received message from client:%@",str);
}
}
} while (strcmp(buf, "exit") != 0);
NSLog(@"收到exit信號(hào)肴甸,本次socket通信完畢");
// 第六步:關(guān)閉socket
close(peerSocketId);
}
}
}
}
服務(wù)器端的打印日志
2015-12-06 18:34:31.258 iOS-Socket-C-Version-Server[39929:2622200] server socket create success
2015-12-06 18:34:31.258 iOS-Socket-C-Version-Server[39929:2622200] bind server socket success
2015-12-06 18:34:31.259 iOS-Socket-C-Version-Server[39929:2622200] listen server socket success
2015-12-06 18:35:00.743 iOS-Socket-C-Version-Server[39929:2622200] accept server socket success,remote address:192.168.1.100,port:50311
2015-12-06 18:35:00.743 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈,server您好囚巴!
2015-12-06 18:35:00.743 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈原在,server您好友扰!
2015-12-06 18:35:00.743 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈,server您好庶柿!
2015-12-06 18:35:00.744 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈村怪,server您好!
2015-12-06 18:35:00.744 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈浮庐,server您好甚负!
2015-12-06 18:35:00.744 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈,server您好兔辅!
2015-12-06 18:35:00.744 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈腊敲,server您好击喂!
2015-12-06 18:35:00.744 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈维苔,server您好!
2015-12-06 18:35:00.744 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈懂昂,server您好介时!
2015-12-06 18:35:00.745 iOS-Socket-C-Version-Server[39929:2622200] received message from client:哈哈,server您好凌彬!
2015-12-06 18:35:00.745 iOS-Socket-C-Version-Server[39929:2622200] received message from client:exit
2015-12-06 18:35:00.745 iOS-Socket-C-Version-Server[39929:2622200] 收到exit信號(hào)沸柔,本次socket通信完畢
我們這里打印出了客戶端發(fā)來的消息,由于上面實(shí)現(xiàn)的代碼中铲敛,只發(fā)10次褐澎,所以這里只有10條。