原文作者:aircraft

原文链接:https://www.cnblogs.com/DOMLX/p/9614820.html

一.标准I/O

1,什么是标准I/O?其实是指C语言里的文件操作函数,如:fopen,feof,fgetc,fputs等函数,他们和平台无关。

2,网络通信中使用标准I/O的优点:

  • 良好的移植性。良好移植性这个不需多解释,不仅是I/O函数,所有的标准函数都具有良好的移植性。因为,为了支持所有的操作系统(编译器),这些函数都是按照ANSI C标准定义的。

  • 标准I/O函数可以利用缓冲提高性能。在网络通信中,read,write传输数据只有一种套接字缓冲,但使用标准I/O传输会有额外的缓冲,即I/O缓冲和套接字缓冲两个。使用I/O缓冲主要是为了提高性能,需要传输的数据越多时越明显。因为,一次发送更多的数据要比分多次发送同样的数据性能要高。发送一次数据就对应一个数据包,往往数据包的头信息比较大,它与数据大小无关。

3,网络通信中使用标准I/O的缺点:

  • 不容易进行双向通信。
  • 有时可能频繁调用fflush函数。
  • 需要以FILE结构体指针的形式返回文件描述符。

4,转换函数

//将文件描述符转换为标准I/O函数中使用的FILE结构体指针
FILE * fdopen(int fildes, const char *mode);
成功时返回转换的FILE结构体指针,失败返回NULL

//将FILE结构体指针转换为文件描述符
int fileno(FILE *stream);
成功返回转换后的文件描述符,失败返回-1

注释:套接字中使用标准I/O,其实主要是运用在需要传输大量数据的情况,因为其需要编写额外代码,所以并不像想象中的那么常用。

先给个fdopen函数的简单示例:

#include <stdio.h>
main()
{
FILE * fp = fdopen(, "w+");
fprintf(fp, "%s\n", "hello!");
fclose(fp);
}

fileno示例:

#include <stdio.h>
main()
{
FILE *fp;
int fd;
fp = fopen("/etc/passwd", "r");
fd = fileno(fp);
printf("fd = %d\n", fd);
fclose(fp);
}

5.fgets与fputs函数:

1.fgets()

功能:有文件中读取一字符串

定义:char *fgets(char *s, int size, FILE *stream)

说明:
fgets()用来从参数stream所指的文件读入字符并存到参数s所指向的内存空间,

直到读到换行字符\n,读到文件尾或是读到size-1个字符为止,最后会加入NULL作为文件结束。

返回值:
成功 返回s的指针
失败 返回NULL

2.fputs()

功能:将一指定的字符串写入文件内

定义: char * fputs(const *char s, FILE *stream)

说明:
fputs()用来将s所指的字符串写到参数stream所指向的文件中

返回值:
成功 返回写入字符串的个数
失败 返回EOF

示例代码:

include <stdio.h>

int main()
{
int str[];
fputs(fgets(str, , strin), strout); return ;
}

6.feof介绍:

1.在stdio.h中的宏定义

#define  _IOEOF  0x0010 
#define  feof(_stream)  ((_stream)->_flag & _IOEOF)

2.feof的使用:

feof用检测流上的文件结束符,其返回值有两种情况:如果遇到文件结束,函数值为非零值,否则函数值为0。

注:此处的文件结束标志是EOF,EOF的16进制代码为0xFF(十进制为-1),特用在文本文件中,因为在文本文件中数据是以ASCⅡ代码值的形式存放,普通字符的ASCⅡ代码的范围是32到127(十进制),与EOF不冲突,因此可以直接使用。但是在二进制文件中,数据有可能出现-1,因此不能用EOF来作为二进制文件的结束标志,可以通过feof函数来判断。

注意了这些标准I/O函数速度是比平常的函数快很多很多的,不过也不是每次都用到,具体看对什么情况了

二.基于标准I/O函数实现套接字服务端与客户端通信

