Tcp是面向連接的,安全可靠的傳輸協(xié)議。Tcp的程序基本框架設計圖
Tcp Socket客戶端
客戶端的工作流程:首先調用socket函數(shù)創(chuàng)建一個Socket,然后指定服務器的IP地址河端口號,就可以調用sendto將字符串傳給服務器端,并可以調用recvfrom接收服務器端返回的字符串,最后關閉該socket。
1.第一步:創(chuàng)建socket并配置socket
2.第二步:調用bind綁定監(jiān)聽ip和端口號
3.第三步:調用connect連接服務器
4.第四步:調用getsockname獲取套接字信息
5.第五步:調用send發(fā)送消息到服務器端
6.第六步:調用close關閉socket
客戶端的代碼實現(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;
// 第二步:綁定端口號
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);
// 指定服務端的ip地址,測試時,修改成對應自己服務器的ip
peerAddr.sin_addr.s_addr = inet_addr("192.168.1.107");
socklen_t addrLen;
addrLen = sizeof(peerAddr);
NSLog(@"will be connecting");
// 第三步:連接服務器
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ā)送消息到服務端
send(clientSocketId, "哈哈,server您好!", 1024, 0);
count--;
// 告訴server,客戶端退出了
if (count == 0) {
send(clientSocketId, "exit", 1024, 0);
}
} while (count >= 1);
// 第六步:關閉套接字
close(clientSocketId);
}
} else {
NSLog(@"connect failed");
// 第六步:關閉套接字
close(clientSocketId);
}
}
}
TCP Socket服務器端
? ? ? ? 服務器端的工作流程:首先調用socket函數(shù)創(chuàng)建一個套接字,然后調用bind函數(shù)將其與本機地址以及一個本地端口號綁定,接收到一個客戶端時,服務器顯示該客戶端的IP地址,并將字串返回給客戶端。
1.第一步:創(chuàng)建socket并配置socket
2.第二步:調用bind綁定服務器本機ip及端口號
3.第三步:調用listen監(jiān)聽客戶端的連接,并指定同時最多可讓accept的數(shù)量
4.第四步:調用accept等待客戶端的連接
5.第五步:調用recvfrom接收來自客戶端的消息
6.第六步:調用close關閉socket
服務器端代碼實現(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);
// 第二步:綁定端口號
if (success) {
NSLog(@"server socket create success");
// Socket address
struct sockaddr_in addr;
// 初始化全置為0
memset(&addr, 0, sizeof(addr));
// 指定socket地址長度
addr.sin_len = sizeof(addr);
// 指定網(wǎng)絡協(xié)議,比如這里使用的是TCP/UDP則指定為AF_INET
addr.sin_family = AF_INET;
// 指定端口號
addr.sin_port = htons(1024);
// 指定監(jiān)聽的ip,指定為INADDR_ANY時,表示監(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);
// 第四步:等待客戶端連接
// 服務器端等待從編號為serverSocketId的Socket上接收客戶連接請求
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);
// 第五步:接收來自客戶端的信息
// 當客戶端輸入exit時才退出
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信號,本次socket通信完畢");
// 第六步:關閉socket
close(peerSocketId);
}
}
}
}