工作以来,写了很多socket相关的代码。磕磕碰碰,走了很多弯路,也积累了一些东西,今天正好整理一下。为了证明不是从书上抄来的,逻辑会有点乱(借口,呵呵)!知识点的介绍也不会像书上说的那么详细和精准,毕竟个人水平也就这样了。当然,主要还是以上手为主,不过分剖析原理性内容。一些陌生的函数要用到的头文件,使用man查看一下就能解决了。既然该文的名称为“快速上手”,那个人认为下述内容都不存在水分,都是必须要掌握的,一点都不能急躁!

一、socket连接流程:

对于程序员来说,开始的时候只会把socket编程当成一个工具,尽快上手,尽快解决战斗。于是乎最关心的就是socket那些函数的调用顺序,那就先给出UDP/TCP的流程图(从《UNIX网络编程》)吧:

有了流程图,再找一些资料,就很容易写出下面这样的代码(以TCP为例):

服务器程序:

 #include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h> #define PORT 1234
#define BACKLOG 5
#define MAXDATASIZE 1000 int main()
{
int listenfd, connectfd;
struct sockaddr_in server;
struct sockaddr_in client;
socklen_t addrlen;
char szbuf[MAXDATASIZE] = {}; if ((listenfd = socket(AF_INET, SOCK_STREAM, )) == -)
{
perror("Creating socket failed.");
exit();
} bzero(&server, sizeof(server));
server.sin_family = AF_INET;
server.sin_port = htons(PORT);
server.sin_addr.s_addr = htonl(INADDR_ANY);
if (bind(listenfd, (struct sockaddr *)&server, \
sizeof (server)) == -)
{
perror("Bind()error.");
exit();
}
if (listen(listenfd, BACKLOG) == -)
{
perror("listen()error\n");
exit();
} addrlen = sizeof(client);
if ((connectfd = accept(listenfd, \
(struct sockaddr*)&client, &addrlen)) == -)
{
perror("accept()error\n");
exit();
}
printf("You got a connection from cient's ip is %s, \
prot is %d\n", inet_ntoa(client.sin_addr), \
htons(client.sin_port)); memset(szbuf, 'a', sizeof(szbuf));
while ()
{
send(connectfd, szbuf, sizeof(szbuf), );
} close(connectfd);
close(listenfd); return ;
}

TCP服务器代码

客户端程序:

 #include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <netdb.h> #define PORT 1234
#define MAXDATASIZE 1000 int main(int argc, char *argv[])
{
int sockfd, num;
char buf[MAXDATASIZE + ] = {};
struct sockaddr_in server; if (argc != )
{
printf("Usage:%s <IP Address>\n", argv[]);
exit();
} if ((sockfd=socket(AF_INET, SOCK_STREAM, )) == -)
{
printf("socket()error\n");
exit();
}
bzero(&server, sizeof(server));
server.sin_family = AF_INET;
server.sin_port = htons(PORT);
server.sin_addr.s_addr = inet_addr(argv[]);
if (connect(sockfd, (struct sockaddr *)&server, \
sizeof(server)) == -)
{
printf("connect()error\n");
exit();
} while ()
{
memset(buf, , sizeof(buf));
if ((num = recv(sockfd, buf, \ MAXDATASIZE,)) == -)
{
printf("recv() error\n");
exit();
}
buf[num - ]='\0';
printf("Server Message: %s\n",buf);
} close(sockfd); return ;
}

TCP客户端代码

二、基础知识不能忽略:

有了这些代码,顺利地话一次性就能编过,也都正常跑起来。有些人就认为已经学会了socket,哈哈,当时我就是这样!更多的情况是,我们参考资料所用的运行环境往往和我们自己电脑不一致,就需要自己来修改代码。然后就会发现代码里出现AF_INET、htons、inet_addr、struct sockaddr……这些完全不知道干什么的东西。更有甚者,那人还不知道使用man,不喜欢看书,那就只能剩下猜谜了。运气好一点,最后能瞎猫碰见死老鼠,要是运气差一点,估计头就大了。

Socket编程的关键函数自然不能不懂,但基础知识同样重要,不理解基础知识到最后只会一团乱。书上写了好几页,其实内容也并不是太多。耐住性子弄明白,后面就会没那么添堵了。下面整理了三点:

1.网络选择:

一般编程而言,socket(特指socket这个系统调用)函数的第一个参数都使用AF_INET,表示是在IPv4因特网域。也可以改变为其它的值,用于IPv6网络等。

第二个参数为套接字类型,TCP链接使用SOCK_STREAM,UDP链接使用SOCK_DGRAM,基本就这两个。如果想写一个类似于ping这样直接访问网络层的一个程序,那就要使用其他类型了。

2.网络字节序:

和1不同,1设置不对你就别想正确建立链接,但是字节序问题跳过去,可能开始并不会造成太大的影响。所以字节序转换往往就被人忽略了。

