Chinaunix首页 | 论坛 | 博客
  • 博客访问: 748194
  • 博文数量: 370
  • 博客积分: 2334
  • 博客等级: 大尉
  • 技术积分: 3222
  • 用 户 组: 普通用户
  • 注册时间: 2012-05-06 16:56
文章分类

全部博文(370)

文章存档

2013年(2)

2012年(368)

分类:

2012-05-16 13:06:14

++++++APUE读书笔记-16网络通信-06套接字选项++++++

 

6、套接字选项
================================================
 套接字机制提供了两个套接字选项接口来控制套接字的行为。一个接口用来设置选项,另外一个接口用来允许我们请求选项的状态。我们可以获得以及设置三种类型的选项。
 1.通用选项,可以工作在所有的套接字类型。
 2.在套接字层次上面进行管理的选项,但是依赖底部协议的支持。
 3.和每个协议相关的协议选项。
 Single UNIX Specification 只定义了套接字层的选项(上面所提到的前面两项)

 我们可以通过setsockopt函数来设置套接字选项。
 #include
 int setsockopt(int sockfd, int level, int option, const void *val, socklen_t len);
 返回:如果成功返回0,如果错误返回1。
 参数level用来分辨option所应用的协议。如果option是通用套接字层次的选项,那么level设置成SOL_SOCKET。否则level设置成控制option的协议号。例如IPPROTO_TCP用于TCP选项,以及IPPROTO_IP用于IP选项。下面的表中就列出了Single UNIX Specification定义的通用的套接字层次的选项。

      套接字选项
+-------------------------------------------------------------------------------------------------------------+
|    Option     | Type of val argument |                             Description                              |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_ACCEPTCONN | int                  | 返回套接字是否激活用于侦听(只用于getsockopt)。                     |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_BROADCAST  | int                  | 如果*val非0那么广播数据报。                                          |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_DEBUG      | int                  | 如果*val非0那么激活网络驱动的调试。                                  |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_DONTROUTE  | int                  | 如果*val非0,那么忽略通常的路由。                                    |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_ERROR      | int                  | 返回并且清除提交的套接字错误(只用于getsockopt)。                   |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_KEEPALIVE  | int                  | 如果*val非0,那么激活定期活动的消息。                                |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_LINGER     | struct linger        | 如果有未发送的消息存在以及套接字关闭,那么做延迟。                   |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_OOBINLINE  | int                  | 如果*val非0,那么将带外数据嵌入到正常数据中。                        |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_RCVBUF     | int                  | 接收缓存的字节大小。                                                 |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_RCVLOWAT   | int                  | receive调用返回的最小数据字节。                                      |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_RCVTIMEO   | struct timeval       | 套接字receive调用的超时值。                                          |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_REUSEADDR  | int                  | 如果*val非0,那么重复使用bind的地址。                                |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_SNDBUF     | int                  | send缓存中的字节大小。                                               |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_SNDLOWAT   | int                  | 一次send调用传输的最小数据字节量。                                   |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_SNDTIMEO   | struct timeval       | 一个套接字send调用的超时值。                                         |
|---------------+----------------------+----------------------------------------------------------------------|
| SO_TYPE       | int                  | 辨别套接字类型(只在getsockopt中)。                                 |
+-------------------------------------------------------------------------------------------------------------+
 参数val指向一个数据结构或者整数,这取决于option。有些选项是on/off开关。 如果这个整数非0,那么option被激活。如果这个整数是0,那么option不被激活。len参数指定val指向的对象的大小。

 我们可以通过函数getsockopt来获取当前option的值。
 #include
 int getsockopt(int sockfd, int level, int option, void *restrict val, socklen_t *restrict lenp);
 返回:如果成功返回0,如果错误返回1。
 注意lenp参数是一个指向整数的指针。在调用getsockopt之前,我们设置整数为option被拷贝的缓存的大小,如果实际的option大小比这个大小大,那么option就会被截断。如果实际的option大小比size小或者同样大,那么整数会在返回的时候被更新成实际的大小。

 例子
 前面的(initserver)函数当服务进程终止的时候操作失败,然后我们尝试立即重新启动。一般来说,TCP的实现会会阻止我们将同样一个地址绑定,除非超时。SO_REUSEADDR的套接字选项允许我们忽略这个限制,下面的代码就展示了这个特性。

 为了激活SO_REUSEADDR选项,我们设置整数为一个非0的值然后将整数的地址做为val参数传递给setsockopt函数。我们设置len参数为一个表示整数的大小,来表示val所指向的对象的大小。

 服务进程使用地址重用初始化一个将要被使用的套接字末端
 #include "apue.h"
 #include
 #include
 int initserver(int type, const struct sockaddr *addr, socklen_t alen, int qlen)
 {
     int fd, err;
     int reuse = 1;

     if ((fd = socket(addr->sa_family, type, 0)) < 0)
         return(-1);
     if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(int)) < 0) {
         err = errno;
         goto errout;
     }
     if (bind(fd, addr, alen) < 0) {
         err = errno;
         goto errout;
     }
     if (type == SOCK_STREAM || type == SOCK_SEQPACKET) {
         if (listen(fd, qlen) < 0) {
             err = errno;
             goto errout;
         }
     }
     return(fd);

 errout:
     close(fd);
     errno = err;
     return(-1);
 }

参考:

 

 

阅读(350) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~