{项目管理项目报告}Linux程序设计技术技巧与项目实践网络编程第9章

上传人:精****库 文档编号:141145840 上传时间:2020-08-04 格式:PPTX 页数:45 大小:238.77KB
返回 下载 相关 举报
{项目管理项目报告}Linux程序设计技术技巧与项目实践网络编程第9章_第1页
第1页 / 共45页
{项目管理项目报告}Linux程序设计技术技巧与项目实践网络编程第9章_第2页
第2页 / 共45页
{项目管理项目报告}Linux程序设计技术技巧与项目实践网络编程第9章_第3页
第3页 / 共45页
{项目管理项目报告}Linux程序设计技术技巧与项目实践网络编程第9章_第4页
第4页 / 共45页
{项目管理项目报告}Linux程序设计技术技巧与项目实践网络编程第9章_第5页
第5页 / 共45页
点击查看更多>>
资源描述

《{项目管理项目报告}Linux程序设计技术技巧与项目实践网络编程第9章》由会员分享,可在线阅读,更多相关《{项目管理项目报告}Linux程序设计技术技巧与项目实践网络编程第9章(45页珍藏版)》请在金锄头文库上搜索。

1、第九章 网络编程,9.1 SOCKET的概念 9.2 SOCKET的建立与配置 9.3 SOCKET的连接建立 9.4 数据传输 9.5 SOCKET编程实例 9.6 PING命令解析 9.7 实战技巧 光驱与软驱的加载方法,2020/8/4,1,9.1 Socket的概念,Socket是TCP/IP协议传输层提供的接口或套接字,供用户编程访问网络资源时的工具。Socket接口是TCP/IP网络的API,socket接口定义了许多函数或例程,程序员用它们开发TCP/IP网络上的应用程序。TCP/IP协议(Transmission Control Protocol/Internet Protoc

2、ol)是传输控制/网际协议,又叫网络通信协议,这个协议是Internet国际互联网络的基础。要学Internet上的TCP/IP网络编程,必须理解socket接口。 Linux的套接口通信模式与日常生活中的电话通信非常类似,套接字代表通信线路中的端点,端点之间通过通信网络来相互联系。 Socket接口设计者最先是将接口放在Unix操作系统里面的。如果了解Unix系统的输入和输出,就很容易了解socket。网络的socket数据传输是一种特殊的I/O,socket也是一种文件描述符。Socket也具有一个类似于打开文件的函数调用socket( ),该函数返 回一个整型的socket描述符,随后的

3、连接建立、数据传输等操作都是通过该socket实现的。常用的socket类型有两种:流式socket(SOCK_STREAM)和数据报式socket(SOCK_DGRAM)。流式是一种面向连接的socket,针对于面向连接的TCP服务应用;数据报式socket是一种无连接的socket,对应于无连接的UDP服务应用。,2020/8/4,2,9.2 Socket的建立与配置,为了建立socket,程序可以调用socket函数,该函数返回一个类似于文件描述符的句柄。socket函数原型为: int socket(int domain, int type, int protocol); domain

4、指明所使用的协议族,通常为PF_INET,表示互联网协议族(TCP/IP协议族);type参数指定socket的类型:SOCK_STREAM 或SOCK_DGRAM,socket接口还定义了原始socket(SOCK_RAW),允许程序使用底层协议;protocol通常赋值为0。 socket()调用返回一个整型socket描述符,可在后面直接使用它。 Socket描述符是一个指向内部数据结构的指针,它指向描述符表入口。调用socket函数时,socket执行体将“建立一个socket”,这意味着为一个socket数据结构分配存储空间。Socket执行体用来管理描述符表。 两个网络程序之间的一

5、个网络连接包括五种信息:通信协议、本地协议地址、本地主机端口、远端主机地址和远端协议端口。Socket数据结构中包含这五种信息。,2020/8/4,3,通过socket调用返回一个socket描述符后,在使用socket进行网络传输以前,必须配置该socket。面向连接的socket客户端通过调用connect( )函数在socket数据结构中保存本地和远端信息。无连接socket的客户端和服务端以及面向连接socket的服务端通过调用bind( )函数来配置本地信息。 Bind函数将socket与本机上的一个端口相关联,随后就可以在该端口监听服务请求。Bind函数原型为: int bind(

6、int sockfd,struct sockaddr *my_addr, int addrlen); 其中的sockfd是调用socket函数返回的socket描述符,my_addr是一个指向包含有本机IP地址及端口号等信息的sockaddr类型的指针;addrlen常被设置为sizeof(struct sockaddr)。 struct sockaddr结构类型是用来保存socket信息的: struct sockaddr unsigned short sa_family; /* 地址族, AF_xxx */ char sa_data14; /* 14个字节的协议地址 */ ; sa_fam

7、ily一般为AF_INET,代表Internet(TCP/IP)地址族;sa_data则包含该socket的IP地址和端口号。,2020/8/4,4,另外还有一种结构类型: struct sockaddr_in short int sin_family; /* 地址族 */ unsigned short int sin_port; /* 端口号 */ struct in_addr sin_addr; /* IP地址 */ unsigned char sin_zero8; /* 填充0保持与struct sockaddr大小相同 */ ; 这个结构更方便使用。sin_zero用来将sockadd

8、r_in结构填充到与struct sockaddr同样的长度,可以用bzero()或memset()函数将其置为零。指向sockaddr_in 的指针和指向sockaddr的指针可以相互转换,这意味着如果一个函数所需参数类型是sockaddr时,你可以在函数调用的时候将一个指向 sockaddr_in的指针转换为指向sockaddr的指针或相反。 使用bind函数时,可以用下面的赋值实现自动获得本机IP地址和随机获取一个没有被占用的端口号: my_addr.sin_port = 0; /* 系统随机选择一个未被使用的端口号 */ my_addr.sin_addr.s_addr = INADDR

