简单理解Socket
TCP/IP:
要想理解socket首先得熟悉一下TCP/IP协议族, TCP/IP(Transmission Control Protocol/Internet Protocol)即传输控制协议/网间协议,定义了主机如何连入因特网及数据如何再它们之间传输的标准,
从字面意思来看TCP/IP是TCP和IP协议的合称,但实际上TCP/IP协议是指因特网整个TCP/IP协议族。不同于ISO模型的七个分层,TCP/IP协议参考模型把所有的TCP/IP系列协议归类到四个抽象层中
应用层:TFTP,HTTP,SNMP,FTP,SMTP,DNS,Telnet 等等
传输层:TCP,UDP
网络层:IP,ICMP,OSPF,EIGRP,IGMP
数据链路层:SLIP,CSLIP,PPP,MTU
每一抽象层建立在低一层提供的服务上,并且为高一层提供服务,看起来大概是这样子的:
在TCP/IP协议中两个因特网主机通过两个路由器和对应的层连接。各主机上的应用通过一些数据通道相互执行读取操作。
socket:
我们知道两个进程如果需要进行通讯最基本的一个前提能能够唯一的标示一个进程,在本地进程通讯中我们可以使用PID来唯一标示一个进程,但PID只在本地唯一,网络中的两个进程PID冲突几率很大,这时候我们需要另辟它径了,我们知道IP层的ip地址可以唯一标示主机,而TCP层协议和端口号可以唯一标示主机的一个进程,这样我们可以利用ip地址+协议+端口号唯一标示网络中的一个进程。
能够唯一标示网络中的进程后,它们就可以利用socket进行通信了,什么是socket呢?我们经常把socket翻译为套接字,socket是在应用层和传输层之间的一个抽象层,它把TCP/IP层复杂的操作抽象为几个简单的接口供应用层调用已实现进程在网络中通信。
socket起源于UNIX,在Unix一切皆文件哲学的思想下,socket是一种”打开—读/写—关闭”模式的实现,服务器和客户端各自维护一个”文件”,在建立连接打开后,可以向自己文件写入内容供对方读取或者读取对方内容,通讯结束时关闭文件。
socket通信流程
socket是”打开—读/写—关闭”模式的实现,以使用TCP协议通讯的socket为例,其交互流程大概是这样子的:
服务器根据地址类型(ipv4,ipv6)、socket类型、协议创建socket。
服务器为socket绑定ip地址和端口号。
服务器socket监听端口号请求,随时准备接收客户端发来的连接,这时候服务器的socket并没有被打开。
客户端创建socket。
客户端打开socket,根据服务器ip地址和端口号试图连接服务器socket。
服务器socket接收到客户端socket请求,被动打开,开始接收客户端请求,直到客户端返回连接信息。这时候socket进入阻塞状态,所谓阻塞即accept()方法一直到客户端返回连接信息后才返回,开始接收下一个客户端连接请求。
客户端连接成功,向服务器发送连接状态信息。
服务器accept方法返回,连接成功。
客户端向socket写入信息。
服务器读取信息。
客户端关闭。
服务器端关闭。
三次握手:
在TCP/IP协议中,TCP协议通过三次握手建立一个可靠的连接。
第一次握手:客户端尝试连接服务器,向服务器发送syn包(同步序列编号Synchronize Sequence Numbers),syn=j,客户端进入SYN_SEND状态等待服务器确认。
第二次握手:服务器接收客户端syn包并确认(ack=j+1),同时向客户端发送一个SYN包(syn=k),即SYN+ACK包,此时服务器进入SYN_RECV状态。
第三次握手:第三次握手:客户端收到服务器的SYN+ACK包,向服务器发送确认包ACK(ack=k+1),此包发送完毕,客户端和服务器进入ESTABLISHED状态,完成三次握手。
服务器socket与客户端socket建立连接的部分其实就是大名鼎鼎的三次握手。
下面是一个用php实现的socket通信:
<?php
//server1.php
/*服务器端代码*/
//确保在连接客户端时不会超时
set_time_limit(0);
$ip = '127.0.0.1';
$port = 1935;
/*
*
+-------------------------
*socket通信过程
*
+-------------------------
*/
if(($socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP)) < 0) {
echo 'socket_create()失败,原因是:'.socket_strerror($socket)."\n";
}
if(($ret = socket_bind($socket, $ip, $port)) < 0) {
echo 'socket_bind()失败,原因是:'.socket_strerror($ret)."\n";
}
if(($ret = socket_listen($socket, 4)) < 0) {
echo 'socket_listen失败,原因是:'.socket_strerror($ret)."\n";
}
echo "Binding the socket on $ip:$port ... \n";
$count = 0;
do {
if(($msgsock = socket_accept($socket)) < 0) {
echo 'socket_accept() failed: reason: ' . socket_strerror($msgsock) . "\n";
break;
} else {
echo "测试成功啦\n";
$buf = socket_read($msgsock, 8192);
$talkback = "收到的信息:{$buf}\n";
echo $talkback;
//发送到客户端
$msg = $buf." 你好!\n";
socket_write($msgsock, $msg, strlen($msg));
if(++$count >= 5) {
break;
}
socket_close($msgsock);
} while (true);
socket_close($socket);
使用php server1.php开启服务端监听:
现在什么都没有。
下面启用客户端socket:
客户端代码:
<?php
//client1.php
set_time_limit(0);
echo "+--------TCP/IP Connection--------+\n\n";
$ip = '127.0.0.1';
$port = 1935;
$socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
if($socket < 0) {
echo 'socket_create() failed: reasion: '. socket_strerror($socket) . "\n";
} else {
echo "ok.\n";
}
echo "试图连接‘{$ip}‘端口‘{$port}’......\n";
$result = socket_connect($socket, $ip, $port);
if($result < 0) {
echo 'socket_connect() failed .\n Reason: ($result) '. socket_strerror($result) . "\n";
} else {
echo "连接OK\n\n";
}
$in = "张三: ";
//$in .= "first blood \r\n";
$out = '';
if(!socket_write($socket, $in, strlen($in))) {
echo 'socket_write() failed: reason: ' . socket_strerror($socket) . "\n";
} else {
echo "发送到服务器信息成功!\n";
echo "发送的内容是:**{$in}**\n";
}
while($out = socket_read($socket, 8192)) {
echo "\n接收服务器回转信息成功!\n";
echo "接收的内容为:###", $out;
}
echo "\n关闭socket...\n";
socket_close($socket);
echo "关闭成功\n";
通过php client1.php启用客户端socket,启用后效果如下图:
[客户端效果图]
[服务端效果图]
一个用php实现的简单socket就完成了。