首頁  >  文章  >  後端開發  >  php事件驅動化設計

php事件驅動化設計

不言
不言原創
2018-04-09 17:39:351745瀏覽

這篇文章主要介紹了php事件驅動化設計,現在分享給大家,有需要的朋友可以參考下

本文實例講述了php事件驅動化設計。分享給大家供大家參考,具體如下:

最近在做一個需要用到異步php的項目, 翻閱php源碼的時候,發現了三個沒有用過的模組,sysvsem,sysvshm,sysvmsg,一番研究以後,受益非淺。

在php中有這麼一族函數,他們是對unix的v ipc函數族的包裝。
它們很少被人們用到,但是它們卻很強大。巧妙的運用它們,可以讓你事倍功半。

它們包括:

信號量(semaphores)
共享記憶體(shared memory)
進程間通訊(inter-process messaging, ipc)

基於這些,我們完全有可能將php包裝成一基於訊息驅動的系統。

但是,首先,我們需要介紹幾個重要的基礎:

1. ftok

int ftok ( string pathname, string proj )

ftok將一個路徑名pathname和一個項目名(必須為一個字元), 轉換成一個整形的用來使用系統v ipc的key

2. ticks

ticks是從php 4.0.3開始才加入到php中的,它是一個在declare 程式碼段中解釋器每執行n 條低階語句就會發生的事件。 n 的值是在 declare 中的 directive 部分用 ticks=n 來指定的。

function getstatus($arg){
  print_r(connection_status());
  debug_print_backtrace();
}
reigster_tick_function("getstatus", true);
declare(ticks=1){
  for($i =1; $i<999; $i++){
 echo "hello";
 }
}
unregister_tick_function("getstatus");

這個基本上就相當於:

#
function getstatus($arg){
  print_r(connection_status());
  debug_print_backtrace();
}
reigster_tick_function("getstatus", true);
declare(ticks=1){
  for($i =1; $i<999; $i++){
 echo "hello"; getstatus(true);
 }
}
unregister_tick_function("getstatus");

訊息,我現在用一個例子來說明,如何結合ticks來實現php的訊息通訊。

$mesg_key = ftok(__file__, &#39;m&#39;);
$mesg_id = msg_get_queue($mesg_key, 0666);
function fetchmessage($mesg_id){
 if(!is_resource($mesg_id)){
  print_r("mesg queue is not ready");
 }
 if(msg_receive($mesg_id, 0, $mesg_type, 1024, $mesg, false, msg_ipc_nowait)){
  print_r("process got a new incoming msg: $mesg ");
 }
}
register_tick_function("fetchmessage", $mesg_id);
declare(ticks=2){
 $i = 0;
 while(++$i < 100){
  if($i%5 == 0){
msg_send($mesg_id, 1, "hi: now index is :". $i);
  }
 }
}
//msg_remove_queue($mesg_id);

在這個範例中,首先將我們的php執行process加入到一個由ftok產生的key所獲得的訊息佇列中。

然後,透過ticks,沒隔兩個語句,就去查詢一次訊息佇列。

然後模擬了訊息發送。

在瀏覽器存取這個腳本,結果如下:

process got a new incoming msg: s:19:"hi: now index is :5";
process got a new incoming msg: s:20:"hi: now index is :10";
process got a new incoming msg: s:20:"hi: now index is :15";
process got a new incoming msg: s:20:"hi: now index is :20";
process got a new incoming msg: s:20:"hi: now index is :25";
process got a new incoming msg: s:20:"hi: now index is :30";
process got a new incoming msg: s:20:"hi: now index is :35";
process got a new incoming msg: s:20:"hi: now index is :40";
process got a new incoming msg: s:20:"hi: now index is :45";
process got a new incoming msg: s:20:"hi: now index is :50";
process got a new incoming msg: s:20:"hi: now index is :55";
process got a new incoming msg: s:20:"hi: now index is :60";
process got a new incoming msg: s:20:"hi: now index is :65";
process got a new incoming msg: s:20:"hi: now index is :70";
process got a new incoming msg: s:20:"hi: now index is :75";
process got a new incoming msg: s:20:"hi: now index is :80";
process got a new incoming msg: s:20:"hi: now index is :85";
process got a new incoming msg: s:20:"hi: now index is :90";
process got a new incoming msg: s:20:"hi: now index is :95";

看到這裡是不是,大家已經對怎麼模擬php為事件驅動已經有了一個概念了? 別急,我們繼續完善。

2. 信號量

信號量的概念,大家應該都很熟悉。透過信號量,可以實現進程通信,競爭等。 再次就不贅述了,只是簡單的列出php中提供的信號量函數集

sem_acquire -- acquire a semaphore
sem_get -- get a semaphore id
sem_release -- release a semaphore
sem_remove -- remove a semaphore

具體信息,可以翻閱php手冊。

3. 記憶體共享

php sysvshm提供了一個記憶體共享方案:sysvshm,它是和sysvsem,sysvmsg一個系列的,但在此處,我並沒有使用它,我使用的shmop系列函數,結合ticks

function memoryusage(){
 printf("%s: %s<br/>", date("h:i:s",time()), memory_get_usage());
 //var_dump(debug_backtrace());
 //var_dump(__function__);
 //debug_print_backtrace();
}
register_tick_function("memoryusage");
declare(ticks=1){
$shm_key = ftok(__file__, &#39;s&#39;);
$shm_id = shmop_open($shm_key, &#39;c&#39;, 0644, 100);
}
printf("size of shared memory is: %s<br/>", shmop_size($shm_id));
$shm_text = shmop_read($shm_id, 0, 100);
eval($shm_text);
if(!empty($share_array)){
 var_dump($share_array);
 $share_array[&#39;id&#39;] += 1;
}else{
 $share_array = array(&#39;id&#39; => 1);
}
$out_put_str = "$share_array = " . var_export($share_array, true) .";";
$out_put_str = str_pad($out_put_str, 100, " ", str_pad_right);
shmop_write($shm_id, $out_put_str, 0);
?>

運行這個例子,不斷刷新,我們可以看到index在遞增。

單單使用這個shmop就能完成一下,php腳本之間共享資料的功能:以及,例如緩存,計數等等。

相關推薦:

PHP事件機制的實作

#詳細解析:關於PHP事件驅動問題_PHP教學

#

以上是php事件驅動化設計的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn