您的位置:首页 > 编程语言

socket编程之setsockopt和getpeername

2013-10-13 21:01 309 查看
setsockopt  最重要的选项是SOL_SOCKET, SO_REUSEADDR 在UNP中接受比较详细.这里列出来加深下记忆.

(1) SO_REUSEADDR允许启动一个监听服务器并捆绑其众所周知端口,即使以前建立的将该端口作用它们的本地端口的链接仍然存在

如果尝试再次捆绑以前已经捆绑过的端口,会失败.也就是bind调用会失败. 我们只要在 socket和bind之间弄个setsockopt函数,并携带上SO_REUSEADDR选项,

那就可以解决这个问题了. 说白了就是连接了以后还可以连接..  如果没有这个选项的话那只能连接一次哦, 第二次想连接的话就会connect refuse了

(2)SO_REUSADDR允许在一个端口上启动同一服务器的多个实例,只要每个实例捆绑一个不同的本地IP地址即可.

还有些作用,自己还没有用到,所以就不写出来了.详细可以了解UNP(3版) 165 pages

getpeername 这个家伙能把客户端的身份给弄过来,也就是能从某个套接字上获取已经绑定有外地协议地址. 说白了就是可以用这个函数获取客户端的地址和端口.

当然,getpeername使用的时机是在连接以后,也就是getpeername的sockfd参数必须是已经连接套接字的描述符.. 所以getpeername应该在 accetp之后使用.

在介绍两个函数的时候只介绍了它们的作用. 下面贴出运用代码

#include <sys/types.h>
#include <sys/socket.h>
#include <sys/unistd.h>

#include <string.h>
#include <errno.h>
#include <stdio.h>
#include <time.h>

#include <netinet/in.h>
#include <arpa/inet.h>
#include <netinet/in.h>

#define DATE_PORT        9813

#define bzero(ptr, n)    memset(ptr, 0, n)
#define MAXLINE          4096
#define LISTENQ          1024

/********************************************************************************
*  Description:
*   Input Args:
*  Output Args:
* Return Value:
********************************************************************************/
int main (int argc, char **argv)
{
int                     listenfd, connfd;
int                     opt = 1;
struct sockaddr_in      servaddr,cliaddr;
char                    buff[MAXLINE];
time_t                  ticks;
socklen_t               sock_len = (sizeof(struct sockaddr_in));
socklen_t               len;

if( (listenfd = socket(AF_INET,SOCK_STREAM, 0)) < 0)
{
printf("socket error : %s\n", strerror(errno));
}

if(setsockopt(listenfd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof opt) != 0)
{
printf("setsokopt error : %s\n", strerror(errno));
return -1;
}
bzero(&servaddr, sizeof(servaddr));

servaddr.sin_family      = AF_INET;
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
servaddr.sin_port        = htons(DATE_PORT);

if(bind(listenfd, (struct sockaddr *)&servaddr, sizeof(servaddr)) < 0)
{
printf("bind error : %s\n", strerror(errno));
return -1;
}

if( listen(listenfd, LISTENQ) < 0)
{
printf("listen error : %s\n", strerror(errno));
return -1;
}

#if 0

if(-1 == getpeername(listenfd, (struct sockaddr *)&cliaddr, &sock_len))
{
printf("getpeername error: %s\n", strerror(errno));

}
else
{
printf("Add new event: socket[%d] from [%s:%d]\n", listenfd,inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port));
}
#endif

for (; ;)
{
len = sizeof(cliaddr);

if((connfd = accept(listenfd, (struct sockaddr*)&cliaddr, &len)) != 3)
{

printf("Accept new client Error : %s\n", strerror(errno));
return -1;
}
#if 1
if( -1 == getpeername(connfd, (struct sockaddr *)&cliaddr, &sock_len) )
{
printf("getpeername error: %s\n", strerror(errno));
return -1;
}
else
{
printf("Add new event: socket[%d] from [%s:%d]\n", connfd, inet_ntoa(cliaddr.sin_addr), ntohs(cliaddr.sin_port));
}
#endif

ticks = time(NULL);

snprintf(buff, sizeof(buff), "%.24s\r\n", ctime(&ticks));

if(write(connfd, buff, strlen(buff)) < 0)
{
printf("write to buff error : %s\n", strerror(errno));
return -1;
}

if(close(connfd) < 0)
{
printf("close error %s\n", strerror(errno));
return -1;
}
}

}
/* ----- End of main() ----- */
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: