accept函数的参数不是随便填的

2024-06-18 13:08

本文主要是介绍accept函数的参数不是随便填的,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

#include<sys/socket.h>
int accept(int sockfd, struct sockaddr* addr, socklen_t* len)
返回:非负描述字——成功, -1——失败

accept默认会阻塞进程,直到有一个客户连接建立后返回,它返回的是一个新可用的套接字,这个套接字是连接套接字。此时我们需要区分两种套接字,一种套接字正如accept的参数sockfd,它是监听套接字,在调用listen函数之后,一个套接字会从主动连接的套接字变身为一个监听套接字;而accept返回是一个连接套接字,它代表着一个网络已经存在的点点连接。自然要问的是:为什么要有两种套接字?原因很简单,如果使用一个描述字的话,那么它的功能太多,使得使用很不直观,同时在内核确实产生了一个这样的新的描述字。

如果accept成功返回,则服务器与客户已经正确建立连接了,此时服务器通过accept返回的套接字来完成与客户的通信。

参数sockfd

参数sockfd就是上面解释中的监听套接字,这个套接字用来监听一个端口,当有一个客户与服务器连接时,它使用这个一个端口号,而此时这个端口号正与这个 套接字关联。当然客户不知道套接字这些细节,它只知道一个地址和一个端口号。

参数addr

这是一个结果参数,它用来接受一个返回值,这返回值指定客户端的地址,当然这个地址是通过某个地址结构来描述的,用户应该知道这一个什么样的地址结 构。如果对客户的地址不感兴趣,那么可以把这个值设置为NULL。

参数len

如同大家所认为的,它也是结果的参数,用来接受上述addr的结构的大小的,它指明addr结构所占有的字节个数。同样的,它也可以被设置为NULL。

但是,如果你给accept函数第2和第3个参数赋值后,就要记得“用之”,否则就会报错。

举个简单的例子

客户端程序都相同,如下:

#include <stdio.h>
#include <string.h>//memset strlen
#include <stdlib.h>#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>#include <ctime>
#include <arpa/inet.h>/*
unistd.h 中所定义的接口通常都是大量针对系统调用的封装(英语:wrapper functions)
如 fork、pipe 以及各种 I/O 原语(read、write、close 等等)
*/
#include <unistd.h>#include <wait.h>#define MAX_STR 256
int main(){char buf[MAX_STR];char succ[]="success";char fail[]="failed";int clientFd;clientFd=socket(AF_INET,SOCK_STREAM,0);//0:按给定的域和套接字类型选择默认协议struct sockaddr_in serverAddr;memset(&serverAddr,0,sizeof(serverAddr));serverAddr.sin_family=AF_INET;serverAddr.sin_port=htons(12345);serverAddr.sin_addr.s_addr=inet_addr("127.0.0.1");printf("Socket has been created.\n");int cRes=connect(clientFd,(struct sockaddr*)&serverAddr,sizeof(serverAddr));//判断是否成功if(cRes<0){printf("fail to connect\n");exit(1);}else{printf("connect success!\n");}int recvLen=recv(clientFd,(void*)buf,MAX_STR,0);if(recvLen>0){printf("recv success:\n");printf("recv: %s\n",buf);send(clientFd,(void*)succ,strlen(succ),0);//不接受带外数据,阻塞式,可将数据路由出本地网络}else{printf("fail to recv the server's data\n");send(clientFd,(void*)fail,strlen(fail),0);}close(clientFd);printf("Now the connection has been broken\n");close(clientFd);return 0;
}


服务端1:
acceptFd = accept(serverFd,(struct sockaddr*)&clientAddr,(socklen_t*)&clientLen);

后续程序中使用到了clientAddr,clinetLen,程序没有问题

