网络编程 read() 函数详解

  • 格式:doc
  • 大小:101.00 KB
  • 文档页数:9

下载文档原格式

  / 7
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

2015-8-7网络编程总结

read()读函数特性 + 实例:

问题描述:网络编程,运行的时候有没有遇到过程序运行没有按自

己设计的流程走,打印乱码,打印多了少了。如果有且翻来覆去找

不到问题所在,那么你可能需要详细的了解一下read函数的运行

机制。

客户端与服务器连接套接字sockfd;

1.sockfd好比是连接呼客户端与服务器的管道,管道有内存大

小,可以用write函数

向里面写入数据,也可以read读出数据。遵循先进先出原则,sockfd

中的数据不会自动

清除,只可以被read读出n个少n个;

服务器代码: while(proc_echo(sockfd);

客户端代码:

输入:

输出

解释:这段代码是客户端服务器write送据,服务器read后原样打印,且每次只read一个字符,大家思考一下,我明明只write 了两次,而服务器每次read一个字符总共成功read了26次(26个字符,我测试时是一个字符一个字符打印的)。是不是很想C里的scanf之类的函数?

结论:read,write次数并不一定是对应的,read函数类似scanf getchar 函数,只要套接字sockfd(相当与缓存区)里有数据就能被read读取并接触read的阻塞状态,直到sockfd是空的,read 函数读不到数据才会阻塞,当然前提是程序能执行到read。至于write函数,没什么好说的,次数取决于你,你想输如几次就几次。由此可见,编程时不需要考虑read write函数执行先后顺序。最后提醒大家,网络编程时控制好read write数据块的大小。

乱码:多半是你打印的字符串缺少结束符。

什么时候需要加结束符呢?当然是缺少结束符时加。比如:

服务器:write(sockfd, “123456\0”, 7);

客户端:read(sockfd, buf, 3);

read(sockfd, buf, 4);

第二次不必加。

问题二:我以上的代码服务器客户端各有一对 read write(拿掉注释)。而且都是同一个套接字sockfd,那么问题来了,在一端write 的数据会不会被同一端的read读到呢(比如服务器write数据buf,

紧跟着服务器read,read会不会读到buf)?我们可以做这样的测试。

客户端服务器我注释掉客户端的write,放出服务器的read write 伪代码如下:

客户端:只向服务器发送输入的数据

while(1)

{

printf("Enter the message : ");

fgets(buffer,sizeof(buffer)-1,stdin);

ret=write(sockfd,buffer,strlen(buffer)); buffer[ret]=0; //

}

服务器:getchar停顿,read接收发来的数据并打印,再write字符串“@@”同样是循环的

int ret;

char buf[BUFFER_SIZE];

getchar();//停顿

ret = read(sockfd, buf, 1);//读4个字符

buf[ret] = 0;

printf("%s", buf);//注意打印加换行

fflush(stdout);

ret=write(sockfd,"@@",3);

return -1;

测试:

客户端输入:服务器输出:

结论:不会,整个程序只有服务器端有read,服务器write“@@”,并没有打印出来,所以read不会接收本端发送的write。

有人会问,那么服务器write的“@@”哪去了?其实还在sockfd 中,只是本端read不到的。你只要把客户端的read放出来测试就会发现客户端能read到服务器的write,打印“@@”。

总结:套接字类似全双工管道,两边都能读写,一边写的只能在另一边读出来,遵循先进先出。可以理解成两个方向相反合并的队列

问题三:大家做项目时有没有遇到过这样的问题,聊天模块多线程或多进程,有时自己write发的数据被自己的read接收了。那么是不是真的进程线程之间会相互干扰呢?大家可以通过以上的分

析思考下。

本人测试过了,答案是不会,这里简单分析下就不上图了。

先说多线程,子线程只是一个函数,它与主线程共用争夺一个套接字sockfd,这样的话问题就和上面的问题一样了,客户端服务器之间只有一条”管道“sockfd“,因此理论上是不会的实际上也不会。

至于多进程,子进程是父进程的一个副本,等于各自有一条通向服务器的一模一样”管道“,两个管道之间当然是不会存在通信的。

附加:上述问题究竟出在哪,我只能说是程序逻辑出了问题,只有尽量理清你的程序架构吧。