Socket

一切皆 Socket

我们已经知道网络中的进程是通过 socket 来通信的,那什么是 socket 呢?

 socket 起源于 UNIX,而 UNIX/Linux 基本哲学之一就是「一切皆文件」,都可以用「open → write/read → close」模式来操作。 

socket 其实就是该模式的一个实现,socket 即是一种特殊的文件,一些 socket 函数就是对其进行的操作。

使用 TCP/IP 协议的应用程序通常采用系统提供的编程接口:UNIX BSD 的套接字接口(Socket Interfaces)以此来实现网络进程之间的通信。

 就目前而言,几乎所有的应用程序都是采用 socket,所以说现在的网络时代,网络中进程通信是无处不在,一切皆 socket

套接字接口 Socket Interfaces

套接字接口是一组函数,由操作系统提供,用以创建网络应用。 大多数现代操作系统都实现了套接字接口,包括所有 Unix 变种,Windows 和 Macintosh 系统。

套接字接口的起源套接字接口是加州大学伯克利分校的研究人员在 20 世纪 80 年代早起提出的。 伯克利的研究者使得套接字接口适用于任何底层的协议,第一个实现就是针对 TCP/IP 协议,他们把它包括在 Unix 4.2 BSD 的内核里,并且分发给许多学校和实验室。 这在因特网的历史成为了一个重大事件。 —— 《深入理解计算机系统》

从 Linux 内核的角度来看,一个套接字就是通信的一个端点。 从 Linux 程序的角度来看,套接字是一个有相应描述符的文件。 普通文件的打开操作返回一个文件描述字,而 socket() 用于创建一个 socket 描述符,唯一标识一个 socket。 这个 socket 描述字跟文件描述字一样,后续的操作都有用到它,把它作为参数,通过它来进行一些操作。

常用的函数有:

socket()

bind()

listen()

connect()

accept()

write()

read()

close()

Socket 的交互流程

图中展示了 TCP 协议的 socket 交互流程,描述如下:

服务器根据地址类型、socket 类型、以及协议来创建 socket。

服务器为 socket 绑定 IP 地址和端口号。

服务器 socket 监听端口号请求,随时准备接收客户端发来的连接,这时候服务器的 socket 并没有全部打开。

客户端创建 socket。

客户端打开 socket,根据服务器 IP 地址和Port端口号试图连接服务器 socket

服务器 socket 接收到客户端 socket 请求,被动打开,开始接收客户端请求,直到客户端返回连接信息。这时候 socket 进入阻塞状态,阻塞是由于 accept() 方法会一直等到客户端返回连接信息后才返回,然后开始连接下一个客户端的连接请求。

客户端连接成功,向服务器发送连接状态信息。

服务器 accept() 方法返回,连接成功。

服务器和客户端通过网络 I/O 函数进行数据的传输。

客户端关闭 socket。

服务器关闭 socket。

这个过程中,服务器和客户端建立连接的部分,就体现了 TCP 三次握手的原理。

下面详细讲一下 socket 的各函数。

Socket 接口

socket 是系统提供的接口,而操作系统大多数都是用 C/C++ 开发的,自然函数库也是 C/C++ 代码。

socket 函数

该函数会返回一个套接字描述符(socket descriptor),但是该描述符仅是部分打开的,还不能用于读写。 如何完成打开套接字的工作,取决于我们是客户端还是服务器。

函数原型

#includeintsocket(intdomain,inttype,intprotocol);

参数说明

domain: 协议域,决定了 socket 的地址类型,在通信中必须采用对应的地址。 常用的协议族有:AF_INET(ipv4地址与端口号的组合)、AF_INET6(ipv6地址与端口号的组合)、AF_LOCAL(绝对路径名作为地址)。 该值的常量定义在sys/socket.h文件中。

type: 指定 socket 类型。 常用的类型有:SOCK_STREAM、SOCK_DGRAM、SOCK_RAW、SOCK_PACKET、SOCK_SEQPACKET等。 其中SOCK_STREAM表示提供面向连接的稳定数据传输,即 TCP 协议。 该值的常量定义在sys/socket.h文件中。

protocol: 指定协议。 常用的协议有:IPPROTO_TCP(TCP协议)、IPPTOTO_UDP(UDP协议)、IPPROTO_SCTP(STCP协议)。 当值位 0 时,会自动选择type类型对应的默认协议。

