PHP Socket编程及聊天室实例

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

PHP 代码:
<?
// 设置两个变量
$host="192.168.1.99"
;
$port=1234
;
?>

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

 

PHP 代码:
<?
// 超时时间
set_time_limit(0
);
?>

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

 

PHP 代码:
<?
// 创建Socket
$socket=socket_create(AF_INET,SOCK_STREAM,0) or die(
"Could not create socket\n"
);
?>

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

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

 

PHP 代码:
<?
// 创建 socket
$socket=socket_create(AF_INET,SOCK_DGRAM,0) or die(
"Could not create socket\n"
);
?>

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

 

PHP 代码:
<?
// 绑定 socket to 指定地址和端口
$result=socket_bind($socket,$host,$port) or die(
"Could not bind to socket\n"
);
?>

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

 

PHP 代码:
<?
// 开始监听连接
$result=socket_listen($socket,3) or die(
"Could not set up socket listener\n"
);
?>

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

 

PHP 代码:
<?
//接受请求链接
// 调用子socket 处理信息
$spawn=socket_accept($socket) or die(
"Could not accept incoming connection\n");
?>

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

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

 

PHP 代码:
<?
// 读取客户端输入
$input=socket_read($spawn,1024) or die("Could not read input\n"
);
?>

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

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

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

 

PHP 代码:
<?
// 处理客户端输入并返回数据
$output=strrev($input) ."\n"
;
socket_write($spawn,$output,strlen($output)) or die(
"Could not write output\n"
);
?>

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

 

PHP 代码:

<?
// 关闭 sockets
socket_close($spawn
);
socket_close($socket
);
?>

[代码] php服务器端的

run();
class Sock{
public $sockets;
public $users;
public $master;

public function __construct($address, $port){
$this->master=$this->WebSocket($address, $port);
$this->sockets=array(‘s’=>$this->master);
}

function run(){
while(true){
$changes=$this->sockets;
socket_select($changes,$write=NULL,$except=NULL,NULL);
foreach($changes as $sock){
if($sock==$this->master){
$client=socket_accept($this->master);
//$key=uniqid();
$this->sockets[]=$client;
$this->users[]=array(
‘socket’=>$client,
‘shou’=>false
);
}else{
$len=socket_recv($sock,$buffer,2048,0);
$k=$this->search($sock);
if($len<7){ $name=$this->users[$k][‘ming’];
$this->close($sock);
$this->send2($name,$k);
continue;
}
if(!$this->users[$k][‘shou’]){
$this->woshou($k,$buffer);
}else{
$buffer = $this->uncode($buffer);
$this->send($k,$buffer);
}
}
}

}

}

function close($sock){
$k=array_search($sock, $this->sockets);
socket_close($sock);
unset($this->sockets[$k]);
unset($this->users[$k]);
$this->e(“key:$k close”);
}

function search($sock){
foreach ($this->users as $k=>$v){
if($sock==$v[‘socket’])
return $k;
}
return false;
}

function WebSocket($address,$port){
$server = socket_create(AF_INET, SOCK_STREAM, SOL_TCP);
socket_set_option($server, SOL_SOCKET, SO_REUSEADDR, 1);
socket_bind($server, $address, $port);
socket_listen($server);
$this->e(‘Server Started : ‘.date(‘Y-m-d H:i:s’));
$this->e(‘Listening on : ‘.$address.’ port ‘.$port);
return $server;
}

function woshou($k,$buffer){
$buf = substr($buffer,strpos($buffer,’Sec-WebSocket-Key:’)+18);
$key = trim(substr($buf,0,strpos($buf,”\r\n”)));

$new_key = base64_encode(sha1($key.”258EAFA5-E914-47DA-95CA-C5AB0DC85B11″,true));

$new_message = “HTTP/1.1 101 Switching Protocols\r\n”;
$new_message .= “Upgrade: websocket\r\n”;
$new_message .= “Sec-WebSocket-Version: 13\r\n”;
$new_message .= “Connection: Upgrade\r\n”;
$new_message .= “Sec-WebSocket-Accept: ” . $new_key . “\r\n\r\n”;

socket_write($this->users[$k][‘socket’],$new_message,strlen($new_message));
$this->users[$k][‘shou’]=true;
return true;

}

function uncode($str){
$mask = array();
$data = ”;
$msg = unpack(‘H*’,$str);
$head = substr($msg[1],0,2);
if (hexdec($head{1}) === 8) {
$data = false;
}else if (hexdec($head{1}) === 1){
$mask[] = hexdec(substr($msg[1],4,2));
$mask[] = hexdec(substr($msg[1],6,2));
$mask[] = hexdec(substr($msg[1],8,2));
$mask[] = hexdec(substr($msg[1],10,2));

$s = 12;
$e = strlen($msg[1])-2;
$n = 0;
for ($i=$s; $i<= $e; $i+= 2) {
$data .= chr($mask[$n%4]^hexdec(substr($msg[1],$i,2)));
$n++;
}
}
return $data;
}

function code($msg){
$msg = preg_replace(array(‘/\r$/’,’/\n$/’,’/\r\n$/’,), ”, $msg);
$frame = array();
$frame[0] = ’81’;
$len = strlen($msg);
$frame[1] = $len<16?’0′.dechex($len):dechex($len); $frame[2] = $this->ord_hex($msg);
$data = implode(”,$frame);
return pack(“H*”, $data);
}

function ord_hex($data) {
$msg = ”;
$l = strlen($data);
for ($i= 0; $i<$l; $i++) { $msg .= dechex(ord($data{$i})); } return $msg; } function send($k,$msg){ /*$this->send1($k,$this->code($msg),’all’);*/
parse_str($msg,$g);
$this->e($msg);
$ar=array();
if($g[‘type’]==’add’){
$this->users[$k][‘ming’]=$g[‘ming’];
$ar[‘add’]=true;
$ar[‘nrong’]=’欢迎’.$g[‘ming’].’加入!’;
$ar[‘users’]=$this->getusers();
$key=’all’;
}else if($g[‘type’]==’ltiao’){
$ar[‘nrong’]=$g[‘nr’];
$key=$g[‘key’];
}
$msg=json_encode($ar);
$this->e($msg);
$msg = $this->code($msg);
$this->send1($k,$msg,$key);
//socket_write($this->users[$k][‘socket’],$msg,strlen($msg));
}

function getusers(){
$ar=array();
foreach($this->users as $k=>$v){
$ar[$k]=$v[‘ming’];
}
return $ar;
}

function send1($k,$str,$key=’all’){
if($key==’all’){
foreach($this->users as $v){
socket_write($v[‘socket’],$str,strlen($str));
}
}else{
if($k!=$key)
socket_write($this->users[$k][‘socket’],$str,strlen($str));
socket_write($this->users[$key][‘socket’],$str,strlen($str));
}
}

function send2($ming,$k){
$ar[‘remove’]=true;
$ar[‘removekey’]=$k;
$ar[‘nrong’]=$ming.’退出聊天室’;
$str = $this->code(json_encode($ar));
$this->send1(false,$str,’all’);
}

function e($str){
$path=dirname(__FILE__).’/log.txt’;
$str=$str.”\n”;
error_log($str,3,$path);
echo iconv(‘utf-8′,’gbk//IGNORE’,$str);
}
}
?>