#include <stdio.h>
#include <string.h>//memset strlen
#include <stdlib.h>#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>#include <ctime>
#include <arpa/inet.h>/*
unistd.h 中所定义的接口通常都是大量针对系统调用的封装(英语:wrapper functions)
如 fork、pipe 以及各种 I/O 原语(read、write、close 等等)
*/
#include <unistd.h>#include <wait.h>#define MAX_STR 256
int count=0;
int main(){char buf[MAX_STR];char sayHello[]="Good morning Mr S!";int serverFd;serverFd=socket(AF_INET,SOCK_STREAM,0);int acceptFd;	struct sockaddr_in clientAddr;int clientLen;struct sockaddr_in serverAddr;memset(&serverAddr,0,sizeof(serverAddr));serverAddr.sin_family=AF_INET;serverAddr.sin_port= htons(12345);serverAddr.sin_addr.s_addr=htonl(INADDR_ANY);bind(serverFd,(sockaddr*)&serverAddr,sizeof(serverAddr));printf("Socket has been created and bind\n");listen(serverFd,5);while(true){printf("--------------------------%d\n",count++);printf("Now listening...\n");/* 接受客户端申请的连接 */acceptFd = accept(serverFd,(struct sockaddr*)&clientAddr,(socklen_t*)&clientLen);/* 如果client成功连接到server, 则执行 */if(acceptFd >= 0){printf("Now the link has been connected.\n");/* 从客户端的套接字中提取出IP地址 和其他信息*/int clientip = clientAddr.sin_addr.s_addr;printf("Client ip : %d.%d.%d.%d\n",clientip&255,(clientip>>8)&255,(clientip>>16)&255,(clientip>>24)&255);printf("Client prot : %d\n",ntohs(clientAddr.sin_port));/* 使用send向client发送信息 */sprintf(buf,"THE SEND MSG");printf("[SEND] Starting sending [send] msg ...\n");send(acceptFd, (void*)buf, strlen(buf),0);//发送数据 strlen(buff) bytes数据recv(acceptFd, (void*)buf, MAX_STR, 0 );//接收客户端反馈,成功或失败?//MAXN_STR:最大努力接收if(strlen(buf) > 0)printf("[SUCC] Sending succeed.\n");elseprintf("[FAIL] Sending failed.\n");/* 关闭此连接 */close(acceptFd);printf("Disconnect the link.\n");}else{/*  与client连接失败 */printf("ERROR: Failed while establish the link!\n");}}close(serverFd);return 0;
}


服务端程序2:
acceptFd = accept(serverFd,NULL,NULL);
也没有问题

#include <stdio.h>
#include <string.h>//memset strlen
#include <stdlib.h>#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>#include <ctime>
#include <arpa/inet.h>/*
unistd.h 中所定义的接口通常都是大量针对系统调用的封装(英语:wrapper functions)
如 fork、pipe 以及各种 I/O 原语(read、write、close 等等)
*/
#include <unistd.h>#include <wait.h>#define MAX_STR 256
int count=0;
int main(){char buf[MAX_STR];char sayHello[]="Good morning Mr S!";int serverFd;serverFd=socket(AF_INET,SOCK_STREAM,0);int acceptFd;	struct sockaddr_in clientAddr;int clientLen;struct sockaddr_in serverAddr;memset(&serverAddr,0,sizeof(serverAddr));serverAddr.sin_family=AF_INET;serverAddr.sin_port= htons(12345);serverAddr.sin_addr.s_addr=htonl(INADDR_ANY);bind(serverFd,(sockaddr*)&serverAddr,sizeof(serverAddr));printf("Socket has been created and bind\n");listen(serverFd,5);while(true){printf("--------------------------%d\n",count++);printf("Now listening...\n");/* 接受客户端申请的连接 */acceptFd = accept(serverFd,NULL,NULL);/* 如果client成功连接到server, 则执行 */if(acceptFd >= 0){printf("Now the link has been connected.\n");/* 从客户端的套接字中提取出IP地址 和其他信息int clientip = clientAddr.sin_addr.s_addr;printf("Client ip : %d.%d.%d.%d\n",clientip&255,(clientip>>8)&255,(clientip>>16)&255,(clientip>>24)&255);printf("Client prot : %d\n",ntohs(clientAddr.sin_port));*//* 使用send向client发送信息 */sprintf(buf,"THE SEND MSG");printf("[SEND] Starting sending [send] msg ...\n");send(acceptFd, (void*)buf, strlen(buf),0);//发送数据 strlen(buff) bytes数据recv(acceptFd, (void*)buf, MAX_STR, 0 );//接收客户端反馈,成功或失败?//MAXN_STR:最大努力接收if(strlen(buf) > 0)printf("[SUCC] Sending succeed.\n");elseprintf("[FAIL] Sending failed.\n");/* 关闭此连接 */close(acceptFd);printf("Disconnect the link.\n");}else{/*  与client连接失败 */printf("ERROR: Failed while establish the link!\n");}}close(serverFd);return 0;
}
但是,如果你使用了
acceptFd = accept(serverFd,(struct sockaddr*)&clientAddr,(socklen_t*)&clientLen);
却没有用之,程序就会报错!可以自己试下如下程序,具体原因不知道,但是恶心是恶心够了。。。
#include <stdio.h>
#include <string.h>//memset strlen
#include <stdlib.h>#include <sys/types.h>
#include <sys/stat.h>
#include <sys/socket.h>#include <ctime>
#include <arpa/inet.h>/*
unistd.h 中所定义的接口通常都是大量针对系统调用的封装(英语:wrapper functions)
如 fork、pipe 以及各种 I/O 原语(read、write、close 等等)
*/
#include <unistd.h>#include <wait.h>#define MAX_STR 256
int count=0;
int main(){char buf[MAX_STR];char sayHello[]="Good morning Mr S!";int serverFd;serverFd=socket(AF_INET,SOCK_STREAM,0);int acceptFd;	struct sockaddr_in clientAddr;int clientLen;struct sockaddr_in serverAddr;memset(&serverAddr,0,sizeof(serverAddr));serverAddr.sin_family=AF_INET;serverAddr.sin_port= htons(12345);serverAddr.sin_addr.s_addr=htonl(INADDR_ANY);bind(serverFd,(sockaddr*)&serverAddr,sizeof(serverAddr));printf("Socket has been created and bind\n");listen(serverFd,5);while(true){printf("--------------------------%d\n",count++);printf("Now listening...\n");/* 接受客户端申请的连接 */acceptFd = accept(serverFd,(struct sockaddr*)&clientAddr,(socklen_t*)&clientLen);/* 如果client成功连接到server, 则执行 */if(acceptFd >= 0){printf("Now the link has been connected.\n");/* 从客户端的套接字中提取出IP地址 和其他信息int clientip = clientAddr.sin_addr.s_addr;printf("Client ip : %d.%d.%d.%d\n",clientip&255,(clientip>>8)&255,(clientip>>16)&255,(clientip>>24)&255);printf("Client prot : %d\n",ntohs(clientAddr.sin_port));*//* 使用send向client发送信息 */sprintf(buf,"THE SEND MSG");printf("[SEND] Starting sending [send] msg ...\n");send(acceptFd, (void*)buf, strlen(buf),0);//发送数据 strlen(buff) bytes数据recv(acceptFd, (void*)buf, MAX_STR, 0 );//接收客户端反馈,成功或失败?//MAXN_STR:最大努力接收if(strlen(buf) > 0)printf("[SUCC] Sending succeed.\n");elseprintf("[FAIL] Sending failed.\n");/* 关闭此连接 */close(acceptFd);printf("Disconnect the link.\n");}else{/*  与client连接失败 */printf("ERROR: Failed while establish the link!\n");}}close(serverFd);return 0;
}


这篇关于accept函数的参数不是随便填的的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1072077

相关文章

Kotlin 作用域函数apply、let、run、with、also使用指南

《Kotlin作用域函数apply、let、run、with、also使用指南》在Kotlin开发中,作用域函数(ScopeFunctions)是一组能让代码更简洁、更函数式的高阶函数,本文将... 目录一、引言:为什么需要作用域函数?二、作用域函China编程数详解1. apply:对象配置的 “流式构建器”最

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda

一文带你了解SpringBoot中启动参数的各种用法

《一文带你了解SpringBoot中启动参数的各种用法》在使用SpringBoot开发应用时,我们通常需要根据不同的环境或特定需求调整启动参数,那么,SpringBoot提供了哪些方式来配置这些启动参... 目录一、启动参数的常见传递方式二、通过命令行参数传递启动参数三、使用 application.pro

C++中::SHCreateDirectoryEx函数使用方法

《C++中::SHCreateDirectoryEx函数使用方法》::SHCreateDirectoryEx用于创建多级目录,类似于mkdir-p命令,本文主要介绍了C++中::SHCreateDir... 目录1. 函数原型与依赖项2. 基本使用示例示例 1:创建单层目录示例 2:创建多级目录3. 关键注

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程

基于@RequestParam注解之Spring MVC参数绑定的利器

《基于@RequestParam注解之SpringMVC参数绑定的利器》:本文主要介绍基于@RequestParam注解之SpringMVC参数绑定的利器,具有很好的参考价值,希望对大家有所帮助... 目录@RequestParam注解:Spring MVC参数绑定的利器什么是@RequestParam?@

kotlin的函数forEach示例详解

《kotlin的函数forEach示例详解》在Kotlin中,forEach是一个高阶函数,用于遍历集合中的每个元素并对其执行指定的操作,它的核心特点是简洁、函数式,适用于需要遍历集合且无需返回值的场... 目录一、基本用法1️⃣ 遍历集合2️⃣ 遍历数组3️⃣ 遍历 Map二、与 for 循环的区别三、高

C语言字符函数和字符串函数示例详解

《C语言字符函数和字符串函数示例详解》本文详细介绍了C语言中字符分类函数、字符转换函数及字符串操作函数的使用方法,并通过示例代码展示了如何实现这些功能,通过这些内容,读者可以深入理解并掌握C语言中的字... 目录一、字符分类函数二、字符转换函数三、strlen的使用和模拟实现3.1strlen函数3.2st

MySQL中COALESCE函数示例详解

《MySQL中COALESCE函数示例详解》COALESCE是一个功能强大且常用的SQL函数,主要用来处理NULL值和实现灵活的值选择策略,能够使查询逻辑更清晰、简洁,:本文主要介绍MySQL中C... 目录语法示例1. 替换 NULL 值2. 用于字段默认值3. 多列优先级4. 结合聚合函数注意事项总结C

SpringBoot接收JSON类型的参数方式

《SpringBoot接收JSON类型的参数方式》:本文主要介绍SpringBoot接收JSON类型的参数方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、jsON二、代码准备三、Apifox操作总结一、JSON在学习前端技术时,我们有讲到过JSON,而在