linux soket id 复位,Linux下Socket编程中注意的几个问题及要点总结

概述:在学习linux下socket编程中,我遇到了一些问题和本身感受比较重要的一些知识点,这边作一个总结,看成是学习笔记,也算是一个记录,以便之后翻阅吧。

问题及要点:

(1)bind error : Address already in use .地址绑定错误问题。

(2)大端小端字节序,网络字节序。

(3)URL(域名)转化问题。

(4)读写函数read(),write()返回值问题。

(5)非阻塞下connect成功失败问题。

分析问题及要点:

1,bind error : Address already in use .地址绑定错误问题。

在咱们编写TCP服务器过程,咱们都要绑定一个地址端口去监听,当咱们强制结束咱们的服务器而后再启动时,有时候就会报出地址绑定出错的问题。这是由于系统没有当即释放端口。

咱们能够经过setsockopt()函数,开启端口重用机制,便可解决此问题。如下一段伪代码已供参考。html

/*设置端口处于像TIMEWAIT状态可当即重复使用,不设置程序重启则可能会出现bind失败*/

int reuse = 1;

if (-1 == setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse))) {

printf("setsockopt error\n");

return 1;

}

2,大端小端字节序,网络字节序。

大端字节序:低位地址存高位数据,高位地址存低位数据。

小端字节序:低位地址存低位数据,高位地址存高位数据。linux

如一个整型数据0x12345678,大端字节序从低位地址到高位地址存储的数据分别为0x12,0x34,0x56,0x78.而小端字节序则相反,依次为0x78,0x56,0x34,0x12.web

为什么又多出一个网络字节序呢,其实网络字节序就是大端字节序,由于设备平台不一样,各个设备平台字节序不一样,有的大端有的小端,为了统一,就制定了网络字节序,因此咱们通常在TCP编程时都是将主机的字节序统一转化为网络字节序。如htonl(),htons()函数。

这有一篇文章,写的还比较详细能够参考学习。面试

3,URL(域名)转化问题。

在咱们访问网页时,通常都是输入域名,好比咱们访问百度网页时通常都是输入”www.baidu.com”,输入的都是域名而不是IP,由于IP对于咱们人类来讲不容易记忆,因此咱们在socket客户端编程中,如何将域名转化为咱们可用的IP地址呢,咱们能够调用现成的API,getaddrinfo()函数。

这里是我本身写的一段伪代码,供参考。编程

#define URL "www.baidu.com"

/*getaddrinfo的使用,域名解析*/

struct addrinfo hints,*result;

memset(&hints, 0, sizeof(hints));

hints.ai_family = AF_INET;

hints.ai_socktype = SOCK_STREAM;

if (0 != getaddrinfo(URL,NULL,&hints,&result)) {

printf("get addrinfo error\n");

return 1;

}

/*打印解析的IP*/

char server_ip[20];

inet_ntop(AF_INET,&((struct sockaddr_in *)result->ai_addr)->sin_addr.s_addr,server_ip);

printf("server ip:%s\n",server_ip);

4,读写函数read(),write()返回值问题。

read():

返回值大于0:正常,表示读到的字节数

返回值等于0:出错,表示对端已关闭链接

返回值小于0:分两种状况,阻塞模式下,表示出错。非阻塞模式下,看errno值,在errno值等于EINTR,EAGAIN,EWOULDBLOCK任何一个,表示链接正常,可忽略,不然出错。服务器

write():

返回值大于0:正常,表示写的字节数

返回值等于0:出错,表示对端已关闭链接

返回值小于0:分两种状况,阻塞模式下,表示出错。非阻塞模式下,看errno值,在errno值等于EINTR,EAGAIN,EWOULDBLOCK任何一个,表示链接正常,可忽略,不然出错。网络

下面试read和write非阻塞状况下的一段代码,供参考。socket

static int write_data(int sockfd, char *data)

{

int bytes = write(sockfd,data,strlen(data)+1);

if (bytes < 0) {

if (EINTR != errno && EAGAIN != errno && EWOULDBLOCK != errno) {

printf("write error\n");

return 1;

}

} else if (0 == bytes) {

printf("socket is close,write error\n");

return 1;

} else {

printf("write success\n");

}

return 0;

}

static int read_data(int sockfd)

{

char buffer[1024];

memset(buffer,0,sizeof(buffer));

int bytes = read(sockfd,buffer,256);

if (bytes < 0 ) {

if (EINTR != errno && EAGAIN != errno && EWOULDBLOCK != errno) {

printf("read error\n");

return 1;

}

} else if (0 == bytes) {

printf("socket is close,read error\n");

return 1;

} else {

buffer[bytes] = '\0';

printf("recvdata:%s\n",buffer);

}

return 0;

}

5,非阻塞下connect成功失败问题。

在socket客户端编程中,咱们都须要链接服务器,在非阻塞模式下,connect会当即返回值,那么咱们此时该如何判断链接服务器是否成功呢。咱们经过select函数判断套接字sockfd是否可读可写,及getsockopt()函数来进行判断。

链接成功状况:

(1)sockfd可写但不可读,表示链接成功。

(2)sockfd可写可读且getsockopt()返回值为0,error值也为0,表示链接成功。

这里是写的一段代码,供参考。svg

static int connect_check(int sockfd)

{

fd_set rd_perim,wr_perim;

int erro;

socklen_t optlen = sizeof(erro);

/*设置超时时间为500ms*/

struct timeval waittime;

waittime.tv_sec = 0;

waittime.tv_usec = 500000;

/*检测sockfd是否可读写*/

FD_ZERO(&rd_perim);

FD_ZERO(&wr_perim);

FD_SET(sockfd,&rd_perim);

FD_SET(sockfd,&wr_perim);

int ret = select(sockfd+1,&rd_perim,&wr_perim,NULL,&waittime);

if (ret < 0) {

printf("select error\n");

return 1;

}

/*connect成功:1,sockfd可写不可读。2,sockfd可写可读,getsockopt返回值为0且erro值也为0*/

if (FD_ISSET(sockfd,&wr_perim)) {

if (!FD_ISSET(sockfd,&rd_perim)) {

printf("connect success\n");

return 0;

} else {

ret = getsockopt(sockfd,SOL_SOCKET,SO_ERROR,&erro,&optlen);

if (0 == ret && 0 == erro) {

printf("connect success\n");

return 0;

} else {

printf("connect timeout\n");

return 1;

}

}

} else {

printf("connect timeout\n");

return 1;

}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值