• setsocket、getsocket 函数详解


    背景

    以前用到socket的时候会调用setsocket进行设置,现在整理有关的笔记的时候,重新查阅资料发现有点奇怪,发现大家比较少使用到这个。

    setsocket/getsocket

    #include <sys/types.h>          /* See NOTES */
    #include <sys/socket.h>
    
    int getsockopt(int sockfd, int level, int optname,
                   void *optval, socklen_t *optlen);
                   
    int setsockopt(int sockfd, int level, int optname,
                   const void *optval, socklen_t optlen);
    

    描述:

    获取或者设置与某个套接字关联的选项。选项可能存在于多层协议中,它们总会出现在最上面的套接字层。当操作套接字选项时,选项位于的层和选项的名称必须给出。为了操作套接字层的选项,应该 将层的值指定为SOL_SOCKET。为了操作其它层的选项,控制选项的合适协议号必须给出。例如,为了表示一个选项由TCP协议解析,level应该设定为TCP的协议号。

    参数解析

    sock:将要被设置或者获取选项的套接字。
    level:选项所在的协议层。 level指定控制套接字的层次.可以取三种值:

    • SOL_SOCKET:通用套接字选项.

    • IPPROTO_IP:IP选项.

    • IPPROTO_TCP:TCP选项. 

    optname:需要访问的选项名。(下文中会有一个表)
    optval:值,对于getsockopt(),指向返回选项值的缓冲;对于setsockopt(),指向包含新选项值的缓冲。
    optlen:值长度,对于getsockopt(),作为入口参数时,选项值的最大长度;作为出口参数时,选项值的实际长度。对于setsockopt(),现选项的长度。

    返回值: 成功执行时,返回0。失败返回-1,errno被设为以下的某个值 :

    • EBADF:sock不是有效的文件描述词
    • EFAULT:optval指向的内存并非有效的进程空间
    • EINVAL:在调用 setsockopt()时,optlen无效
    • ENOPROTOOPT:指定的协议层不能识别选项
    • ENOTSOCK:sock描述的不是套接字

    optname 参数详细说明:

    • SOL_SOCKET
    选项名称 说明 数据类型
    SO_BROADCAST 允许发送广播数据 int
    SO_DEBUG 允许调试 int
    SO_DONTROUTE 不查找路由 int
    SO_ERROR 获得套接字错误 int
    SO_KEEPALIVE 保持连接 int
    SO_LINGER 延迟关闭连接 struct linger
    SO_OOBINLINE 带外数据放入正常数据流 int
    SO_RCVBUF 接收缓冲区大小 int
    SO_SNDBUF 发送缓冲区大小 int
    SO_RCVLOWAT 接收缓冲区下限 int
    SO_SNDLOWAT 发送缓冲区下限 int
    SO_RCVTIMEO 接收超时 struct timeval
    SO_SNDTIMEO 发送超时 struct timeval
    SO_REUSERADDR 允许重用本地地址和端口 int
    SO_TYPE 获得套接字类型 int
    SO_BSDCOMPAT 与BSD系统兼容 int
    • IPPROTO_IP
    选项名称 说明 数据类型
    IP_HDRINCL 在数据包中包含IP首部 int
    IP_OPTINOS IP首部选项 int
    IP_TOS 服务类型 int
    IP_TTL 生存时间 int
    • IPPRO_TCP
    选项名称 说明 数据类型
    TCP_MAXSEG TCP最大数据段的大小 int
    TCP_NODELAY 不使用Nagle算法 int

    小的实例

    改变默认缓冲区大小

    SO_RCVBUF和SO_SNDBUF每个套接口都有一个发送缓冲区和一个接收缓冲区,使用这两个套接口选项可以改变默认缓冲区大小。

     // 接收缓冲区
     int nRecvBuf=32*1024;     //设置为32K
     setsockopt(s,SOL_SOCKET,SO_RCVBUF,(const char*)&nRecvBuf,sizeof(int));
    
    
     //发送缓冲区
     int nSendBuf=32*1024;//设置为32K
     setsockopt(s,SOL_SOCKET,SO_SNDBUF,(const char*)&nSendBuf,sizeof(int));
    

    如何正常的关闭一个套接字

    我们在利用IOCP(完成端口)进行程序设计的时候,经常要关闭一些不满足条件的套接字。假如我们直接采用close方法进行关闭的话,绑定到IO端口的此套接字的未发送的数据就会丢失,这种情况是我们不愿意发生的。下面介绍一种合理关闭此套接字的方法:

    首先,利用setsockopt函数设定套接字的选项,我们把此套接字设定为:假如有数据未发送,当数据发送完后再关闭此套接字。
    代码如下:

    struct linger linger_setvalue;
    linger_setvalue.l_onoff = 1;
    linger_setvalue.l_linger = 0;
    
    setsockopt(socket, SOL_SOCKET, SO_LINGER, (char *)&lingerStruct, sizeof(lingerStruct) );
    
    close(Socket);
    

    当在完成端口的数据被发送出去之后,套接字就会被关闭,这样我们就完成了一个套接字的关闭。

    基于 UDP 的 组播、广播

    基于 UDP 的 组播、广播详解

    socket 端口复用

    socket 端口复用

    注意:

    一般在调用socket以后就使用setsocket进行设置。

    举例说明:

    当设置TCP套接口接收缓冲区的大小时,函数调用顺序是很重要的,因为TCP的窗口规模选项是在建立连接时用SYN与对方互换得到的。对于客户端,O_RCVBUF选项必须在connect之前设置;对于服务器,SO_RCVBUF选项必须在listen前设置。

    如果说我的文章对你有用,只不过是我站在巨人的肩膀上再继续努力罢了。
    若在页首无特别声明,本篇文章由 Schips 经过整理后发布。
    博客地址:https://www.cnblogs.com/schips/
  • 相关阅读:
    [原创软件]友盟自动添加版本工具
    【Python3.6+Django2.0+Xadmin2.0系列教程之三(入门篇-下)】学生信息管理系统
    【Python3.6+Django2.0+Xadmin2.0系列教程之二(入门篇-中)】学生信息管理系统
    [原创软件]PC端与移动端文件信息互通工具
    【Python3.6+Django2.0+Xadmin2.0系列教程之一(入门篇-上)】环境搭建及项目创建
    [原创软件]考勤查询工具
    [原创软件]友盟自定义事件批量上传器
    [原创软件]全线升级管理平台
    三步走起 提升 iOS 审核通过率 下篇
    三步走起 提升 iOS 审核通过率 上篇
  • 原文地址:https://www.cnblogs.com/schips/p/function-setsocket-and-getsocket.html
Copyright © 2020-2023  润新知