字节序问题是由“大端系统”和“小端系统”这两者的差异造成的。网络通信时,我们并不知道对端系统采用什么样的存储方式,这个时候,就引入了一个网络字节序的概念。网络协议统一使用大端字节序,不管什么类型的主机,在通过协议栈发送和接收时就知道数据需不需要转换了。

另外,并不是所有数据都需要转换的。如果采用字节流(没有类似int,short这种多字节格式化数据类型)方式进行通信,无需字节序转换,因为数据都是有序的,不管在什么系统中顺序都是一致的。

对于字节序的转换关系,建议还是自己画图理解一下,光动脑子想不是太容易理解。

字节序转换函数不多,只有四个,并且很容易记忆,它们分别是:

htonl、htons、ntohl、ntohs

其中h表示“主机(host)字节序”,n表示“网络(network)字节序”,l表示“长(long)整数(4个字节)”,s表示“短(short)整数(2个字节)”,to表示“从谁向谁转换”……比Stevens的原著都详细,这样再无法记住,还真是没什么办法了。

有时也会涉及到更多字节的字节序转换,比如long long类型(8个字节),其实,用上述函数也是有办法能够完成的。

3.网络地址转换:

实际使用时,IPv4和IPv6使用的地址结构是不同的。IPv4使用struct sockaddr_in,而后者使用struct sockaddr_in。但不同网络(比如IPv4、IPv6)编程使用的socket接口函数都是一样的,这是怎么做到的呢?答案是接口函数都使用了统一的地址结构struct sockaddr。这些地址结构的具体定义可以使用man查看,可以发现,这些结构的框架是相同的,在正确地使用方法下地址结构间可以相互强制转换。它们都符合

struct sockaddr {
  unsigned short sa_family;     /* address family, AF_xxx */
  char sa_data[14];                 /* 14 bytes of protocol address */
  };

这种形式。因此,不管是在哪个网域下调用socket函数,只要将有差异的地址结构使用struct sockaddr进行强制类型转换就能实现接口的统一了。

接着,针对具体的IPv4编程中地址的赋值详细说明一下。上面提到,IPv4使用的地址结构及其定义为

struct sockaddr_in {

short sin_family; /* Address family */

unsigned short sin_port; /* Port number */

struct in_addr sin_addr; /* Internet address */

unsigned char sin_zero[8]; /* Same size as struct sockaddr */

};

在linux下:

in_addr结构

typedef struct in_addr{

unsigned long s_addr;

};

上面说到协议栈都需要使用网络字节序,sin_family自然是等于AF_INET(协议栈提供的值,不需要考虑字节序了)。sin_port(端口)是双字节变量,需要使用htons转字节序。sin_addr.s_addr(网络地址),它不是使用192.168.1.2这种点分十进制形式,而是需要将点分十进制地址转换成网络字节序的一个整数。这时就需要对192.168.1.2这种地址进行转换,系统提供了inet_addr函数(只能用于IPv4地址转换)。sin_addr.s_addr赋值为inet_addr(192.168.1.2)。能转过去就一定能变回来,系统提供了inet_ntoa函数进行逆变换。inet_addr和inet_ntoa只能适用于IPv4地址转换,系统还提供了inet_ntop和inet_pton函数,后两个函数兼容IPv4和IPv6,使用起来更加方便。

掌握这些基础知识之后写起代码来就不一定再需要拿着参考书来看了,这也是为一个健壮、稳定的网络程序做了热身运动,这可是必不可少的。

本来想一篇就写完的,写到这发现内容还真没想得那么少,在下一章会讲述网络程序的一些细节问题,也都是必不可少的。

