电竞比分网-中国电竞赛事及体育赛事平台

分享

自己動手用c語言寫一個基于服務(wù)器和客戶端(TCP)

 jingxin95 2019-06-01

如果想要自己寫一個服務(wù)器和客戶端,我們需要掌握一定的網(wǎng)絡(luò)編程技術(shù),個人認(rèn)為,網(wǎng)絡(luò)編程中最關(guān)鍵的就是這個東西——socket(套接字)。

socket(套接字):簡單來講,socket就是用于描述IP地址和端口,是一個通信鏈的句柄,可以用來實現(xiàn)不同虛擬機或不同計算機之間的通信。

TCP協(xié)議

TCP協(xié)議:是一種面向連接的、可靠的、基于字節(jié)流的傳輸層通信協(xié)議,由IETF的RFC 793定義。在簡化的計算機網(wǎng)絡(luò)OSI模型中,它完成第四層傳輸層所指定的功能。

關(guān)鍵詞:三次握手,可靠,基于字節(jié)流。

可能有朋友會問,TCP就這么簡單一句話嗎?當(dāng)然不是,TCP作為非常重要的傳輸協(xié)議,細(xì)節(jié)知識是很多的,細(xì)講起來這一篇文章怕是不夠。不過在本篇內(nèi)容中,我們只需了解他的幾個關(guān)鍵詞特性,就能很好的理解下面的內(nèi)容。

自己動手用c語言寫一個基于服務(wù)器和客戶端(TCP)

TCP服務(wù)器端和客戶端的運行流程

如圖,這是一個完整的TCP服務(wù)器——客戶端的運行流程圖,其實我個人認(rèn)為程序啊,不管哪個語言都是一樣,核心就在于算法的設(shè)計和函數(shù)的調(diào)用。那么圖中的函數(shù)都是什么意思呢?

1.創(chuàng)建socket

socket是一個結(jié)構(gòu)體,被創(chuàng)建在內(nèi)核中

 sockfd=socket(AF_INET,SOCK_STREAM,0); //AF_INT:ipv4, SOCK_STREAM:tcp協(xié)議

2.調(diào)用bind函數(shù)

將socket和地址(包括ip、port)綁定。

需要定義一個結(jié)構(gòu)體地址,以便于將port的主機字節(jié)序轉(zhuǎn)化成網(wǎng)絡(luò)字節(jié)序

 struct sockaddr_in myaddr; //地址結(jié)構(gòu)體

bind函數(shù)

 bind(sockfd,(struct sockaddr*)&myaddr,sizeof(serveraddr))

3.listen監(jiān)聽,將接收到的客戶端連接放入隊列

 listen(sockfd,8) //第二個參數(shù)是隊列長度

4.調(diào)用accept函數(shù),從隊列獲取請求,返回socket描 述符

如果無請求,將會阻塞,直到獲得連接

 int fd=accept(sockfd, NULL,NULL);//這邊采用默認(rèn)參數(shù)

5.調(diào)用read/write進行雙向通信

6.關(guān)閉accept返回的socket

 close(scokfd);

下面放出完整代碼

/*服務(wù)器*/
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <strings.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <netinet/in.h>
int main()
{
int sockfd = socket(AF_INET, SOCK_STREAM, 0);//創(chuàng)建套接字
if (sockfd < 0)
{
perror("socket");
return -1;
} //創(chuàng)建失敗的錯誤處理
printf("socket.............. "); //成功則打印“socket。。。?!?br>
struct sockaddr_in myaddr; //創(chuàng)建“我的地址”結(jié)構(gòu)體
memset(&myaddr, 0, sizeof(myaddr)); //對內(nèi)存清零(保險起見)
myaddr.sin_family = AF_INET; //選擇IPV4地址類型
myaddr.sin_port = htons(8888); //選擇端口號
myaddr.sin_addr.s_addr = inet_addr("192.168.3.169"); //選擇IP地址

if (0 > bind(sockfd, (struct sockaddr*)&myaddr, sizeof(myaddr)))//綁定套接字
{
perror("bind");
return -1;
}
printf("bind.......... ");

if (0 > listen(sockfd, 8))//調(diào)用listen對指定端口進行監(jiān)聽
{
perror("listen");
return -1;
}
printf("listen............ ");

int connfd = accept(sockfd, NULL, NULL);//使用accept從消息隊列中獲取請求
if (connfd < 0)
{
perror("accept");
return -1;
}
printf("accept.............. ");
char buf[100];//定義一個數(shù)組用來存儲接收到的數(shù)據(jù)
int ret;
while (1)
{
memset(buf, 0, sizeof(buf));
ret = read(connfd, buf, sizeof(buf));
if (0 > ret)
{
perror("read");
break;
}//執(zhí)行while循環(huán)讀取數(shù)據(jù),當(dāng)
else if (0 == ret)
{
printf("write close! ");
break;
}
printf("recv: ");
fputs(buf, stdout);//打印接收到的數(shù)據(jù)
}
close(sockfd);//關(guān)閉套接字
close(connfd);//斷開連接
return 0;
}
/*客戶端*/(具體功能和服務(wù)器一樣,所以不再加注釋)
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <strings.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
int main()
{
int sockfd;
if (0 > (sockfd = socket(AF_INET, SOCK_STREAM, 0)))
{
perror("socket");
return -1;
}
printf("socket........... ");

struct sockaddr_in srv_addr;
memset(&srv_addr, 0, sizeof(srv_addr));
srv_addr.sin_family = AF_INET;
srv_addr.sin_port = htons(8888);
srv_addr.sin_addr.s_addr = inet_addr("192.168.3.169");
if (0 > connect(sockfd, (struct sockaddr*)&srv_addr, sizeof(srv_addr)))
{
perror("connect");
return -1; //exit //pthread_exit
}
printf("connect.............. ");
char buf[100];
int ret;
while (1)
{
printf("send: ");
fgets(buf, sizeof(buf), stdin);
ret = write(sockfd, buf, sizeof(buf));
if (ret < 0)
{
perror("write");
break;
}
if (strncmp(buf, "quit", 4) == 0)
break;
}
close(sockfd);
return 0;
}

    本站是提供個人知識管理的網(wǎng)絡(luò)存儲空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點。請注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購買等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請點擊一鍵舉報。
    轉(zhuǎn)藏 分享 獻花(0

    0條評論

    發(fā)表

    請遵守用戶 評論公約

    類似文章 更多