1.TCP流程圖
2.TCP編程
服務(wù)器
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <string.h>
#include <errno.h>
//分別打印錯誤信息,函數(shù)名和行號到標準錯誤流中
#define ERR_MSG(msg) do{\
fprintf(stderr,"__%s__ __%s__ __%d__",__FILE__,__func__,__LINE__);\
perror(msg);\
}while(0)//循環(huán)只是為了去括號,無任何意義
#define IP "000.000.000.000" //ifconfig,ip根據(jù)自己的去設(shè)置
int main(int argc, char const *argv[])
{
//創(chuàng)建流式套接字
int sfd = socket(AF_INET,SOCK_STREAM,0);
if(sfd < 0)
{
ERR_MSG("socket");
return -1;
}
printf("sock success sfd=%d\n",sfd);
//允許端口快速被覆蓋重用。
int reuse = 1;
if(setsockopt(sfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) < 0)
{
ERR_MSG("setsockopt");
return -1;
}
printf("允許端口快速被覆蓋重用成功\n");
//填充地址信息結(jié)構(gòu)體,真實的地址信息結(jié)構(gòu)體AF_INET:man 7 IP
struct sockaddr_in sin;//定義結(jié)構(gòu)體變量,引出成員并填充地址
sin.sin_family = AF_INET; //這里填ipv4協(xié)議
sin.sin_port = htons(6666);//端口號的網(wǎng)絡(luò)字節(jié)序1024~49151,將端口號轉(zhuǎn)換成網(wǎng)絡(luò)字節(jié)序
sin.sin_addr.s_addr = inet_addr(IP);//將ip地址的點分十進制轉(zhuǎn)換成網(wǎng)絡(luò)字節(jié)序,并存到sin下的sin_addr下的s_addr中
//綁定服務(wù)器的地址信息
//bind函數(shù)可以綁定地址信息到套接字文件中,參數(shù)分別是
//指定要綁定到那個套接字上,填對應(yīng)的文件描述符
//通用地址結(jié)構(gòu)體,真實的地址信息結(jié)構(gòu)體根據(jù)地址族制定,綁定IP和端口號到服務(wù)器套接字上
//真實的地址信息結(jié)構(gòu)體的大小
if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin)) < 0)
{//將sin中得地址信息綁定到sfd套接字文件中(因為sin的變量類型不一樣
//需要強制類型轉(zhuǎn)換),最后可以使用sizeof直接算出sin結(jié)構(gòu)體變量的大小
ERR_MSG("bind");
return -1;
}
printf("bind success __%d__\n",__LINE__);//輸出當(dāng)前行號
//將套接字設(shè)置為被動監(jiān)聽狀態(tài)
if(listen(sfd,128) < 0)
{//將sfd套接字設(shè)置為被監(jiān)聽狀態(tài),允許同時有128個客戶端未完成鏈接
ERR_MSG("listen");
return -1;
}
printf("listen success __%d__\n",__LINE__);//鏈接成功后輸出行號
struct sockaddr_in cin;//存儲客戶端信息,定義存儲地址信息結(jié)構(gòu)體變量
socklen_t addrlen = sizeof(cin);//定義結(jié)構(gòu)體類型存取結(jié)構(gòu)體長度
//超時檢測
//struct timeval tm = {60,0};
//if(setsockopt(sfd,SOL_SOCKET,SO_RCVTIMEO,&tm,sizeof(tm)));
//生成新的文件描述符與客戶端通信
int newfd = accept(sfd,(struct sockaddr*)&cin,&addrlen);
//定義整形變量,接收accept函數(shù)的返回值,從以完成的鏈接的sfd隊列中獲取一個客戶端的
//信息,生成一個新的cin文件描述符,該文件描述符是通信使用的文件描述副
if(newfd < 0)
{
if(11 == errno)
{
printf("time out...\n");
return -1;
}
ERR_MSG("accept");
return -1;
}
printf("[%s:%d] newfd=%d 客戶端鏈接成功 __%d__\n",\
inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),newfd,__LINE__);
//客戶端鏈接成功后輸出,轉(zhuǎn)換后的ip地址和轉(zhuǎn)換后的主機字節(jié)序,和行號
char buf[128] = "";//定義字符型數(shù)組存取數(shù)據(jù)
ssize_t res = 0;//定義結(jié)構(gòu)提變量接收返回值
while(1)
{
bzero(buf,sizeof(buf));//將buf數(shù)組清零
//接收
res = recv(newfd,buf,sizeof(buf),0);//從得到accept返回值的newfd套接字中
//接收緩沖區(qū)的數(shù)據(jù),并存儲在數(shù)組中,用sizeof算出指定讀取字節(jié)最后用0表示以阻塞方式
//讀取
if(res <0)//返回值小于0,讀取失敗
{
ERR_MSG("recv");
return -1;
}
else if(0 == res)//返回值等于0,客戶端下線
{
printf("[%s:%d] newfd=%d 客戶端下線__%d__\n",\
inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),newfd,__LINE__);
//輸出轉(zhuǎn)換后的ip地址和主機字節(jié)序,和新的文件描述符
break;
}
printf("[%s:%d] newfd=%d : %s __%d__\n",\
inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),newfd,buf,__LINE__);
//輸出ip地址和主機字節(jié)序,新的文件描述符和buf中的內(nèi)容
//發(fā)送
strcat(buf,"*_*");//在buf數(shù)組的末尾追加*_*,證明服務(wù)器以收到數(shù)據(jù)
if(send(newfd,buf,sizeof(buf),0) < 0)//如果返回值小于0,發(fā)送失敗
{//send函數(shù)可以將but數(shù)組中的數(shù)據(jù)發(fā)送給accept函數(shù)中得到的文件描述符,
//第三項是大小,第四項0是阻塞發(fā)送
ERR_MSG("send");
return -1;
}
printf("send success __%d__\n",__LINE__);//發(fā)送成功后顯示數(shù)據(jù)
}
//關(guān)閉
if(close(newfd) < 0)//關(guān)閉newfd套接字
{
ERR_MSG("close");//關(guān)閉失敗返回-1,并輸出錯誤內(nèi)容
return -1;
}
if(close(sfd) < 0)//關(guān)閉sfd套接字
{
ERR_MSG("close");
return -1;
}
return 0;
}
客戶端文章來源:http://www.zghlxwxcb.cn/news/detail-708275.html
#include <stdio.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <string.h>
#define ERR_MSG(msg) do{\
fprintf(stderr,"__%s__ __%s__ __%d__",__FILE__,__func__,__LINE__);\
perror(msg);\
}while(0)
#define IP "000.000.000.000" //ip地址根據(jù)自己的去設(shè)置
#define PORT 6666 //端口號 1024~49151
int main(int argc, char const *argv[])
{
//創(chuàng)建流式套接字
int cfd = socket(AF_INET,SOCK_STREAM,0);
//定義整形變量接收socket函數(shù)返回的文件描述副
if(cfd < 0)
{
ERR_MSG("socket");
return -1;
}
printf("create socket success cfd=%d __%d__\n",cfd,__LINE__);
//綁定客戶端自身的地址信息結(jié)構(gòu)體---》非必須綁定
//若客戶端沒有綁定地址信息,
//測操作系統(tǒng)會自動幫客戶端綁定本機IP,以及49151~65535內(nèi)的隨機端口
//填充服務(wù)器的地址信息結(jié)構(gòu)體,給connect函數(shù)使用
//要鏈接哪個服務(wù)器,就填哪個服務(wù)器的地址信息
struct sockaddr_in sin;
sin.sin_family = AF_INET; //這里填ipv4協(xié)議
sin.sin_port = htons(PORT);//端口號的網(wǎng)絡(luò)字節(jié)序1024~49151,將端口號轉(zhuǎn)換成網(wǎng)絡(luò)字節(jié)序
sin.sin_addr.s_addr = inet_addr(IP);//將ip地址的點分十進制轉(zhuǎn)換成網(wǎng)絡(luò)字節(jié)序,并存到sin下的sin_addr下的s_addr中
//鏈接服務(wù)器
if(connect(cfd,(struct sockaddr*)&sin,sizeof(sin)) < 0)//當(dāng)返回值小于0,失敗
{//通過connect函數(shù)鏈接服務(wù)器:將指定要鏈接的文件描述符綁定sin的結(jié)構(gòu)體變量鏈接服務(wù)器,
ERR_MSG("connect");
return -1;
}
printf("connect success __%d__\n",__LINE__);
//成功后返回行號
char buf[128] = "";//定義字符型數(shù)組
ssize_t res = 0;//定義結(jié)構(gòu)體變量,接收返回值
while(1)
{
//從終端獲取數(shù)據(jù)
printf("請輸入>>> ");
fgets(buf,sizeof(buf),stdin);//從標準輸入流中讀取數(shù)據(jù),存入到buf數(shù)組中
buf[strlen(buf)-1] = 0;//將buf的最后一位‘\n’改為0
//發(fā)送
if(send(cfd,buf,sizeof(buf),0) < 0)//返回值小于0,發(fā)送失敗
{//通過cfd套接字文件描述符向緩沖區(qū)發(fā)送buf中的數(shù)據(jù)
ERR_MSG("send");
return -1;
}
printf("send success __%d__\n",__LINE__);//成功輸出行號
//接收
bzero(buf,sizeof(buf));//將buf數(shù)組清零
res = recv(cfd,buf,sizeof(buf),0);
//通過該函數(shù)從該套接字文件描述符中接收數(shù)據(jù),并存入buf數(shù)組中
if(res < 0)//如果返回值小于0
{
ERR_MSG("recv");//接收失敗
return -1;
}
else if(0 == res)//返回之等于0,斷線
{
printf("服務(wù)器離線\n");
break;
}
printf("%s __%d__\n",buf,__LINE__);//接收成功輸出buf數(shù)組中的內(nèi)容和行號
}
//關(guān)閉
if(close(cfd) < 0)//關(guān)閉cfd套接字文件描述符
{
ERR_MSG("close");
return -1;
}
return 0;
}
????????以上就是用c語言搭建的tcp服務(wù)器和客戶端,IP地址的地方可以根據(jù)自己本機的IP地址去修改(在命令提示符中可以使用ifconfig命令查看本機IP地址),端口號用的是6666,也可自己修改,但是IP地址和端口號服務(wù)器和客戶端必須一至。文章來源地址http://www.zghlxwxcb.cn/news/detail-708275.html
到了這里,關(guān)于用C語言搭建TCP服務(wù)器/客戶端的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!