socket网络编程快速上手(一)的更多相关文章

  1. socket网络编程快速上手(二)——细节问题(5)(完结篇)

    6.Connect的使用方式 前面提到,connect发生EINTR错误时,是不能重新启动的.那怎么办呢,是关闭套接字还是直接退出进程呢?如果EINTR前,三次握手已经发起,我们当然希望链路就此已经建 ...

  2. socket网络编程快速上手(二)——细节问题(4)

    5.慢系统调用及EINTR 还记得前面readn和writen函数么?里面有个EINTR,现在就来谈谈这个,这个很重要. Linux世界有个叫信号的东西,感觉他就像一位隐士,很少遇到他,而他又无处不在 ...

  3. socket网络编程快速上手(二)——细节问题(1)

    三.细节问题一个也不能少 Socket编程说简单也简单,程序很容易就能跑起来,说麻烦还真是麻烦,程序动不动就出问题.记得刚开始写网络代码的时候,那真是令人抓狂的经历,问题一个套一个,一会服务器起不来了 ...

  4. socket网络编程快速上手(二)——细节问题(3)

    3.SIGPIPE问题 人怕牺牲,我们写的程序也一样,人有死不瞑目,程序又何尝不是?程序跑着跑着,突然就崩掉了.好一点的牺牲前告诉你些打印,差点的也能用core文件等一些手段查出死在哪了,最惨不忍睹的 ...

  5. socket网络编程快速上手(二)——细节问题(2)

    2.TCP数据包接收问题 对初学者来说,很多都会认为:客户端与服务器最终的打印数据接收或者发送条数都该是一致的,1000条发送打印,1000条接收打印,长度都为1000.但是,事实上并不是这样,发送打 ...

  6. Java网络编程快速上手(SE基础)

    参考资料:百度百科TCP协议 本文涉及Java IO流.异常的知识,可参考我的另外的博客 一文简述Java IO 一文简述JAVA内部类和异常 1.概述 计算机网络相关知识: OSI七层模型 一个报文 ...

  7. SOCKET网络编程5

    SOCKET网络编程快速上手(二)——细节问题(5)(完结篇) 6.Connect的使用方式 前面提到,connect发生EINTR错误时,是不能重新启动的.那怎么办呢,是关闭套接字还是直接退出进程呢 ...

  8. SOCKET网络编程细节问题(4)

    SOCKET网络编程快速上手(二)——细节问题(4) 5.慢系统调用及EINTR 还记得前面readn和writen函数么?里面有个EINTR,现在就来谈谈这个,这个很重要. Linux世界有个叫信号 ...

  9. SOCKET网络编程细节问题3

    SOCKET网络编程快速上手(二)——细节问题(3) 3.SIGPIPE问题 人怕牺牲,我们写的程序也一样,人有死不瞑目,程序又何尝不是?程序跑着跑着,突然就崩掉了.好一点的牺牲前告诉你些打印,差点的 ...

随机推荐

  1. shell 命名管道,进程间通信

    命名管道基础 命名管道也被称为FIFO文件, 在文件系统中是可见的,并且跟其它文件一样可以读写! 命名管道特点: 当写进程向管道中写数据的时候,如果没有进程读取这些数据,写进程会堵塞 当读取管道中的数 ...

  2. jQuery插件编写及链式编程模型

    jQuery插件编写及链式编程模型小结 JQuery极大的提高了我们编写JavaScript的效率,让我们可以愉快的编写代码,做出各种特效.大多数情况下,我们都是使用别人开发的JQuery插件,今天我 ...

  3. 内网穿透&UDP打洞

    这两天找度度重新回忆了一下关于内网穿透的事情,在百度文库上找到了两三篇写的比较通俗易懂的文章,把内网穿透做个简单总结. 首先文章建议 Cone NAPT 还有希望,要是 Symmetri NAPT 就 ...

  4. ASP.NET MVC基于标注特性的Model验证:一个Model,多种验证规则

    原文:ASP.NET MVC基于标注特性的Model验证:一个Model,多种验证规则 对于Model验证,理想的设计应该是场景驱动的,而不是Model(类型)驱动的,也就是对于同一个Model对象, ...

  5. 【phpMyAdmin】更改配置文件连接到其他server

    默认phpMyAdmin安装完毕后对机器的访问mysql,但有时我们需要访问其它server的mysql数据库,所以我们需要配置. 真,phpMyAdmin已经为我们做了配置的选项.可是须要我们进行一 ...

  6. javascript 控制台输出 图片 console.log 真强大 真佩服你们的创造力

    无意中,在百度知道页面发现了这货.居然能输出图片到控制台. 完全颠覆自己的三观,果断查阅其输出方法.后得知,原来如此. 曾经做过的项目中,同事把控制台做成一个网页形式方便远程控制和远程调用.没想到过这 ...

  7. uva 11529 - Strange Tax Calculation(计数问题)

    题目链接:uva 11529 - Strange Tax Calculation 题目大意:给出若干个点,保证随意三点不共线.随意选三个点作为三角行,其它点若又在该三角形内,则算是该三角形内部的点.问 ...

  8. JS中的模块规范(CommonJS,AMD,CMD)

    JS中的模块规范(CommonJS,AMD,CMD) 如果你听过js模块化这个东西,那么你就应该听过或CommonJS或AMD甚至是CMD这些规范咯,我也听过,但之前也真的是听听而已. 现在就看看吧, ...

  9. orcale复制表结构及其数据

    http://hi.baidu.com/tag/Oracle/feeds http://hi.baidu.com/gqftuisidibabiq/item/14d306cc87cbdf45bcef69 ...

  10. Linux环境下搭建php开发环境的操作步骤

    本文主要记载了通过编译方式进行软件/开发环境的安装过程,其他安装方式忽略! 文章背景: 因为php和Apache等采用编译安装方式进行安装,然而编译安装方式,需要c,c++编译环境, 通过apt方式安 ...