UNIX网络编程--简介(一)【转】
本文转载自:http://blog.csdn.net/yusiguyuan/article/details/11760187
一、概述
a) 在编写与计算机通信的程序时,首先要确定的就是和计算机通信的协议,从高层次来确定通信由哪个程序发起以及响应在合适产生。大多数网络应用按照划分成客户和服务器来组织。在设计网络应用时,确定总是由客户发起请求往往能够简化协议和程序本身。当然一个较为复杂的网络应用还需要异步回调通信,也就是由服务器向客户发起请求消息。
i. 在本书原著上多次提到客户(client)和服务器(server)这两个术语。但是他们在具体上下文的含义不同,有的是指静态的程序(客户程序和服务器程序),有的是指动态的进程(客户进程和服务器进程),有时指运行进程的主机(客户主机和服务器主机)。
ii.可认为客户与服务器之间是通过某个网络协议通信的,但实际上,这样的通信通常涉及到多个网络协议层。本书的焦点是TCP/IP协议簇,也称为网际协议簇。举例,WEB客户与服务器之间使用TCP(Transmission Control Protocol,传输控制协议)通信。TCP又转而使用IP(InternetProtocol,网际协议)通信,IP再通过某种形式的数据链路层通信。
尽管客户与服务器之间使用某个应用协议通信,传输层却使用TCP通信。数据流的流向在一端从上到下,在另一端从下到上。通常,客户和服务器通常是用户进程,
而TCP和IP协议通常是内核中协议栈的一部分。其中七层OSI模型(物理层、数据链路层、网络层、传输层、表示层、会话层、应用层)。
上图讲述的是处于同一个局域网里的通信情况,在不同局域网的情况是:通过路由器连接到广域网。(路由器是广域网的架构设备)
关于结构体struct sockaddr,也就是通过套接字地址结构,每当一个套接字函数需要一个指向某个套接字地址结构的指针时,这个指针必须强制类型转换成一个指向通过套接字地址结构的指针。
二、一个简单的时间获取客户程序
本章在举例说明的时候,使用了获取时间的服务器函数之类的东西,这是一个函简单的程序,按照的基本的TCP客户/服务流程交互,在服务端在填写字符的时候使用了函数snprintf,这个函数和sprintf的功能相似。调用sprintf无法检查目的缓冲区是否溢出。Snprintf要求其第二个参数指定目的缓冲区的大小,因此可确保该缓冲区不溢出。
单个处理一个客户的服务器程序称为迭代服务器,因为对于每个客户它都迭代执行一次。同时能处理多个客户的称为并发服务器。
下面是客户程序的main函数:
- #define MAXLIEN 1024
- int main(int argc,char *argv[])
- {
- int i,sockfd,n;
- struct sockaddr_in serveraddr;
- char recvline[MAXLIEN+1];
- sockfd=socket(AF_INET,SOCK_STREAM,0);
- //if(argc<2)
- // printf("usage:IP \r\n");
- if(sockfd<0)
- printf("error exit!\r\n");
- bzero(&serveraddr,sizeof(struct sockaddr_in));
- serveraddr.sin_family=AF_INET;
- serveraddr.sin_port=htons(13);
- inet_aton("127.0.0.1",&serveraddr.sin_addr);
- if(argc==2){
- if(inet_pton(AF_INET,argv[1],&serveraddr.sin_addr)<0)
- printf("error trans!\r\n");
- }
- if(connect(sockfd,(struct sockaddr*)&serveraddr,sizeof(serveraddr))<0)
- printf("error connect!\r\n");
- while((n=read(sockfd,recvline,MAXLIEN))>0)
- {
- i++;
- recvline[n]=0;
- if(fputs(recvline,stdout)==EOF)
- printf("fputs error!\r\n");
- }
- if(n<0)
- printf("error\r\n");
- exit 0;
- }
这里使用最标准的客户程序:初始化、连接、处理(和服务器使用字节流/数据报交互),里面使用的函数的作用和使用方法在后续章节都会陆续讲解。
其中最后一行的语句exit是程序退出,UNIX在一个进程终止时总是关闭该进程所有打开的描述符,我们的套接字描述符也就此关闭。
在使用read时,若果返回为0,那么说明对端已经关闭连接,如果返回值为负值,那么说明出现错误。还需要细细揣摩这句哈:TCP是一个没有边界的字节流协议。、
计算机网络各对等实体间交换的单元信心称为协议数据单元(protocol data unit PDU),分节(segment)就是对应于TCP传输层的PDU.按照协议与服务之间的关系,除了最底层物理层外,每层的PDU通过由紧邻下层提供给本层的服务接口,作为下层的服务数据单元传递给下层…………
应用层实体(如客户或服务器进程)间交换的PDU称为应用数据,其中在TCP应用进程之间交换的是没有长度限制的单个双向字节流,在UDP应用进程之间交换的是其长度不超过UDP发送缓冲区大小的单个记录,在SCTP应用进程之间交换的是没有总长度显示的单个或多个双向字节流。传输层实体(对应某个端口的传输层协议代码的一次运行)间交换的PDU称为消息,其中TCP的PDU特称为分节。消息或分节的长度是由限的。在TCP传输层中,发送端TCP把来自应用进程的字节流数据(既由应用进程通过一次输出操作写出到发送端TCP套接字中的数据)按顺序经分割后封装在各个各个分节中传送给接受端TCP其中每个分节所封装的数据既可能是发送端应用进程单词操作的结果,也可能是连续数次输出操作的结果,具体取决于可在连续建立阶段由对端通告的最大分节大小(maximum segmen size MSS)以及外出接口的最大传输单元(maximumtransmission unit MTU)或外出路经的路径MTU(如果网络层具有路径MTU发现功能)。分节除了用于承载应用数据外,也用于建立连接(SYN分节)、终止连接(FIN分节)、中止连接(RST分节)、确认数据接收(ACK分节)刷送待发数据(PSH分节)和携带紧急数据指针(URG分节),而且这些功能(包括承载数据)可以灵活组合。UDP传输层相当简单,发送端UDP就把俩字应用进程的单个记录整个封装在UDP消息中传送给接受端UDP。SCTP引入了称为块的数据单元,SCTP消息就由一个公共首部加上一个或多个块构成,公共首部类似UDP消息的首部,仅仅给出源目的端口和整个SCTP消息的校验和,块则可既可以承载数据,也可以承载控制信息。
网络层实体间的PDU称为IP数据报(IPdatagram),其长度有限:Ipv4数据报最大65535字节,Ipv6数据报最大65575字节。发送端IP把来自传输层的消息(或TCP分节)整个封装在IP数据报中发送。链路层实体间交换的PDU称为帧,其长度取决于具体的接口。IP数据报由IP首部和所承载的传输层数据(既网络层的SDU,由上面的定义可以看出来,上层的PDU也就是下层的SDU,所以这里也可认为是传输层传递的消息或者分节)构成。过长的IP数据报无法封装在单个帧中,需要先对其SDU进行分片,再把分成的各个片段冠以新的IP首部封装到多个帧中。在一个IP数据报从源到目的端的传送过程中,分片操作既可能发生在源端,也可能发生在途中,而其逆操作为重组。
TCP/IP协议簇为提高效率会尽可能避免IP的分片/重组操作,TCP根据MSS和MTU限定每个分节的大小以及SCTP根据MTU分片/重组过长记录都是这个目的。无论是否分片,都由IP作为链路层的SDU传输链路层,并由链路层封装在帧中的数据称为分组(俗称包)可见一个分组既可能是一个完整的IP数据报,也可能是某个IP数据报的SDU的一个片段被冠以新的IP首部后的结果。文中讨论的MSS是应用层(TCP)与传输层之间的接口属性,MTU则是网络层和链路层之间的接口属性。
三、一个简单的时间获取服务器程序
- #define MAXLIEN 1024
- int main(int argc,char *argv[])
- {
- int i,sockfd,n,connfd;
- time_t ticks;
- struct sockaddr_in serveraddr;
- char recvline[MAXLIEN+1];
- sockfd=socket(AF_INET,SOCK_STREAM,0);
- //if(argc<2)
- // printf("usage:IP \r\n");
- if(sockfd<0)
- printf("error exit!\r\n");
- bzero(&serveraddr,sizeof(struct sockaddr_in));
- serveraddr.sin_family=AF_INET;
- serveraddr.sin_port=htons(13);
- serveraddr.sin_addr.s_addr=htonl(INADDR_ANY);
- bind(sockfd,(struct sockaddr*)&serveraddr,sizeof(serveraddr));
- listen(sockfd,4);
- //if(inet_pton(AF_INET,argv[1],&serveraddr.sin_addr)<=0)
- // printf("error trans!\r\n");
- for(;;)
- {
- connfd=accept(sockfd,NULL,NULL);
- ticks=time(NULL);
- snprintf(recvline,sizeof(recvline),"%.24s\r\n",ctime(&ticks));
- write(connfd,recvline,strlen(recvline));
- close(connfd);
- }
- return 0;
- }
遵循了服务器程序的流程:初始化、绑定、监听、接受、处理(和客户进行交互)
程序中的函数在后续章节有讲解。
四、OSI模型
网络层由IPv4和IPv6这两个协议处理,可以选择的传输层有TCP或UDP,在上图中TCP与UDP之间留有间隙,表明网络应用绕过传输层直接使用IPv4或IPv6是可能的,这就是所谓的原始套接字。
OSI模型的顶上三层被合并成一层,称为应用层。这就是web客户(浏览器)、telnet客户、web服务器、ftp服务器和其其他我们在使用的网络应用所在的层。对于网际协议,OSI模型的顶上三层协议几乎没有区别。
在这一系列的讲述中,套接字编程接口是从顶上三层(网际协议的应用层)进入传输层的接口。所有文章的焦点是:如何使用套接字编写使用TCP或UDP的网络应用程序。
五、介绍几个命令
1) netstat netstat –i提供网络接口的信息。我们还指定-n标志以输出数值地址,而不是试图把他们反响解析成名字。
2) netstat –r 展示路由表
这一章中主要讲解了和基本的信息,从宏观上讲解了本书的大体框架
ps:
UNIX系统中程序和进程是在系统调用上exec上衔接的。exec既可以由shell隐式调用(直接在shell输入命令执行程序属于这种情况),也可以在用户程序中显示调用。
从上层到下层的PDU(协议数据单元)的名字为:应用数据,消息/分节(TCP协议特有的)、帧、分组(俗称包)
关于UNIX中的errno的值:
只要一个UNIX函数(例如某个套接字函数)中有错误发生,全局变量errno就被置为一个指明错误类型的正直,函数本身则通过返回-1.err_sys查看errno变量的值并输出响应的出错消息,例如errno=ETIMEDOUT,输出为“Connection timed out”
在全局变量中存放errno值对于共享所有全局变量的多个线程并不合适,在26章有讲解
如果想要了解红字及其下面的部分就需要看作者的另外一本书了:TCP/IP详解
转载请标明出处
UNIX网络编程--简介(一)【转】的更多相关文章
- UNIX网络编程---简介
UNIX网络编程---简介 一. 概述 a) 在编写与计算机通信的程序时,首先要确定的就是和计算机通信的协议,从高层次来确定通信由哪个程序发起以及响应在合适产生.大多数 ...
- UNIX网络编程--读书笔记
会集中这段时间写UNIX网络编程这本书的读书笔记,准备读三本,这一系类的文章会不断更新,一直会持续一个月多,每篇的前半部分是书中讲述的内容,每篇文章的后半部分是自己的心得体会,文章中的红色内容是很重要 ...
- UNIX网络编程读书笔记:简介
认知套接口编程接口 理解原始套接口(raw socket)的概念 值得注意的是,客户和服务器是典型的用户进程,而TCP和IP协议则通常是系统内核协议栈的一部分. 上图中在TCP和UDP之间留有间隙 ...
- Unix网络编程--卷二:进程间通信
Unix网络编程--卷二:进程间通信 本书是一部Unix网络编程的经典之作!进程间通信(IPC)几乎是所有Unix程序性能的关键,理解IPC也是理解如何开发不同主机网络应用程序的必要条件.本书从对Po ...
- Unix网络编程--卷一:套接字联网API
UNIX网络编程--卷一:套接字联网API 本书面对的读者是那些希望自己编写的程序能够使用成为套接字(socket)的API进行彼此通信的人. 目录: 0.准备环境 1.简介 2.传输层:TCP.UD ...
- [转载] 读《UNIX网络编程 卷1:套接字联网API》
原文: http://cstdlib.com/tech/2014/10/09/read-unix-network-programming-1/ 文章写的很清楚, 适合初学者 最近看了<UNIX网 ...
- 《Unix 网络编程》08:基本UDP套接字编程
基本UDP套接字编程 系列文章导航:<Unix 网络编程>笔记 UDP 概述 流程图 recvfrom 和 sendto #include <sys/socket.h> ssi ...
- 《Unix 网络编程》11:名字和地址转换
名字和地址转换 系列文章导航:<Unix 网络编程>笔记 域名系统 简介 域名系统主要用于主机名字和 IP 地址之间的映射.主机名可以是: 简单名字,如:centos01 全限定域名(FQ ...
- 《Unix 网络编程》14:高级 I/O 函数
高级 I/O 函数 ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ★ ...
随机推荐
- Django实现电影论坛
主要实现功能: 注册,登陆,发帖,评论帖子,xadmin实现后台提供帖子管理,板块管理等等 首页(搜索,筛选) 发布帖子 帖子详情页(显示帖子具体内容,提供评论) 个人信息页面(显示个人信息,修改个人 ...
- java实现从服务端下载文件
这边用一个简单的servlet实现java从服务端下载文件的操作 写一个servlet: <servlet> <servlet-name>DownloadServlet< ...
- android(七)Looper Handler分析
一.总结 Looper有一个MessageQueue,用于封装消息循环. Handler封装了消息投递,消息处理等的辅助类 二.分析 1.从Looper的用法开始分析 class LooperTh ...
- lua源代码学习(一)lua的c api外围实现
工作后,整个人已经比較松懈了.尽管一直在看lua的源代码.可是一直是比較零碎的时间,没有系统的整理,所以还是收获不多.由于近期工作也不是非常忙了,就想整理下lua的源代码学习的笔记.加深下印象,并分享 ...
- CMSPRESS-PHP无限级分类2
原文章地址:http://www.thinkphp.cn/code/170.html 超级无限分类 使用简单 效率极高 核心代码10行不到 另外 求这个分类的不足,和更高效简单的无限分类方法 ^_^ ...
- 循环结构 while,do while
while:先判断条件表达式是否成立,成立则执行循环体,不成立则不执行. 格式:while(条件表达式){ 执行语句(控制循环次数): } 例如: int x=1; while(x<3/*条件表 ...
- angular.element 动态添加和删除元素
addClass()-为每个匹配的元素添加指定的样式类名after()-在匹配元素集合中的每个元素后面插入参数所指定的内容,作为其兄弟节点append()-在每个匹配元素里面的末尾处插入参数内容att ...
- LNMPA架构剖析
LAMP或LNMP的劣势: Nginx是小巧而高效的Linux下Web服务器,跟Apache相比,它消耗资源更少,支持的并发连接更多,反向代理功能效率高.静态文件处理更快等等,Nginx可以承受3万以 ...
- myeclipse自定义JSP模板
- 改变 select下拉框 样式
select{ outline: none; text-indent: 10px; height: 45px; line-height: 45px; width: 100%; border:1px s ...