第三节、验证TCP传输的数据没有数据边界

为验证这一点,需要让write函数的调用次数不同于read函数的调用次数。因此,在客户端分别调用read函数以接收服务器端发送的全部数据。

tcp_server.c

/*
面向连接的套接字(SOCK_STREAM)

1.传输过程中数据不会消失
2.按序传输数据
3.传输的数据不存在数据边界

总结:可靠的、按序传递的、基于字节的面向连接的数据传输方式


面向消息的套接字(SOCK_DGRAM)

1.强调快速传输而非传输顺序
2.传输的数据可能丢失也可能损毁
3.传输的数据有数据边界
4.限制每次传输的数据大小

总结:不可靠的、不按序传递的、以数据的高速传输为目的的套接字
*/



#include 
#include 
#include 
#include 
#include 
#include 

void error_handling(const char* message);

int main(int argc, char const *argv[])
{
     
	int serv_sock;
	int clnt_sock;
	struct sockaddr_in serv_addr;
	struct sockaddr_in clnt_addr;
	socklen_t clnt_addr_size;

	char message[] = "Hello world!";

	if(argc != 2)
	{
     
		printf("Usage : %s \n",argv[0] );
		exit(1);
	}

	serv_sock = socket(PF_INET,SOCK_STREAM,0); //创建套接字
	if(serv_sock == -1)
		error_handling("socket() error");

	memset(&serv_addr,0,sizeof(serv_addr));
	serv_addr.sin_family = AF_INET;
	serv_addr.sin_addr.s_addr = htonl(INADDR_ANY);
	serv_addr.sin_port = htons(atoi(argv[1]));

	if(bind(serv_sock,(struct sockaddr*)& serv_addr,sizeof(serv_addr)) == -1)//调用bind函数分配ip和端口
		error_handling("bind() error");

	if(listen(serv_sock,5) == -1) 		//调用listen函数将套接字转为可接收连接状态
		error_handling("listen() error");

	clnt_addr_size = sizeof(clnt_addr);

	//调用accept函数受理连接请求,如果在没有连接的情况下调用该函数,则不会返回,直到有连接请求为止.
	//意思是这里会一直阻塞,直到有连接请求
	clnt_sock = accept(serv_sock,(struct sockaddr*)&clnt_addr,&clnt_addr_size);
	if(clnt_sock == -1)
		error_handling("accept() error");

	//打印连接上来的客户端ip port
	printf("client ip:%s,port:%d\n",inet_ntoa(clnt_addr.sin_addr),ntohs(clnt_addr.sin_port));

	write(clnt_sock,message,sizeof(message));
	close(clnt_sock);
	close(serv_sock);

	return 0;
}

void error_handling(const char* message)
{
     
	fputs(message,stderr);
	fputc('\n',stderr);
	exit(1);
}

tcp_client.c

#include 
#include 
#include 
#include 
#include 
#include 

void error_handling(const char* message);

int main(int argc, char const *argv[])
{
     
	if (argc != 3)
	{
     
		printf("Usage : %s  \n",argv[0] );
		exit(1);
	}

	int sock = socket(PF_INET,SOCK_STREAM,0);
	if(sock == -1)
		error_handling("socket() error");

	struct sockaddr_in serv_addr;
	memset(&serv_addr,0,sizeof(serv_addr));
	serv_addr.sin_family = AF_INET;
	serv_addr.sin_addr.s_addr = inet_addr(argv[1]);
	serv_addr.sin_port = htons(atoi(argv[2]));

	if(connect(sock,(struct sockaddr*)&serv_addr,sizeof(serv_addr)) == -1)
		error_handling("connect() error");

	int str_len = 0;
	int idx = 0,read_len = 0;
	char message[30];
	while(read_len = read(sock,&message[idx++],1)) //read_len = 0时 证明读到末尾
	{
     
		if(read_len == -1)
			error_handling("read() error");

		str_len += read_len;
	} 

	printf("message from server: %s\n",message );
	printf("Function read call count: %d\n",str_len );

	close(sock);

	return 0;
}

void error_handling(const char* message)
{
     
	fputs(message,stderr);
	fputc('\n',stderr);
	exit(1);
}

你可能感兴趣的:(网络编程TCP/IP)