LINUX下服务端:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h> #define BUF_SIZE 1024
void error_handling(char *message); int main(int argc, const char * argv[]) {
int serv_sock, clnt_sock;
char message[BUF_SIZE];
int str_len, i; struct sockaddr_in serv_adr, clnt_adr;
socklen_t clnt_adr_sz;
FILE *readfp;
FILE *writefp; if(argc != )
{
printf("Usage: %s <port> \n", argv[]);
exit();
} serv_sock = socket(PF_INET, SOCK_STREAM, );
if(serv_sock == -)
error_handling("socket() error"); memset(&serv_adr, , sizeof(serv_adr));
serv_adr.sin_family = AF_INET;
serv_adr.sin_addr.s_addr = htonl(INADDR_ANY);
serv_adr.sin_port = htons(atoi(argv[])); if(bind(serv_sock, (struct sockaddr *) &serv_adr, sizeof(serv_adr)) == -)
error_handling("bind() error"); if(listen(serv_sock, ) == -)
error_handling("listen() error");
clnt_adr_sz = sizeof(clnt_adr); for (i = ; i < ; i++) {
clnt_sock = accept(serv_sock, (struct sockaddr *) &clnt_adr, &clnt_adr_sz);
if(clnt_sock == -)
error_handling("accept() error");
else
printf("Connected client %d \n", i+); //将文件描述符转换为FILE结构体指针
readfp = fdopen(clnt_sock, "r");
writefp = fdopen(clnt_sock, "w");
while (!feof(readfp))
{
//转化为标准I/O操作
fgets(message, BUF_SIZE, readfp);//相当于read,接收
fputs(message, writefp); //相当于write,发送
fflush(writefp); //刷新缓冲,立即显示而不是一直放缓冲中,保证立即将数据传输到客服端
}
fclose(readfp);
fclose(writefp);
} close(serv_sock);
return ;
} void error_handling(char *message)
{
fputs(message, stderr);
fputc('\n', stderr);
exit();
}

LINUX下客户端:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <sys/socket.h> #define BUF_SIZE 1024
void error_handling(char *message); int main(int argc, const char * argv[]) {
int sock;
char message[BUF_SIZE];
int str_len;
struct sockaddr_in serv_adr;
FILE *readfp;
FILE *writefp; if(argc != )
{
printf("Usage: %s <IP> <port> \n", argv[]);
exit();
} sock = socket(PF_INET, SOCK_STREAM, );
if(sock == -)
error_handling("socket() error"); memset(&serv_adr, , sizeof(serv_adr));
serv_adr.sin_family = AF_INET;
serv_adr.sin_addr.s_addr = inet_addr(argv[]);
serv_adr.sin_port = htons(atoi(argv[])); if (connect(sock, (struct sockaddr *) &serv_adr, sizeof(serv_adr)) == -)
error_handling("connect() error");
else
puts("Connected ..............."); readfp = fdopen(sock, "r");
writefp = fdopen(sock, "w"); while ()
{
fputs("Input message(Q to quit): ", stdout);
fgets(message, BUF_SIZE, stdin);
if (!strcmp(message, "q\n") || !strcmp(message, "Q\n"))
break; fputs(message, writefp);
fflush(writefp);
fgets(message, BUF_SIZE, readfp);
printf("Message from server : %s", message);
} fclose(writefp);
fclose(readfp);
return ;
} void error_handling(char *message)
{
fputs(message, stderr);
fputc('\n', stderr);
exit();
}

最后说一句啦。本网络编程入门系列博客是连载学习的,有兴趣的可以看我博客其他篇。。。。c++ 网络编程课设入门超详细教程 ---目录

好了今天对网络编程的学习就到这里结束了,小飞机我要撤了去吃饭了。,,,很多人大学都很迷茫不知道学点什么好,,,,,管他的,想那么多干嘛,先学了再说,对技术如有偏见,那么你的领域就局限于此了---《一专多精》

参考博客:https://blog.csdn.net/u010223072/article/details/48316117

参考博客:https://blog.csdn.net/qq_32103869/article/details/50834629

参考书籍:《TCP/IP 网络编程 --尹圣雨》

