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!";
?>