利用不定长的结构体发送socket数据

//Server端代码
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/types.h>
#include<netinet/in.h>
#include<sys/socket.h>


#include<unistd.h>
//使用不定长的结构体发送数据的关键在于:结构体变量必须分配到堆中,而不是栈中
//即只能用malloc或者new来给结构体分配空间
typedef struct Node
{
	int nodeSize;
	int bufSize;
	char buf[0];             //用0字节表示该结构体不定长
} Node;

int main()
{
	int sockfd,new_fd;
	struct sockaddr_in server_addr;
	struct sockaddr_in client_addr;

	if((sockfd = socket(AF_INET,SOCK_STREAM,0)) == -1)
	{
		printf("socket error!\n");
		exit(1);
	}
	server_addr.sin_family = AF_INET;
	server_addr.sin_port = htons(5050);
	server_addr.sin_addr.s_addr = INADDR_ANY;
	bzero(&(server_addr.sin_zero),8);

	if(bind(sockfd,(struct sockaddr *)&server_addr,sizeof(server_addr)) == -1)
	{
		printf("bind error!\n");
		exit(1);
	}
	if(listen(sockfd,100) == -1)
	{
		printf("listen error!\n");
		exit(1);
	}
	printf("Server is listening:\n");
	while(1)
	{
		int sin_size = sizeof(struct sockaddr_in);
		if((new_fd = accept(sockfd,(struct sockaddr *)&client_addr,(socklen_t *)&sin_size)) == -1)
		{
			printf("accept error!\n");
			exit(1);
		}
		int recvSize = 0;//data total length
		if(recv(new_fd,(char *)&recvSize,sizeof(recvSize),0) == -1)
		{
			printf("recv error!\n");
			exit(1);
		}
		int leftSize = sizeof(char) * (recvSize - sizeof(int));
		char *dataBuf = (char *)malloc(leftSize);
		memset(dataBuf,0,leftSize);
		if(recv(new_fd,dataBuf,leftSize,0) == -1)
		{
			printf("recv error!\n");
			exit(1);
		}
		close(new_fd);
		Node *p = (Node *)malloc(sizeof(char) * recvSize);
		p->nodeSize = recvSize;
		memcpy((char *)(&p->bufSize),dataBuf,leftSize);
		printf("nodeSize = %d\n",p->nodeSize);
		printf("bufSize = %d\n",p->bufSize);
		printf("buf = %s\n",p->buf);
	}

	return 0;
}

//Client端代码

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<sys/types.h>
#include<arpa/inet.h>
#include<netinet/in.h>
#include<sys/socket.h>


#include<unistd.h>

typedef struct Node
{
	int nodeSize;
	int bufSize;
	char buf[0];
} Node;

int main()
{
	int sockfd,new_fd;
	struct sockaddr_in server_addr;

	if((sockfd = socket(AF_INET,SOCK_STREAM,0)) == -1)
	{
		printf("socket error!\n");
		exit(1);
	}
	server_addr.sin_family = AF_INET;
	server_addr.sin_port = htons(5050);
	server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
	bzero(&(server_addr.sin_zero),8);

	if(connect(sockfd,(struct sockaddr *)&server_addr,sizeof(server_addr)) == -1)
	{
		printf("connect error!\n");
		exit(1);
	}
	printf("Client is connecting\n");
	const char *tmp = "this is a test!";
	int tmpLen = strlen(tmp);
	Node *dataBuf = (Node *)malloc(sizeof(Node) + tmpLen + 1);
	dataBuf->nodeSize = sizeof(Node) + tmpLen + 1;
	dataBuf->bufSize = tmpLen;
	memset(dataBuf->buf,0,tmpLen + 1);
	memcpy(dataBuf->buf,tmp,tmpLen + 1);
	printf("nodeSize = %d\n",dataBuf->nodeSize);
	printf("bufSize = %d\n",dataBuf->bufSize);
	printf("buf = %s\n",dataBuf->buf);
	if(send(sockfd,(char *)dataBuf,dataBuf->nodeSize,0) == -1)
	{
		printf("send error!\n");
		exit(1);
	}
	close(sockfd);
	return 0;
}


  • 2
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
下面是一个接收不同数据的示例代码,注释已经在代码中给出: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/socket.h> #include <arpa/inet.h> #include <unistd.h> #define MAX_BUF_SIZE 1024 int main(int argc, char *argv[]) { int sockfd; struct sockaddr_in server_addr; char buf[MAX_BUF_SIZE]; int len, total_len = 0; // 创建socket if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) { perror("socket"); exit(EXIT_FAILURE); } // 设置server_addr结构体 memset(&server_addr, 0, sizeof(server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = inet_addr(argv[1]); server_addr.sin_port = htons(atoi(argv[2])); // 连接服务器 if (connect(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) { perror("connect"); exit(EXIT_FAILURE); } // 接收数据 while ((len = recv(sockfd, buf, MAX_BUF_SIZE, 0)) > 0) { total_len += len; // 处理接收到的数据 printf("Received %d bytes: %s\n", len, buf); // 如果接收到的数据度超过了预设的阈值,停止接收数据 if (total_len > 1024) { break; } } // 关闭socket close(sockfd); return 0; } ``` 在以上代码中,我们使用了recv()函数来接收数据,该函数的参数如下: ```c ssize_t recv(int sockfd, void *buf, size_t len, int flags); ``` 其中,sockfd是socket文件描述符,buf是用来存储接收到的数据的缓冲区,len是缓冲区的大小,flags是接收数据时的选项,一般设为0即可。 我们每次调用recv()函数时,都会接收到一定度的数据,因此需要在循环中不断接收数据,直到接收到的数据度超过了预设的阈值。在本例中,我们将阈值设为1024字节,即MAX_BUF_SIZE的大小。 总之,只要不断接收数据,累加接收到的数据度,就可以实现接收不同度的数据

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值