zoukankan      html  css  js  c++  java
  • 深入浅出讲解:php的socket通信

    TCP/IP、UDP、Socket编程这些词你不会很陌生吧?随着网络技术的发展,这些词充斥着我们的耳朵。那么我想问:

    1.         什么是TCP/IP、UDP?
    2.         Socket在哪里呢?
    3.         Socket是什么呢?
    4.         你会使用它们吗?

    什么是TCP/IPUDP

             TCP/IP(Transmission Control Protocol/Internet Protocol)即传输控制协议/网间协议,是一个工业标准的协议集,它是为广域网(WANs)设计的。
             UDP(User Data Protocol,用户数据报协议)是与TCP相对应的协议。它是属于TCP/IP协议族中的一种。
            这里有一张图,表明了这些协议的关系。

      TCP/IP协议族包括运输层、网络层、链路层。现在你知道TCP/IP与UDP的关系了吧。
    Socket在哪里呢?
      在图1中,我们没有看到Socket的影子,那么它到底在哪里呢?还是用图来说话,一目了然。

     

    原来Socket在这里。
    Socket是什么呢?
      Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP /IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。
    你会使用它们吗?
      前人已经给我们做了好多的事了,网络间的通信也就简单了许多,但毕竟还是有挺多工作要做的。以前听到Socket编程,觉得它是比较高深的编程知识,但是只要弄清Socket编程的工作原理,神秘的面纱也就揭开了。
      一个生活中的场景。你要打电话给一个朋友,先拨号,朋友听到电话铃声后提起电话,这时你和你的朋友就建立起了连接,就可以讲话了。等交流结束,挂断电 话结束此次交谈。 生活中的场景就解释了这工作原理,也许TCP/IP协议族就是诞生于生活中,这也不一定。

      先从服务器端说起。服务器端先初始化Socket,然后与端口绑定(bind),对端口进行监听(listen),调用accept阻塞,等待 客户端连接。在这时如果有个客户端初始化一个Socket,然后连接服务器(connect),如果连接成功,这时客户端与服务器端的连接就建立了。客户 端发送数据请求,服务器端接收请求并处理请求,然后把回应数据发送给客户端,客户端读取数据,最后关闭连接,一次交互结束。

    socket相关函数:
    ----------------------------------------------------------------------------------------------
    socket_accept() 接受一个Socket连接
    socket_bind() 把socket绑定在一个IP地址和端口上
    socket_clear_error() 清除socket的错误或者最后的错误代码
    socket_close() 关闭一个socket资源
    socket_connect() 开始一个socket连接
    socket_create_listen() 在指定端口打开一个socket监听
    socket_create_pair() 产生一对没有区别的socket到一个数组里
    socket_create() 产生一个socket,相当于产生一个socket的数据结构
    socket_get_option() 获取socket选项
    socket_getpeername() 获取远程类似主机的ip地址
    socket_getsockname() 获取本地socket的ip地址
    socket_iovec_add() 添加一个新的向量到一个分散/聚合的数组
    socket_iovec_alloc() 这个函数创建一个能够发送接收读写的iovec数据结构
    socket_iovec_delete() 删除一个已经分配的iovec
    socket_iovec_fetch() 返回指定的iovec资源的数据
    socket_iovec_free() 释放一个iovec资源
    socket_iovec_set() 设置iovec的数据新值
    socket_last_error() 获取当前socket的最后错误代码
    socket_listen() 监听由指定socket的所有连接
    socket_read() 读取指定长度的数据
    socket_readv() 读取从分散/聚合数组过来的数据
    socket_recv() 从socket里结束数据到缓存
    socket_recvfrom() 接受数据从指定的socket,如果没有指定则默认当前socket
    socket_recvmsg() 从iovec里接受消息
    socket_select() 多路选择
    socket_send() 这个函数发送数据到已连接的socket
    socket_sendmsg() 发送消息到socket
    socket_sendto() 发送消息到指定地址的socket
    socket_set_block() 在socket里设置为块模式
    socket_set_nonblock() socket里设置为非块模式
    socket_set_option() 设置socket选项
    socket_shutdown() 这个函数允许你关闭读、写、或者指定的socket
    socket_strerror() 返回指定错误号的详细错误
    socket_write() 写数据到socket缓存
    socket_writev() 写数据到分散/聚合数组

    案例一:socket通信演示

    服务器端:

    复制代码
     1 <?php
     2 //确保在连接客户端时不会超时
     3 set_time_limit(0);
     4 
     5 $ip = '127.0.0.1';
     6 $port = 1935;
     7 
     8 /*
     9  +-------------------------------
    10  *    @socket通信整个过程
    11  +-------------------------------
    12  *    @socket_create
    13  *    @socket_bind
    14  *    @socket_listen
    15  *    @socket_accept
    16  *    @socket_read
    17  *    @socket_write
    18  *    @socket_close
    19  +--------------------------------
    20  */
    21 
    22 /*----------------    以下操作都是手册上的    -------------------*/
    23 if(($sock = socket_create(AF_INET,SOCK_STREAM,SOL_TCP)) < 0) {
    24     echo "socket_create() 失败的原因是:".socket_strerror($sock)."
    ";
    25 }
    26 
    27 if(($ret = socket_bind($sock,$ip,$port)) < 0) {
    28     echo "socket_bind() 失败的原因是:".socket_strerror($ret)."
    ";
    29 }
    30 
    31 if(($ret = socket_listen($sock,4)) < 0) {
    32     echo "socket_listen() 失败的原因是:".socket_strerror($ret)."
    ";
    33 }
    34 
    35 $count = 0;
    36 
    37 do {
    38     if (($msgsock = socket_accept($sock)) < 0) {
    39         echo "socket_accept() failed: reason: " . socket_strerror($msgsock) . "
    ";
    40         break;
    41     } else {
    42         
    43         //发到客户端
    44         $msg ="测试成功!
    ";
    45         socket_write($msgsock, $msg, strlen($msg));
    46         
    47         echo "测试成功了啊
    ";
    48         $buf = socket_read($msgsock,8192);
    49         
    50         
    51         $talkback = "收到的信息:$buf
    ";
    52         echo $talkback;
    53         
    54         if(++$count >= 5){
    55             break;
    56         };
    57         
    58     
    59     }
    60     //echo $buf;
    61     socket_close($msgsock);
    62 
    63 } while (true);
    64 
    65 socket_close($sock);
    66 ?>
    复制代码

    这是socket的服务端代码。然后运行cmd,注意是自己的程序存放路径啊。

    没有反映,对现在服务端的程序已经开始运行,端口已经开始监听了。运行netstat -ano可以查看端口情况,我的是1935端口

    看,端口已经处于LISTENING状态了。接下来我们只要运行客户端程序即可连接上。上代码

    复制代码
     1 <?php
     2 error_reporting(E_ALL);
     3 set_time_limit(0);
     4 echo "<h2>TCP/IP Connection</h2>
    ";
     5 
     6 $port = 1935;
     7 $ip = "127.0.0.1";
     8 
     9 /*
    10  +-------------------------------
    11  *    @socket连接整个过程
    12  +-------------------------------
    13  *    @socket_create
    14  *    @socket_connect
    15  *    @socket_write
    16  *    @socket_read
    17  *    @socket_close
    18  +--------------------------------
    19  */
    20 
    21 $socket = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
    22 if ($socket < 0) {
    23     echo "socket_create() failed: reason: " . socket_strerror($socket) . "
    ";
    24 }else {
    25     echo "OK.
    ";
    26 }
    27 
    28 echo "试图连接 '$ip' 端口 '$port'...
    ";
    29 $result = socket_connect($socket, $ip, $port);
    30 if ($result < 0) {
    31     echo "socket_connect() failed.
    Reason: ($result) " . socket_strerror($result) . "
    ";
    32 }else {
    33     echo "连接OK
    ";
    34 }
    35 
    36 $in = "Ho
    ";
    37 $in .= "first blood
    ";
    38 $out = '';
    39 
    40 if(!socket_write($socket, $in, strlen($in))) {
    41     echo "socket_write() failed: reason: " . socket_strerror($socket) . "
    ";
    42 }else {
    43     echo "发送到服务器信息成功!
    ";
    44     echo "发送的内容为:<font color='red'>$in</font> <br>";
    45 }
    46 
    47 while($out = socket_read($socket, 8192)) {
    48     echo "接收服务器回传信息成功!
    ";
    49     echo "接受的内容为:",$out;
    50 }
    51 
    52 
    53 echo "关闭SOCKET...
    ";
    54 socket_close($socket);
    55 echo "关闭OK
    ";
    56 ?>
    复制代码

    至此客户端已经连接上服务端了。

    案例二:代码详解

    // 设置一些基本的变量
    $host "192.168.1.99";
    $port 1234;
    // 设置超时时间
    set_time_limit(0);
    // 创建一个Socket
    $socket socket_create(AF_INETSOCK_STREAM0) or die("Could not createsocket ");
    //绑定Socket到端口
    $result socket_bind($socket$host$port) or die("Could not bind tosocket ");
    // 开始监听链接
    $result socket_listen($socket3) or die("Could not set up socketlistener ");
    // accept incoming connections
    // 另一个Socket来处理通信
    $spawn socket_accept($socket) or die("Could not accept incomingconnection ");
    // 获得客户端的输入
    $input socket_read($spawn1024) or die("Could not read input ");
    // 清空输入字符串
    $input trim($input);
    //处理客户端输入并返回结果
    $output strrev($input) . " ";
    socket_write($spawn$outputstrlen ($output)) or die("Could not write
    output ");
    // 关闭sockets
    socket_close($spawn);
    socket_close($socket);

    下面是其每一步骤的详细说明:

    1.第一步是建立两个变量来保存Socket运行的服务器的IP地址和端口.你可以设置为你自己的服务器和端口(这个端口可以是1到65535之间的数字),前提是这个端口未被使用.

    [Copy to clipboard]
    PHP CODE:
    // 设置两个变量
    $host "192.168.1.99";
    $port 1234;

    2.在服务器端可以使用set_time_out()函数来确保PHP在等待客户端连接时不会超时.

    [Copy to clipboard]
    PHP CODE:
    // 超时时间
    set_time_limit(0);

    3.在前面的基础上,现在该使用socket_creat()函数创建一个Socket了—这个函数返回一个Socket句柄,这个句柄将用在以后所有的函数中.

    [Copy to clipboard]
    PHP CODE:
    // 创建Socket
    $socket socket_create(AF_INETSOCK_STREAM0) or die("Could not create
    socket ");

    第一个参数”AF_INET”用来指定域名;
    第二个参数”SOCK_STREM”告诉函数将创建一个什么类型的Socket(在这个例子中是TCP类型)

    因此,如果你想创建一个UDP Socket的话,你可以使用如下的代码:

    [Copy to clipboard]
    PHP CODE:
    // 创建 socket
    $socket socket_create(AF_INETSOCK_DGRAM0) or die("Could not create
    socket ");

    4.一旦创建了一个Socket句柄,下一步就是指定或者绑定它到指定的地址和端口.这可以通过socket_bind()函数来完成.

    [Copy to clipboard]
    PHP CODE:
    // 绑定 socket to 指定地址和端口
    $result socket_bind($socket$host$port) or die("Could not bind to
    socket ");

    5.当Socket被创建好并绑定到一个端口后,就可以开始监听外部的连接了.PHP允许你由socket_listen()函数来开始一个监听,同时你可以指定一个数字(在这个例子中就是第二个参数:3)

    [Copy to clipboard]
    PHP CODE:
    // 开始监听连接
    $result socket_listen($socket3) or die("Could not set up socket
    listener ");

    6.到现在,你的服务器除了等待来自客户端的连接请求外基本上什么也没有做.一旦一个客户端的连接被收到,socket_accept()函数便开始起作用了,它接收连接请求并调用另一个子Socket来处理客户端–服务器间的信息.

    [Copy to clipboard]
    PHP CODE:
    //接受请求链接
    // 调用子socket 处理信息
    $spawn socket_accept($socket) or die("Could not accept incoming
    connection ");

    这个子socket现在就可以被随后的客户端–服务器通信所用了.

    7.当一个连接被建立后,服务器就会等待客户端发送一些输入信息,这写信息可以由socket_read()函数来获得,并把它赋值给PHP的$input变量.

    [Copy to clipboard]
    PHP CODE:
    // 读取客户端输入
    $input socket_read($spawn1024) or die("Could not read input ");
    ?&gt;

    socker_read的第而个参数用以指定读入的字节数,你可以通过它来限制从客户端获取数据的大小.

    注意:socket_read函数会一直读取壳户端数据,直到遇见 , 或者字符.PHP脚本把这写字符看做是输入的结束符.

    8.现在服务器必须处理这些由客户端发来是数据(在这个例子中的处理仅仅包含数据的输入和回传到客户端).这部分可以由socket_write()函数来完成(使得由通信socket发回一个数据流到客户端成为可能)

    [Copy to clipboard]
    PHP CODE:
    // 处理客户端输入并返回数据
    $output strrev($input) . " ";
    socket_write($spawn$outputstrlen ($output)) or die("Could not write
    output ");

    9.一旦输出被返回到客户端,父/子socket都应通过socket_close()函数来终止

    [Copy to clipboard]
    PHP CODE:
    // 关闭 sockets
    socket_close($spawn);
    socket_close($socket);

    分类: PHP
    好文要顶 关注我 收藏该文
    12
    0
     
    « 上一篇:ecshop调试
    » 下一篇:轮循与连接
    posted @ 2013-02-26 20:32 洒洒 阅读(74695) 评论(27) 编辑 收藏


    #1楼 2013-07-25 09:59 momo_Unique  
    不行啊,那个服务端代码在cmd运行后,查看netstat -ano时,端口没显示连上了,,这是咋回事啊?紧急~~~@洒洒
    #2楼[楼主] 2013-07-25 13:24 洒洒  
    看看代码哪里有问题没?如果不是,那么端口和ip要正确。
    #3楼 2013-07-25 16:14 吾非名家  
    我连在cmd运行php都不行。。。装的也是wamp集成环境。。怎么回事啊。。。@洒洒
    #4楼 2013-07-25 18:12 吾非名家  
    cmd下不能执行php 命令的问题解决了。。不过服务端代码好像有些问题。。。一直没达到演示的结果(好吧,我是复制的),因为一直提示“call to undefinded function socket_create()”的错误。。确认了socket模块开启后还是这样。。。
    #5楼[楼主] 2013-07-26 09:23 洒洒  
    @ 吾非名家
    哦,那是怎么解决的呢?另外你socket确定模块开启了吗?socket_create()是socket模块的内置函数,如果你开启了socket就不会提示socket_create()未定义。
    #6楼 2013-07-26 09:44 吾非名家  
    cmd执行php命令只 要设置环境变量就行了。。。然后,那个socket_create()函数的问题。。我在php.ini里边确认 extension=php_gd2.dll和extension=php_sockets.dll这两个前边是没有分号的,然后phpInfo()里边 的内容看到sockets的模块是enabled,应该是打开了吧。。。。可是现在还是。。。。。难道是win8系统的问题么。。。
    #7楼 2013-07-26 10:19 吾非名家  
    @洒洒 删掉上一条只艾特你的评论吧。。。
    现在socket_create()函数没有问题了,因为我一直改界面的ini,而界面的ini其实是apache文件夹内的,改了php文件夹内的php.ini后,再拷贝到C:/WINDOWS下就解决了。不过我运行服务端的socket时,cmd界面出现了:
    <html>
    <body>
    的情况。。。而不是本文提及的“没情况出现”,而且运行netstat -ano后,压根找不到自己设置的端口。。。请问这是什么情况呢???@洒洒
    #8楼[楼主] 2013-07-26 10:59 洒洒  
    @ 吾非名家
    我刚把服务器端代码全复制测试了,没问题。
    你看看你php.ini配置对吗?另外能正常在cmd下运行php脚本吗?比如php test.php
    #9楼 2013-07-26 15:48 吾非名家  
    @洒洒
    我知道什么问题了。。我运行了socket服务端的php之后就把cmd给关了,我后来另外开一个cmd来运行netstat -ano查看端口确认成功了。。。。。问一个问题啊,执行客户端的php时,之前运行了服务端的php是不是也不能关掉的??
    #10楼[楼主] 2013-07-26 16:11 洒洒  
    @ 吾非名家
    是的,因为你要开启服务器端来监听客户端发送的消息。如果你关了就见听不到了。
    #11楼 2013-07-26 16:17 吾非名家  
    @洒洒
    好像配对好了,不过乱码的问题怎么解决。。。
    #12楼[楼主] 2013-07-26 16:30 洒洒  
    @ 吾非名家
    搜索:Dos编码,你就知道了。
    #13楼 2013-07-26 16:50 吾非名家  
    @洒洒
    乱码问题已经搞掂了,谢谢。你的文章和帮助。
    #14楼[楼主] 2013-07-26 16:56 洒洒  
    @ 吾非名家
    不客气。
    #15楼 2013-12-02 09:50 寻找眼睛  
    非常感谢,demo运行很成功
    #16楼 2014-01-24 14:02 韩天峰  
    可以试试PHP的swoole扩展。
    http://www.swoole.com/
    #17楼 2014-08-15 21:50 walkor  
    用php写socket服务器,强烈推荐看下workerman
    #18楼 2014-10-14 15:41 xiaozhan  
    为什么我运行的时候出错了?

    Parse error: syntax error, unexpected T_LNUMBER in D:AppServwww est.php on line 3
    #19楼 2014-12-10 18:16 BiuBiuBiu666  
    服务器积极拒绝。
    #20楼 2014-12-10 18:17 BiuBiuBiu666  
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    C:Documents and SettingsAdministrator>php  E:/www/test/socket.php
    <h2>TCP/IP Connection</h2>
    OK.
    试图连接 '127.0.0.1' 端口 '1935'...
    PHP Warning:  socket_connect(): unable to connect [0]: 由于目标机器积极拒绝,无法连接。
     in E:www estsocket.php on line 29
    连接OK
    PHP Warning:  socket_write(): unable to write to socket [0]: 由于套接字没有连接并且(当使用一个 sendto 调用发送数据
    接收数据的请求没有被接受。
     in E:www estsocket.php on line 40
    PHP Warning:  socket_strerror() expects parameter 1 to be long, resource given in E:www estsocket.php on line 4
    socket_write() failed: reason:
    PHP Warning:  socket_read(): unable to read from socket [0]: 由于套接字没有连接并且(当使用一个 sendto 调用发送数据
    接收数据的请求没有被接受。
     in E:www estsocket.php on line 47
    关闭SOCKET...
    关闭OK
    #21楼[楼主] 2014-12-13 16:14 洒洒  
    @ 小小de细菌
    关闭防火墙试试
    #22楼 2015-01-14 15:53 todaytoday  
    我按照你这样写的,然后在服务器中输入php文件的地址按回车后,就出现了这个是怎么了
    ---------------------------
    php.exe - 系统错误
    ---------------------------
    无法启动此程序,因为计算机中丢失 php_mbstring.dll。尝试重新安装该程序以解决此问题。
    ---------------------------
    这个该怎么办啊?我用的服务器是appserv的
    #23楼 2015-01-14 15:58 todaytoday  
    但是我找了一遍,能在指定的文件夹中找到该文件啊?这是怎么了?
    #24楼 2015-03-04 11:18 swon  
    我也遇到了这种情况,怎么解决?
    #25楼 2016-08-17 14:47 傻乎乎的大兵  
    如果我想在socket连接上时就开始给前端返回数据,不经过下面的message,怎么办?
    #26楼 2017-02-14 09:59 叨叨的蜗牛  
    受教了,谢谢作者
    #27楼 2017-03-02 16:31 牛顿的小脑  
    PHP socket 那是用来做什么的呢?好像很少用到啊?具体例子,实际场景是什么呢?
  • 相关阅读:
    5-python基础—获取某个目录下的文件列表(适用于任何系统)
    Automated, Self-Service Provisioning of VMs Using HyperForm (Part 1) (使用HyperForm自动配置虚拟机(第1部分)
    CloudStack Support in Apache libcloud(Apache libcloud中对CloudStack支持)
    Deploying MicroProfile-Based Java Apps to Bluemix(将基于MicroProfile的Java应用程序部署到Bluemix)
    Adding Persistent Storage to Red Hat CDK Kit 3.0 (在Red Hat CDK Kit 3.0添加永久性存储)
    Carve Your Laptop Into VMs Using Vagrant(使用Vagran把您笔记本电脑刻录成虚拟机)
    使用Python生成一张用于登陆验证的字符图片
    Jupyter notebook的安装方法
    Ubuntu16.04使用Anaconda5搭建TensorFlow使用环境 图文详细教程
    不同时区的换算
  • 原文地址:https://www.cnblogs.com/aipiaoborensheng/p/6708963.html
Copyright © 2011-2022 走看看