旷世的忧伤

Huoty's Blog

PHP向客户端广播信息

在网络中数据传播分为:Unicast(单播) , Multicast(多播或者组播) 和 Broadcast(广播)。广播和多播仅应用于UDP,它们对需将报文同时传往多个接收者的应用来说十分重要。而 TCP 是一个面向连接的协议,它意味着分别运行于两主机(由IP地址确定)内的两进程(由端口号确定)间存在一条连接。广播地址在默认情况下是不能让路由器转发到别的接口的,广播不能穿越路由器。广播有以下几种形式:

  • 受限的广播:

受限的广播地址是255.255.255.255,该地址用于主机配置过程中IP数据报的地址,此时,主机可能还不知道它所在网络的网络掩码,甚至连它的IP地址也不知道。在任何情况下,路由器都不转发目的地址为受限广播地址的数据报,这样的数据报只出现在本地网络中。

  • 指向网络的广播:

指向网络的广播地址是主机号全为1的地址,A类网络广播地址为netid.255.255.255,其中netid为A类网络的网络号。

  • 指向子网的广播:

指向子网的广播地址是主机号全为1的地址,作为子网直接广播的IP地址需要知道子网的掩码。如果B类网络128.1的子网掩码是255.255.255.0,则地址128.1.2.255就是对应子网的广播地址。

  • 指向所有子网的广播:

指向所有子网的广播也需要知道目的网络的子网掩码。这些广播地址的子网号和主机号全为1。如果目的子网掩码是255.255.255.0,那么IP地址128.1.255.255就是一个指向所有子网的广播地址。

PHP socket 也能实现广播。在 socket 通信中,实现连接的服务器与客户端需要绑定同一端口号,端口号表示发送和接收的进程。下面是一个用 PHP 实现的简单的广播通信例子,同时采用 PHP 和 C 语言作为客户端进行测试:

  • broadcast.php
<?php  # Script -- broadcast.php

/* Author @ Huoty
 *   Date @ 2015-11-17 09:58:25
 *  Brief @
 */

/* 创建广播事件 */
function broadcast()
{
	$sock = socket_create(AF_INET, SOCK_DGRAM, SOL_UDP);
	//使用IPV4格式地址,数据报形式,UDP方式传输数据

	socket_set_option($sock, SOL_SOCKET, SO_BROADCAST, 1); //设置为广播方式

	while ( true ) {
	    $msg = 'Hi! ' . date("y-m-d h:i:s",time());  //要发送的字符串
		socket_sendto($sock, $msg, strlen($msg), 0, "255.255.255.255", 12345);
		//发送,255.255.255.255是广播地址,12345是端口
		//echo "Broadcast...\n";
		sleep( 2 );
	}

	socket_close($sock); //关闭
}

/* 创建守护进程 */
$pid = pcntl_fork();
if ($pid < 0)
{
    die("fork failed!\n");
}
else if ($pid > 0)
{
    exit;
}
else
{
		/* 输出进程ID,便于 kill */
    echo "Daemons ID: " . posix_getpid() . "\n";

		/* 保持程序的运行 */
		set_time_limit(0);

		/* 创建一个新的 Session */
    $sid = posix_setsid();
    if ($sid < 0)
    {
        exit;
    }

		/* 改变工作目录为根目录 */
    chdir("/");

    broadcast();
}

?>
  • client.php
<?php # Script -- client.php

/* Author @ Huoty
 *   Date @ 2015-11-17 09:58:25
 *  Brief @
 */

//error_reporting( E_ALL );
set_time_limit( 0 );
ob_implicit_flush();

$socket = socket_create( AF_INET, SOCK_DGRAM, SOL_UDP );
if ( $socket === false ) {
    echo "socket_create() failed:reason:" . socket_strerror( socket_last_error() ) . "\n";
}

$ok = socket_bind( $socket, '255.255.255.255', 12345 );
if ( $ok === false ) {
    echo "socket_bind() failed:reason:" . socket_strerror( socket_last_error( $socket ) );
}

while ( true ) {
    $from = "";
    $port = 0;
    socket_recvfrom( $socket, $buf, 1024, 0, $from, $port );
    echo $buf . "\n";
    usleep( 1000 );
}
?>
  • client.c
/* client.c */
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <netinet/in.h>

#define MAXLINE 80
#define SERV_PORT 12345

int main(int argc, char *argv[])
{
    struct sockaddr_in servaddr;
    int sockfd, n;
    char buf[MAXLINE];
    char str[INET_ADDRSTRLEN];
    socklen_t servaddr_len;

    sockfd = socket(AF_INET, SOCK_DGRAM, 0);
    bzero(&servaddr, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port = htons(SERV_PORT);

    bind(sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));
    fprintf(stdout, "Accepting connections ...\n");

    memset(buf, 0, sizeof(buf));
    while ( 1 ) {
        n = recvfrom(sockfd, buf, MAXLINE, 0, NULL, 0);
        if (n == -1)
            fprintf(stderr, "recvfrom error");

        fprintf(stdout, "%s\n", buf);
        memset(buf, 0, sizeof(buf));
    }

    close(sockfd);
    return 0;
}

通常,广播是需要长时间进行的任务,所以可以创建一个守护进程来完成广播,以避免程序长时间运行对控制终端的占用。如果不使用守护进程,也可以用 Linux 的 nohup 命令来实现。然而,PHP 的进程控制不能被应用在 Web 服务器环境。那么,要让 PHP 的进程控制在 Web 环境下得到应用,可以用一个迂回的办法,即用 cli 的方式执行包含进程控制的 PHP 文件,所谓 cli 方式是指 shell 的执行方式。还有一个需要注意的问题是,在 Web 环境下,由于 PHP 程序是一个死循环,程序一直运行,所以客户端总是得不到服务器的返回结果。为解决这个问题,可以将用 & 让程序在后台运行,同时将输出重定向到 /dev/null。于是可以创建了一个新文件以保证广播在 Web 服务器环境下能够被触发:

  • startup.php
<?php  # Script -- startup.php

/* Author @ Huoty
 *   Date @ 2015-12-02 16:53:43
 *  Brief @
 */

exec("php ./broadcast_daemons.php >/dev/null &");
echo "Finished!";

?>
Top