linux c socket选项:SO_ERROR

获取套接字选项参数getsockopt原型:

#include <sys/socket.h>
 
int getsockopt(int sockfd, int level, int option, void *optval, socklen_t optlen);

当一个socket发生错误的时候,将使用一个名为so_error的变量记录对应的错误代码,这又叫做pending error,so_error为0时表示没有错误发生。一般来说,有2种方式通知进程有socket错误发生:

1、进程阻塞在select中,有错误发生时,select将返回,并将发生错误的socket标记为可读写;

2、如果进程使用信号驱动的I/O,将会有一个SIGIO产生并发往对应进程;

此时,进程可以通过SO_ERROR取得具体的错误代码。getsockopt返回后,*optval指向的区域将存储错误代码,而so_error被设置为0。

当so_error不为0时,如果进程对socket进行read操作,若此时接收缓存中没有数据可读,则read返回-1,且errno设置为so_error,so_error置为0,否则将返回缓存中的数据而不是返回错误;如果进行write操作,将返回-1,errno置为so_error,so_error清0。

注意,这是一个只可以获取,不可以设置的选项。

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个使用 `select` 函数等待 `accept` 的示例代码: ```c #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #include <sys/socket.h> #include <arpa/inet.h> #include <sys/select.h> #define LISTEN_BACKLOG 50 #define MAX_EVENTS 10 int main(int argc, char *argv[]) { int listen_fd, conn_fd; struct sockaddr_in server_addr, client_addr; socklen_t client_addr_len; fd_set read_fds, write_fds, except_fds; int max_fd, ret; // 创建监听套接字 listen_fd = socket(AF_INET, SOCK_STREAM, 0); if (listen_fd < 0) { perror("socket error"); exit(EXIT_FAILURE); } // 设置地址复用选项 int optval = 1; if (setsockopt(listen_fd, SOL_SOCKET, SO_REUSEADDR, &optval, sizeof(optval)) < 0) { perror("setsockopt error"); exit(EXIT_FAILURE); } // 绑定地址 memset(&server_addr, 0, sizeof(server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_port = htons(8888); server_addr.sin_addr.s_addr = htonl(INADDR_ANY); if (bind(listen_fd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) { perror("bind error"); exit(EXIT_FAILURE); } // 监听 if (listen(listen_fd, LISTEN_BACKLOG) < 0) { perror("listen error"); exit(EXIT_FAILURE); } // 初始化 fd_set FD_ZERO(&read_fds); FD_ZERO(&write_fds); FD_ZERO(&except_fds); FD_SET(listen_fd, &read_fds); max_fd = listen_fd; while (1) { // 使用 select 等待读事件 ret = select(max_fd + 1, &read_fds, &write_fds, &except_fds, NULL); if (ret < 0) { perror("select error"); exit(EXIT_FAILURE); } // 处理读事件 if (FD_ISSET(listen_fd, &read_fds)) { client_addr_len = sizeof(client_addr); conn_fd = accept(listen_fd, (struct sockaddr *)&client_addr, &client_addr_len); if (conn_fd < 0) { perror("accept error"); continue; } // 将新连接的套接字加入 fd_set FD_SET(conn_fd, &read_fds); if (conn_fd > max_fd) { max_fd = conn_fd; } printf("new connection from %s:%d\n", inet_ntoa(client_addr.sin_addr), ntohs(client_addr.sin_port)); } // 处理其它读事件 for (int fd = listen_fd + 1; fd <= max_fd; fd++) { if (FD_ISSET(fd, &read_fds)) { // TODO: 处理读事件 } } // 清空 fd_set FD_ZERO(&read_fds); FD_ZERO(&write_fds); FD_ZERO(&except_fds); FD_SET(listen_fd, &read_fds); for (int fd = listen_fd + 1; fd <= max_fd; fd++) { if (FD_ISSET(fd, &read_fds) || FD_ISSET(fd, &write_fds) || FD_ISSET(fd, &except_fds)) { FD_SET(fd, &read_fds); } } } exit(EXIT_SUCCESS); } ``` 代码中使用 `select` 函数等待监听套接字的读事件,一旦监听套接字上有新的连接请求,就调用 `accept` 函数接受连接,并将新连接的套接字加入 `fd_set` 中,并更新 `max_fd`。然后在 `for` 循环中遍历其它套接字,处理读事件。最后清空 `fd_set`,重新添加需要监听的套接字。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值