c++ 网络编程(七) LINUX下 socket编程 基于套接字的标准I/O函数使用 与 fopen,feof,fgets,fputs函数用法的更多相关文章

  1. Linux下Socket编程的端口问题( Bind error: Address already in use )

    Linux下Socket编程的端口问题( Bind error: Address already in use ) 在进行linux网络编程时,每次修改了源代码并再次编译运行时,常遇到下面的地使用错误 ...

  2. linux下socket编程实例

    linux下socket编程实例一.基本socket函数Linux系统是通过提供套接字(socket)来进行网络编程的.网络的socket数据传输是一种特殊的I/O,socket也是一种文件描述符.s ...

  3. Linux下socket编程基本知识

    本文档主要讲解了Linux下socket编程的一些基本知识,主要包括套接字和字节序的概念,以及一些常用的结构体和函数. 本文是在网易云课堂学习过程中的记录,这个老师讲得很不错,推荐大家围观. Linu ...

  4. [转] - linux下socket编程实例

    一.基本socket函数Linux系统是通过提供套接字(socket)来进行网络编程的.网络的socket数据传输是一种特殊的I/O,socket也是一种文件描述符.socket也有一个类似于打开文件 ...

  5. linux下socket编程

    相关结构 //下边这两个结构定义在<sys/types.h>里 //一般的地址结构,只能用于覆盖(把其他地址转换为此类型),且只能引用该地址的sa_family字段 struct sock ...

  6. LInux下socket编程学习笔记

    1.socket套接字: socket起源于Unix,而Unix/Linux基本哲学之一就是“一切皆文件”,都可以用“打开open –> 读写write/read –> 关闭close”模 ...

  7. Python套接字编程(1)——socket模块与套接字编程

    在Python网络编程系列,我们主要学习以下内容: 1. socket模块与基本套接字编程 2. socket模块的其他网络编程功能 3. SocketServer模块与简单并发服务器 4. 异步编程 ...

  8. c++ 网络编程(五) LINUX下 socket编程 多种I/O函数 -以及readv和writev函数用法

    原文作者:aircraft 原文链接:https://www.cnblogs.com/DOMLX/p/9614056.html 一.多种I/O函数 前言:之前我们讲的数据传输一般Linux上用writ ...

  9. c++ 网络编程(六)LINUX下 socket编程 多播与广播 实现一次发送所有组客户端都能接收到

    原文作者:aircraft 原文链接:https://www.cnblogs.com/DOMLX/p/9614288.html 一.多播 锲子:有这么一种情况,网络电台可能需要同时向成千上万的用户传输 ...

随机推荐

  1. 面试题: 已知一个含有n个不同元素的集合,要求打印其所有具有k个元素的子集(不允许有重复的)

    TX面试题2: 已知一个含有n个元素的集合,要求打印其所有具有k个元素的子集(不允许有重复的) 题目分析, 为了便于说明,不妨将问题简化一下: 已知一个盒子中有n个不同的球,分别标记为{a1,a2,. ...

  2. codeforces|CF1054D Changing Array

    因为数据范围是2e5级别的,所以我们考虑用异或前缀和来处理区间的异或情况.(比如说a包括b,那么我们通过异或可以知道b对于a的补区间的信息) 之后因为对任意\(a_i\)进行取反操作,会改变它和它之后 ...

  3. 【04】循序渐进学 docker:Dockerfile

    写在前面的话 从前面我们简单的了解了镜像,也运行了容器,各种官方的镜像显然无法满足我们自己的需求,我们目的终究是运行自己的业务. 所以,本章节的 Dockerfile 就主要讲怎么在官方镜像的基础上制 ...

  4. nginx location 与 rewrite详解 (转)

    点我

  5. [转]解读Unity中的CG编写Shader系列6——不透明度与混合

    1.不透明度当我们要将两个半透的纹理贴图到一个材质球上的时候就遇到混合的问题,由于前面的知识我们已经知道了片段着色器以及后面的环节的主要工作是输出颜色与深度到帧缓存中,所以两个纹理在每个像素上的颜色到 ...

  6. python-数值类型转换

    常用的数据类型转换 函数 说明 int(x [,base ]) 将x转换为一个整数 long(x [,base ]) 将x转换为一个长整数 float(x ) 将x转换到一个浮点数 complex(r ...

  7. “全栈2019”Java第五十一章:继承与final关键字详解

    难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...

  8. Linux centos 7安装

    Linux 安装 本章节我们将为大家介绍Linux的安装. 本章节以 centos6.4 为例. centos 下载地址: 可以去官网下载最新版本:https://www.centos.org/dow ...

  9. 使用Ajax、json实现京东购物车结算界面的数据交互

    <div class="goodsList_menu"> <div class="goodsList"> <div class=& ...

  10. [ActionSprit 3.0] FMS服务器带宽检测

    package { import flash.display.Sprite; import flash.net.NetConnection; import flash.events.NetStatus ...