socket学习二、accept、read、write函数详解

一、 accept() 函数

1.1、函数定义

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);

作用: accept()系统调用指定服务端去接受客户端的连接,接收后,返回客户端套接字的标识,且获得了客户端套接字的信息(包括客户端IP和端口等)。如果没有客户端连接,则会阻塞等待客户端连接的到来。
accept函数

1.2、参数详解:

1) int sockfd: 用来标识服务端套接字,即由socket函数返回,处于listen态的套接字。
2) struct sockaddr *addr: 用来保存客户端套接字信息(包括客户端IP和端口等)
3) socklen_t *addrlen: 表示addr地址空间大小,与int*类型一样。

1.3、返回值:

accept() 返回一个新的套接字来和客户端通信。

1.4、用法:

connfd = accept(listenfd,(sockaddr *)&clientaddr, &clilen);

注: listen() 只是让套接字进入监听状态,并没有真正接收客户端请求,listen() 后面的代码会继续执行,直到遇到 accept()accept()会阻塞程序执行(后面代码不能被执行),直到有新的请求到来。

1.5、参考资料:

1、listen()和accept()函数:让套接字进入监听状态并响应客户端请求

二、 write() 函数

2.1、函数定义

ssize_t write(int fd, const void *buf, size_t nbytes);

作用: 将缓冲区 buf 中的 nbytes 个字节写入文件 fd

2.2、参数详解:

1) int fd: 表示要写入的文件的描述符
*2) const void buf: 表示要写入的数据的缓冲区地址
3) size_t nbytes: 表示要写入的数据的字节数

2.3、返回值:

1) ssize_t : signed int类型,表示成功则返回写入的字节数,失败则返回 -1。

三、 read() 函数

3.1、函数定义

ssize_t read(int fd, void *buf, size_t nbytes);

作用: 从套接字 fd 中读取 nbytes 个字节并保存到缓冲区 buf。

3.2、返回值:

成功则返回读取到的字节数(但遇到文件结尾则返回0),失败则返回 -1。

3.3、参考资料:

1、write()/read():发送数据和接收数据

  • 2
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值