近两天有个需求,client端需要通过发送一段私有的udp报文协议给server端,server端根据过来的报文来对自己做一些配置。server端在对自己配置成功之后,需要给client端发送配置成功的例子。同时, client端能够读取server端当前的配置信息。
报文是udp报文,我最开始想当然的以为udp client也是可以接收server发过来的信息的,后来发现udp应该是属于单向传输的。也就是说,如果client想要接收server发过来的信息,那么也需要bind一个端口,然后接受来着这个端口的数据,这样一来client也就变成了另外一个udp会话的server。
但事实是,因为client之前给server发消息,系统给client临时分配了一个端口,是否可以仍然利用这个端口来进行server发给client消息这个方向的会话。
答案是可以, 看下面的例子:
-
server端的代码:
-
-
/*************************************************************************
-
> File Name: server.c
-
> Author: SongLee
-
************************************************************************/
-
#include<sys/types.h>
-
#include<sys/socket.h>
-
#include<unistd.h>
-
#include<netinet/in.h>
-
#include<arpa/inet.h>
-
#include<stdio.h>
-
#include<stdlib.h>
-
#include<errno.h>
-
#include<netdb.h>
-
#include<stdarg.h>
-
#include<string.h>
-
-
#define SERVER_PORT 8000
-
#define BUFFER_SIZE 1024
-
#define FILE_NAME_MAX_SIZE 512
-
-
int main()
-
{
-
/* 创建UDP套接口 */
-
struct sockaddr_in server_addr;
-
bzero(&server_addr, sizeof(server_addr));
-
server_addr.sin_family = AF_INET;
-
server_addr.sin_addr.s_addr = htonl(INADDR_ANY);
-
server_addr.sin_port = htons(SERVER_PORT);
-
-
/* 创建socket */
-
int server_socket_fd = socket(AF_INET, SOCK_DGRAM, 0);
-
if(server_socket_fd == -1)
-
{
-
perror("Create Socket Failed:");
-
exit(1);
-
}
-
-
/* 绑定套接口 */
-
if(-1 == (bind(server_socket_fd,(struct sockaddr*)&server_addr,sizeof(server_addr))))
-
{
-
perror("Server Bind Failed:");
-
exit(1);
-
}
-
-
char bufReceived[50] = {"I have received you message"};
-
/* 数据传输 */
-
while(1)
-
{
-
/* 定义一个地址,用于捕获客户端地址 */
-
struct sockaddr_in client_addr;
-
socklen_t client_addr_length = sizeof(client_addr);
-
-
/* 接收数据 */
-
char buffer[BUFFER_SIZE];
-
bzero(buffer, BUFFER_SIZE);
-
if(recvfrom(server_socket_fd, buffer, BUFFER_SIZE,0,(struct sockaddr*)&client_addr, &client_addr_length) == -1)
-
{
-
perror("Receive Data Failed:");
-
exit(1);
-
}
-
-
/* 从buffer中拷贝出file_name */
-
char file_name[FILE_NAME_MAX_SIZE+1];
-
bzero(file_name,FILE_NAME_MAX_SIZE+1);
-
strncpy(file_name, buffer, strlen(buffer)>FILE_NAME_MAX_SIZE?FILE_NAME_MAX_SIZE:strlen(buffer));
-
printf("filename is %s\n", file_name);
-
-
if(sendto(server_socket_fd, bufReceived, 50,0,(struct sockaddr*)&client_addr,sizeof(client_addr)) < 0)
-
{
-
perror("Send File Name Failed:");
-
exit(1);
-
}
-
-
}
-
close(server_socket_fd);
-
return 0;
-
}
-
client端的代码:
-
-
/*************************************************************************
-
> File Name: client.c
-
> Author: SongLee
-
************************************************************************/
-
#include<sys/types.h>
-
#include<sys/socket.h>
-
#include<unistd.h>
-
#include<netinet/in.h>
-
#include<arpa/inet.h>
-
#include<stdio.h>
-
#include<stdlib.h>
-
#include<errno.h>
-
#include<netdb.h>
-
#include<stdarg.h>
-
#include<string.h>
-
-
#define SERVER_PORT 8000
-
#define BUFFER_SIZE 1024
-
#define FILE_NAME_MAX_SIZE 512
-
-
int main()
-
{
-
int ret;
-
int i;
-
struct sockaddr_in self_addr;
-
socklen_t socketLen = sizeof(self_addr);
-
bzero(&self_addr, sizeof(self_addr));
-
char bufReceive[BUFFER_SIZE];
-
bzero(bufReceive, BUFFER_SIZE);
-
socklen_t self_addr_length = sizeof(self_addr);
-
-
/* 服务端地址 */
-
struct sockaddr_in server_addr;
-
bzero(&server_addr, sizeof(server_addr));
-
server_addr.sin_family = AF_INET;
-
server_addr.sin_addr.s_addr = inet_addr("1.1.1.1");
-
server_addr.sin_port = htons(SERVER_PORT);
-
-
/* 创建socket */
-
int client_socket_fd = socket(AF_INET, SOCK_DGRAM, 0);
-
if(client_socket_fd < 0)
-
{
-
perror("Create Socket Failed:");
-
exit(1);
-
}
-
-
#if 1
-
if(-1 == (bind(client_socket_fd,(struct sockaddr*)&self_addr,sizeof(self_addr))))
-
{
-
perror("Server Bind Failed:");
-
exit(1);
-
}
-
#endif
-
-
ret = getsockname(client_socket_fd,(struct sockaddr*)&self_addr,&socketLen);
-
printf("\r\n 1 ret is : %d ",ret);
-
printf("\r\n 1 my ip is : 0x%x",self_addr.sin_addr.s_addr);
-
printf("\r\n 1 my l4port is : %d",self_addr.sin_port);
-
printf("\r\n 1 my client_socket_fd is : %d \r\n",client_socket_fd);
-
-
/* 输入文件名到缓冲区 */
-
char file_name[FILE_NAME_MAX_SIZE+1];
-
bzero(file_name, FILE_NAME_MAX_SIZE+1);
-
printf("Please Input File Name On Server:\t");
-
scanf("%s", file_name);
-
-
char buffer[BUFFER_SIZE];
-
bzero(buffer, BUFFER_SIZE);
-
strncpy(buffer, file_name, strlen(file_name)>BUFFER_SIZE?BUFFER_SIZE:strlen(file_name));
-
-
//for(i=0;i<5;i++){
-
/* 发送文件名 */
-
-
if(sendto(client_socket_fd, buffer, BUFFER_SIZE,0,(struct sockaddr*)&server_addr,sizeof(server_addr)) < 0)
-
{
-
perror("Send File Name Failed:");
-
exit(1);
-
}
-
-
ret = getsockname(client_socket_fd,(struct sockaddr*)&self_addr,&socketLen);
-
printf("\r\n 2 ret is : %d ",ret);
-
printf("\r\n 2 my ip is : 0x%x ",self_addr.sin_addr.s_addr);
-
printf("\r\n 2 my l4port is : %d ",self_addr.sin_port);
-
printf("\r\n 2 my client_socket_fd is : %d \r\n",client_socket_fd);
-
printf("\r\n");
-
//}
-
-
#if 0
-
if(-1 == (bind(client_socket_fd,(struct sockaddr*)&self_addr,sizeof(self_addr))))
-
{
-
perror("Server Bind Failed:");
-
exit(1);
-
}
-
#endif
-
-
if(recvfrom(client_socket_fd, bufReceive, BUFFER_SIZE,0,(struct sockaddr*)&self_addr, &self_addr_length) == -1)
-
{
-
perror("Receive Data Failed:");
-
exit(1);
-
}
-
printf("\r\n %s \r\n",bufReceive);
-
-
close(client_socket_fd);
-
return 0;
-
}
这两段代码, 实现了client给server发送一个文件名, 然后server给client回复一个消息的功能。 主要看client端, 发送报文和接受报文, 用的是同一个socket建立起来的fd和临时端口。注意client代码中, 两个“#if 1 ... #endif和#if 0 ... #endif"的地方, 如果把上面那个注掉,下面那个放开, 会发现bind失败。 如果把下面那个注掉,上面那个放开, 则可以正常运行,而且这时候getsockname可以正常获取到fd的临时端口号, 所以, 我怀疑临时端口号就是在bind这个函数执行的时候会分配的。 但是, 把bind放在一次sendto函数之后, 想再去bind就bind不上了, 因此临时端口号的分配也有可能会发生在sendto执行的时候, 分配好了之后再去bind就bind不上了。
因此, 如果有udp做交互的时候, client端的bind, 需要在sendto之前就执行一次, 否则会bind失败。
阅读(1865) | 评论(0) | 转发(0) |