bind 函数

由服务端调用,把一个地址族中的特定地址和 socket 联系起来。

函数原型

#includeintbind(intsockfd,conststructsockaddr *addr, socklen_t addrlen);

参数说明

sockfd: 即 socket 描述字,由 socket() 函数创建。

*addr: 一个const struct sockaddr指针,指向要绑定给sockfd的协议地址。 这个地址结构根据地址创建 socket 时的地址协议族不同而不同,例如 ipv4 对应sockaddr_in,ipv6 对应sockaddr_in6. 这几个结构体在使用的时候,都可以强制转换成sockaddr。 下面是这几个结构体对应的所在的头文件:

sockaddr:sys/socket.h

sockaddr_in:netinet/in.h

sockaddr_in6:netinet6/in.h

_in 后缀意义:互联网络(internet)的缩写,而不是输入(input)的缩写。

listen 函数

服务器调用,将 socket 从一个主动套接字转化为一个监听套接字(listening socket), 该套接字可以接收来自客户端的连接请求。 在默认情况下,操作系统内核会认为 socket 函数创建的描述符对应于主动套接字(active socket)。

函数原型

#includeintlisten(intsockfd,intbacklog);

参数说明

sockfd: 即 socket 描述字,由 socket() 函数创建。

backlog: 指定在请求队列中的最大请求数,进入的连接请求将在队列中等待 accept() 它们。

connect 函数

由客户端调用,与目的服务器的套接字建立一个连接。

函数原型

#includeintconnect(intclientfd,conststructsockaddr *addr, socklen_t addrlen);

参数说明

clientfd: 目的服务器的 socket 描述符

*addr: 一个const struct sockaddr指针,包含了目的服务器 IP 和端口。

addrlen: 协议地址的长度,如果是 ipv4 的 TCP 连接,一般为sizeof(sockaddr_in);

accept 函数

服务器调用,等待来自客户端的连接请求。 当客户端连接,accept 函数会在addr中会填充上客户端的套接字地址,并且返回一个已连接描述符(connected descriptor),这个描述符可以用来利用 Unix I/O 函数与客户端通信。

函数原型

#indclude intaccept(intlistenfd,structsockaddr *addr,int*addrlen);

参数说明

listenfd: 服务器的 socket 描述字,由 socket() 函数创建。

*addr: 一个const struct sockaddr指针,用来存放提出连接请求客户端的主机的信息

*addrlen: 协议地址的长度,如果是 ipv4 的 TCP 连接,一般为sizeof(sockaddr_in)。

close 函数

在数据传输完成之后,手动关闭连接。

函数原型

#include#includeintclose(intfd);

参数说明

fd: 需要关闭的连接 socket 描述符

网络 I/O 函数

当客户端和服务器建立连接后,可以使用网络 I/O 进行读写操作。 网络 I/O 操作有下面几组:

read()/write()

recv()/send()

readv()/writev()

recvmsg()/sendmsg()

recvfrom()/sendto()

最常用的是 read()/write() 他们的原型是:

ssize_tread(intfd,void*buf,size_tcount);ssize_twrite(intfd,constvoid*buf,size_tcount);

鉴于该文是侧重于描述 socket 的工作原理,就不再详细描述这些函数了。

实现一个简单 TCP 交互

服务端

// socket_server.cpp#include#include#include#include#include#include#include#include#defineMAXLINE 4096// 4 * 1024intmain(intargc,char**argv){intlistenfd,// 监听端口的 socket 描述符connfd;// 连接端 socket 描述符structsockaddr_in servaddr;charbuff[MAXLINE];intn;// 创建 socket,并且进行错误处理if((listenfd = socket(AF_INET, SOCK_STREAM,0)) ==-1)    {printf("create socket error: %s(errno: %d)\n", strerror(errno), errno);return0;    }// 初始化 sockaddr_in 数据结构memset(&servaddr,0,sizeof(servaddr));    servaddr.sin_family = AF_INET;    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);    servaddr.sin_port = htons(6666);// 绑定 socket 和 端口if(bind(listenfd, (structsockaddr *)&servaddr,sizeof(servaddr)) ==-1)    {printf("bind socket error: %s(errno: %d)\n", strerror(errno), errno);return0;    }// 监听连接if(listen(listenfd,10) ==-1)    {printf("listen socket error: %s(errno: %d)\n", strerror(errno), errno);return0;    }printf("====== Waiting for client's request======\n");// 持续接收客户端的连接请求while(true)    {if((connfd = accept(listenfd, (structsockaddr *)NULL,NULL) ==-1))        {printf("accept socket error: %s(errno: %d)\n", strerror(errno), errno);continue;        }        n = recv(connfd, buff, MAXLINE,0);        buff[n] ='\0';printf("recv msg from client: %s\n", buff);        close(connfd);    }    close(listenfd);return0;}

