TCP与UDP基础

思维导图:

TCP:

服务器

cs 复制代码
#include<myhead.h>
#define SER_IP "192.168.252.163"
#define SER_PORT 6666
int main(int argc, const char *argv[])
{
	//1、创建用于监听的套接字
	int sfd=-1;
	sfd=socket(AF_INET,SOCK_STREAM,0);
	//参数1:表示使用的是ipv4的通信
	//参数2:表示使用tcp通信模型
	//参数3:表示前面已经特定了通信协议
	if(sfd==-1)
	{
		perror("socket error");
		return -1;
	}
	printf("sfd=%d\n",sfd);
	//2、绑定ip地址和端口号
	//2.1、填充地址信息结构体
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;//地址族
	sin.sin_port=htons(SER_PORT);//端口号
	sin.sin_addr.s_addr=inet_addr(SER_IP);//IP地址
	//2.2绑定
	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))==-1)
	{
		perror("bind error");
		return -1;
	}
	printf("bind success\n");

	//3、启动监听,允许客户端连接
	if(listen(sfd,128)==-1)
	{
		perror("listen error");
		return -1;
	}
	printf("listen success\n");
	
	//4、当有客户端发来连接请求后,创建新的用于通信的套接字
	//如果不想接收客户端地址信息结构体,则无需传入参数2和参数3
	//如果想要获取客户端共享信息结构体,则需要传入相关参数
	struct sockaddr_in cin;//用于接收客户端共享信息结构体
	socklen_t socklen=sizeof(cin);//用于接收客户端共享信息的大小
	int newfd=accept(sfd,(struct sockaddr*)&cin,&socklen);
	if(newfd==-1)
	{
		perror("accept error");
		return -1;
	}
	printf("newfd=%d 您有新的客户已经上线\n",newfd);
	printf("客户端IP:%s,端口号为:%d\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port));

	//5、通信套接字与客户端进行收发
	char rbuf[128]="";
	while(1)
	{
		//清空容器
		bzero(rbuf,sizeof(rbuf));
		//从套接字中读取数据
		int res=recv(newfd,rbuf,sizeof(rbuf),0);
		if(res==0)
		{
			printf("客户端已经下线\n");
			break;
		}
		printf("[%s:%d]:%s\n",inet_ntoa(cin.sin_addr),ntohs(cin.sin_port),rbuf);

		//将收到的字符加上其他字符回过去
		strcat(rbuf,"*-*");
		send(newfd,rbuf,strlen(rbuf),0);
	}
	//6、关闭套接字
	close(newfd);
	close(sfd);

	return 0;
}

客户端

cs 复制代码
#include<myhead.h>
#define SER_IP "192.168.252.163"
#define SER_PORT 6666
int main(int argc, const char *argv[])
{
	//1、创建用于通信的套接字文件描述符
	int cfd=socket(AF_INET,SOCK_STREAM,0);
	if(cfd==-1)
	{
		perror("socket error");
		return -1;
	}
	printf("cfd=%d\n",cfd);

	//2、绑定(非必须)
	//3、连接服务器
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;
	sin.sin_port=htons(SER_PORT);
	sin.sin_addr.s_addr=inet_addr(SER_IP);


	if(connect(cfd,(struct sockaddr*)&sin,sizeof(sin))==-1)
	{
		perror("connect error");
		return -1;
	}
	printf("connect success\n");

	//4、数据收发
	char wbuf[128]="";
	while(1)
	{
		bzero(wbuf,sizeof(wbuf));
		printf("请输入>>>");
		fgets(wbuf,sizeof(wbuf),stdin);
		wbuf[strlen(wbuf)-1]=0;

		send(cfd,wbuf,strlen(wbuf),0);
		printf("发送成功\n");
		
		if(strcmp(wbuf,"quit")==0)
		{
			break;
		}

		bzero(wbuf,sizeof(wbuf));
		recv(cfd,wbuf,sizeof(wbuf),0);
		printf("收到的数据为:%s\n",wbuf);
	}

	//5、关闭套接字
	close(cfd);
	return 0;
}

UDP

服务器:

cs 复制代码
#include <myhead.h>
#define SERVER_IP "192.168.101.104"
#define SERVER_PORT 1314
int main(int argc, const char *argv[])
{
	//1、创建套接字
	int sfd=socket(AF_INET,SOCK_DGRAM,0);
	if(sfd==-1)
	{
		perror("socket error");
	    return -1;
	}
	//2、绑定ip和端口号
	//先定义地址信息结构体
	struct sockaddr_in sin;
	sin.sin_family=AF_INET;//地址族
	sin.sin_port=htons(SERVER_PORT);//服务器端口号
	sin.sin_addr.s_addr=inet_addr(SERVER_IP);//服务器ip地址
	//再进行绑定
	if(bind(sfd,(struct sockaddr*)&sin,sizeof(sin))==-1)
	{
		perror("bind error");
		return -1;
	}    
	//3、收发数据
	char rbuf[1024];
	//定义容器接收对端地址信息结构体
    struct sockaddr_in cin;
    int size=sizeof(cin);
    while(1)
    {
        bzero(rbuf, sizeof(rbuf));
        //接收数据
        recvfrom(sfd, rbuf, sizeof(rbuf),0,(struct sockaddr*)&cin, &size);
		printf("收到客户端发来的消息为:%s\n", rbuf);
		if(strcmp(rbuf,"end")==0)
		{
			break;
		}
          //回复已收到
        char wbuf[100]="已收到数据";        
		sendto(sfd, wbuf, sizeof(wbuf), 0, (struct sockaddr*)&cin,size);         
    }
 
    //4、关闭套接字
    close(sfd);
 
	return 0;
}

客户端

cs 复制代码
#include <myhead.h>
#define SERVER_IP "192.168.101.104"
#define SERVER_PORT 1314
int main(int argc, const char *argv[])
{
	//1、创建套接字
	int sfd=socket(AF_INET,SOCK_DGRAM,0);
	if(sfd==-1)
	{
		perror("socket error");
		return -1;
	}
	//2、绑定ip和端口号(可选)   
	
	//定义服务器的地址信息结构体
    struct sockaddr_in sin;
    sin.sin_family = AF_INET;
    sin.sin_port = htons(SERVER_PORT);
    sin.sin_addr.s_addr = inet_addr(SERVER_IP);
	int maxsize=sizeof(sin);
	//3、交互数据
	char rbuf[1024];
	while(1)
	{
		//从终端接的数据存入到rbuf中
		bzero(rbuf,sizeof(rbuf));//清空容器
		printf("请输入:");
        fgets(rbuf, sizeof(rbuf),stdin); 
        rbuf[strlen(rbuf)-1]=0;
		sendto(sfd,rbuf,sizeof(rbuf),0,(struct sockaddr*)&sin, sizeof(sin));//客户端数据发送到服务器
		printf("发送成功\n");
		if(strcmp(rbuf,"end")==0)
		{
			break;
		}
		//客户端从服务器接收数据
		char wbuf[100];
		bzero(wbuf,sizeof(wbuf));
		recvfrom(sfd,wbuf,sizeof(wbuf),0,(struct sockaddr*)&sin,&maxsize);
		printf("收到服务器信息:%s\n",wbuf);
	}
	//4、关闭套接字
	close(sfd);
	return 0;
}
相关推荐
xiaoshiguang318 分钟前
LeetCode:222.完全二叉树节点的数量
算法·leetcode
爱吃西瓜的小菜鸡19 分钟前
【C语言】判断回文
c语言·学习·算法
别NULL21 分钟前
机试题——疯长的草
数据结构·c++·算法
TT哇26 分钟前
*【每日一题 提高题】[蓝桥杯 2022 国 A] 选素数
java·算法·蓝桥杯
yuanbenshidiaos2 小时前
C++----------函数的调用机制
java·c++·算法
唐叔在学习2 小时前
【唐叔学算法】第21天:超越比较-计数排序、桶排序与基数排序的Java实践及性能剖析
数据结构·算法·排序算法
ALISHENGYA2 小时前
全国青少年信息学奥林匹克竞赛(信奥赛)备考实战之分支结构(switch语句)
数据结构·算法
chengooooooo2 小时前
代码随想录训练营第二十七天| 贪心理论基础 455.分发饼干 376. 摆动序列 53. 最大子序和
算法·leetcode·职场和发展
jackiendsc2 小时前
Java的垃圾回收机制介绍、工作原理、算法及分析调优
java·开发语言·算法
游是水里的游4 小时前
【算法day20】回溯:子集与全排列问题
算法