一:问题:刚开始学习Linux网络编程,第一次使用bind函数出现Cannot assign requested address,错误。在网上找了很多帖子但是都没有解决。
二:
问题一:也就是网上大部分说的:客户端频繁的连服务器,由于每次连接都在很短的时间内结束,导致很多的TIME_WAIT,以至于用光了可用的端 口号,所以新的连接没办法绑定端口,即“Cannot assign requested address”。是客户端的问题不是服务器端的问题。
方法一:通过netstat -nap查看当前的状态,的确看到很多TIME_WAIT状态的连接。可以通过pid 用命令 kill -9 pid,杀死处于TIME_WAIT状态的进程。
问题二:因为我的测试是以window当服务器,Ubuntu当客户端,所以需
要用到桥接技术,让两个系统在一个网段里面。那么出现bind的错误很有可能是你的网络的问题。我自己在测试的时候,试了很多种方法,还能ping通,但是网络这一块还是有点问题,还是出现bind错误。
解决二:在虚拟机里面,①编辑->虚拟网络编辑器->还原默认设置。②完成后重新选择,网络桥接模式。③进入Ubuntu,命令ifconfig(查看网卡),sudo ifconfig ens33 up (算是打开网卡吧我也不懂),sudo dhclient(动态分配一个IP)。这样就欧克了,就会发现,在同一个网段。就可以使用bind了。
下面是我写的简单代码,看一下吧。
服务器端:
/**********
1、在服务器端接受消息时知道对方的IP和端口
提示:
bind出现:Address already in use 方法:netstat -nap查看 用kill -9 pid杀死
2、利用TCP实现文件传输
**********/
#include
#include /* See NOTES */
#include
#include
#include
#include
#include
#include
#include
#include
int main(void)
{
//建立套接字
int server_fd=socket(AF_INET,SOCK_STREAM,0);
if(server_fd==-1)
{
perror("socket:");
return -1;
}
//绑定本机的IP和端口
struct sockaddr_in saddr;
saddr.sin_family=AF_INET;
saddr.sin_port=htons(7777); //设置端口
saddr.sin_addr.s_addr=inet_addr("192.168.199.205"); //点分十进制-》换成32位的网络字节序二进制值
int ret=bind(server_fd,(struct sockaddr *)&saddr,sizeof(saddr));
if(ret==-1)
{
perror("bind:");
return -1;
}
//设置监听套接字(设置同时来链接数)
ret=listen(server_fd,4); //server_fd设置为监听套接字
if(ret==-1)
{
perror("listen:");
return -1;
}
//等待对端链接
struct sockaddr_in client_buf;
socklen_t len = sizeof(client_buf);
int client_fd = accept(server_fd,(struct sockaddr *)&client_buf,&len); //成功返回对端的文件描述符
if(-1 == client_fd)
{
perror("accept:");
return -1;
}
//显示对端的ip和端口
printf("对端的ip为:%s\n",inet_ntoa(client_buf.sin_addr));
printf("对端的port为:%d\n",ntohs(client_buf.sin_port));
//文件操作
int fd=open("b.txt",O_RDWR | O_CREAT,0777);
if(-1 == fd)
{
printf("open err!\n");
return -1;
}
char buf[20] = {
0};
while(1)
{
//第一题
/*bzero(buf, 20);
read(client_fd, buf, 20);
printf("buf:%s\n", buf);
*/
//第二题
bzero(buf, 20);
read(client_fd, buf, 20);
printf("buf:%s\n", buf);
if(!strcmp(buf, "exit"))
break;
write(fd,buf,strlen(buf));
}
close(client_fd);
close(server_fd);
}
客户端:
/**********
1、在服务器端接受消息时知道对方的IP和端口
提示:
bind出现:Address already in use 方法:netstat -nap查看 用kill -9 pid杀死
2、利用TCP实现文件传输
**********/
#include
#include /* See NOTES */
#include
#include
#include
#include
#include
#include
#include
#include
#include
int main(void)
{
//建立套接字
int client_fd = socket(AF_INET, SOCK_STREAM, 0);
if(-1 == client_fd)
{
perror("socket");
return -1;
}
//链接服务器
struct sockaddr_in saddr;
bzero(&saddr, sizeof(saddr));
saddr.sin_family = AF_INET;
saddr.sin_port = htons(7777);
saddr.sin_addr.s_addr = inet_addr("192.168.199.205");
int ret = connect(client_fd, (struct sockaddr *)&saddr, sizeof(saddr));
if(-1 == ret)
{
perror("connect");
return -1;
}
//文件操作
int fd=open("a.txt",O_RDWR | O_CREAT,0777);
if(-1 == fd)
{
printf("open err!\n");
return -1;
}
char fd_buf[20]={
0};
while(1)
{
bzero(fd_buf,20);
ret = read(fd, fd_buf, 20); //读文件到缓冲区
if(ret==0)
{
write(client_fd,"exit", 20);
break;
}
else
{
printf("fd_buf%s\n",fd_buf);
write(client_fd, fd_buf, 20);
}
/*bzero(fd_buf, 20);
scanf("%s", fd_buf);
write(client_fd, fd_buf, 20);
printf("fd_buf:%s\n", fd_buf);
if(!strcmp(fd_buf, "exit"))
break;*/
}
close(client_fd);
}