客户端

// socket_client.cpp#include#include#include#include#include#include#include#include#include#defineMAXLINE 4096intmain(intargc,char**argv){intsockfd, n;charrecvline[4096], sendline[4096];structsockaddr_in servaddr;if(argc !=2)    {printf("usage: ./client \n");return0;    }// 创建 socket 描述符if((sockfd = socket(AF_INET, SOCK_STREAM,0)) <0)    {printf("create socket error: %s(errno: %d)\n", strerror(errno), errno);return0;    }// 初始化目标服务器数据结构memset(&servaddr,0,sizeof(servaddr));    servaddr.sin_family = AF_INET;    servaddr.sin_port = htons(6666);// 从参数中读取 IP 地址if(inet_pton(AF_INET, argv[1], &servaddr.sin_addr) <=0)    {printf("inet_pton error for %s\n", argv[1]);return0;    }// 连接目标服务器,并和 sockfd 联系起来。if(connect(sockfd, (structsockaddr *)&servaddr,sizeof(servaddr)) <0)    {printf("connect error: %s(errno: %d)\n", strerror(errno), errno);return0;    }printf("send msg to server: \n");// 从标准输入流中读取信息fgets(sendline,4096,stdin);// 通过 sockfd,向目标服务器发送信息if(send(sockfd, sendline,strlen(sendline),0) <0)    {printf("send msg error: %s(errno: %d)\n", strerror(errno), errno);return0;    }// 数据传输完毕,关闭 socket 连接close(sockfd);return0;}

Run

首先创建makefile文件

all:server client

server:socket_server.o

g++ -g -o socket_server socket_server.o

client:socket_client.o

g++ -g -o socket_client socket_client.o

socket_server.o:socket_server.cpp

g++ -g -c socket_server.cpp

socket_client.o:socket_client.cpp

g++ -g -c socket_client.cpp

clean:all

rm all

然后使用命令:

$ make

会生成两个可执行文件:

socket_server

socket_client

分别打开两个终端,运行:

./socket_server

./socket_client 127.0.0.1

然后在socket_client中键入发送内容,可以再socket_server接收到同样的信息。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 161,601评论 4 369
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,367评论 1 305
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 111,249评论 0 254
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,539评论 0 217
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,967评论 3 295
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,929评论 1 224
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,098评论 2 317
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,825评论 0 207
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,609评论 1 249
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,796评论 2 253
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,282评论 1 265
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,603评论 3 261
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,277评论 3 242
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,159评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,959评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 36,079评论 2 285
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,874评论 2 277

推荐阅读更多精彩内容

  • 一、基本socket函数Linux系统是通过提供套接字(socket)来进行网络编程的。网络的socket数据传输...
    WB莫遥燚阅读 1,434评论 0 0
  • 一、网络各个协议:TCP/IP、SOCKET、HTTP等 网络七层由下往上分别为物理层、数据链路层、网络层、传输层...
    杯水救车薪阅读 2,209评论 0 17
  • C语言 粗体字公共部分 斜体字为服务器 其余为客户端 服务器客户端 #include #include #pra...
    dovlie阅读 2,956评论 0 1
  • 简介 Socket理论 Socket工作流程 核心函数讲解 服务的如何获取客户端的信息 字符串ip和网络二进制的转...
    第八区阅读 3,437评论 0 4
  • 公司做了绅宝D50的上市品鉴会,找到几年前购买上代产品以及小E系列的车主清单,近一周的回访邀约希望可以过来品鉴新车...
    丁陈霞阅读 97评论 0 0