9、_ANY; /* 填入本机IP地址 */,2020/8/4,5,通过将my_addr.sin_port置为0,函数会自动选择一个未占用的端口使用。同样,通过将my_addr.sin_addr.s_addr置为INADDR_ANY,系统会自动填入本机IP地址。 注意使用bind函数需要将sin_port和sin_addr转换成为网络字节优先顺序;而sin_addr则不需要转换。实际上,计算机数据存储有两种字节优先顺序:高位字节优先和低位字节优先。Internet上数据以高位字节优先顺序在网络上传输,所以对于在内部是以低位字节优先方式存储数据的机器,在Internet上传输数据时就需要进行转换,否

10、则就会出现数据不一致。下面是几个字节顺序转换函数: htonl():把32位值从主机字节序转换成网络字节序 htons():把16位值从主机字节序转换成网络字节序 ntohl():把32位值从网络字节序转换成主机字节序 ntohs():把16位值从网络字节序转换成主机字节序 Bind()函数在成功被调用时返回0;出现错误时返回 -1并将errno置为相应的错误号。需要注意的是,在调用bind函数时一般不要将端口号置为小于1024的值,因为1到1024是保留端口号,你可以选择大于1024中的任何一个没有被占用的端口号。,2020/8/4,6,大端存储模式和小端存储模式辨析: 在嵌入式系统开发里要

11、对小端Little-endian和大端Big-endian模式非常了解。例如,16bit宽的数0 x1234在Little-endian模式CPU内存中的存放方式(假设从地址0 x4000开始存放)为: 内存地址0 x40000 x4001 存放内容0 x340 x12 而在Big-endian模式CPU内存中的存放方式则为: 内存地址0 x40000 x4001 存放内容0 x120 x34 有时候,用C语言写程序时需要知道是大端模式还是小端模式。 所谓的大端模式,是指数据的低位保存在内存的高地址中,而数据的高位,保存在内存的低地址中;所谓的小端模式,是指数据的低位保存在内存的低地址中,而数

12、据的高位保存在内存的高地址中。为什么会有大小端模式之分呢?这是因为在计算机系统中,我们是以字节为单位的,每个地址单元都对应着一个字节,一个字节为8bit。,2020/8/4,7,但是在C语言中除了8bit的char之外,还有16bit的short型,32bit的long型(要看具体的编译器),另外,对于位数大于8位的处理器,例如16位或者32位的处理器,由于寄存器宽度大于一个字节,那么必然存在着一个如何将多个字节安排的问题,从而导致了大端存储模式和小端存储模式。例如一个16bit的short型x,在内存中的地址为0 x0010,x的值为0 x1122,那么0 x11为高字节,0 x22为低字节

13、。对于大端模式,就将0 x11放在低地址中,即0 x0010中,0 x22放在高地址中,即0 x0011中。小端模式,刚好相反。我们常用的X86结构是小端模式,而KEIL C51则为大端模式。很多的ARM,DSP都为小端模式。有些ARM处理器还可以由硬件来选择是大端模式还是小端模式。下面这段代码可以用来测试一下你的编译器是大端模式还是小端模式: short int x; char x0,x1; x=0 x1122; x0=(char*) 若x0=0 x22,则是小端。,2020/8/4,8,9.3 socket的连接建立,面向连接的客户程序使用connect函数来配置socket并与远端服务器

14、建立一个TCP连接,其函数原型为: int connect(int sockfd, struct sockaddr *serv_addr,int addrlen); sockfd 是socket函数返回的socket描述符;serv_addr是包含远端主机IP地址和端口号的指针;addrlen是远端地址结构的长度。 Connect( )函数在出现错误时返回-1,并设置errno为相应的错误码。进行客户端程序设计无须调用bind( ),因为这种情况下只需知道目标机的IP地址,而客户通过哪个端口与服务器建立连接并不需要关心,socket执行体将为程序自动选择一个未被占用的端口,并通知程序数据什么时

15、候到达该端口。 connect函数启动和远端主机的直接连接。只有面向连接的客户程序使用socket时才将此socket与远端主机相连。无连接协议从不建立直接连接。面向连接的服务器也从不启动一个连接,它只是被动的在协议端口监听客户的请求。 listen( )函数使socket处于被动监听模式,并为该socket建立输入数据队列,将到达的服务请求保存在此队列中,直到程序处理它们。该函数的原型是: int listen(int sockfd,int backlog);,2020/8/4,9,Sockfd依然是socket系统调用返回的socket 描述符;backlog指定在请求队列中允许的最大请求

16、数,进入的连接请求将在队列中等待accept( )接收它们。Backlog对队列中等待服务的请求数目进行限制,大多数系统缺省值为20。如果一个服务请求到来时,输入队列已满,该socket将拒绝连接请求,客户将收到一个出错信息。当出现错误时listen( )函数返回-1,并置相应的errno错误码。 accept()函数让服务器接收客户的连接请求。在建立好输入队列后,服务器就调用accept函数,然后睡眠等待客户的连接请求。该函数的原型是: int accept(int sockfd, void *addr, int *addrlen); sockfd是被监听的socket描述符,addr通常是一个指向sockaddr_in变量的指针,该变量用来存放提出连接请求服务的主机的信息(即某台主机从某个端口发出该请求);addrten通常为一个指向值为sizeof(struct sockaddr_in)的整型指针变量。出现错误时accept函数返回-1并置相应的errno值。 当accept函数监视的socket收到连接请求时,socket执行体将建立一个新的socket,

展开阅读全文
相关资源
相关搜索

当前位置:首页 > 商业/管理/HR > 企业文档

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号