ホームページ > バックエンド開発 > PHPチュートリアル > PHP+memcache によるメッセージ キューのケース共有_PHP チュートリアルの実装

PHP+memcache によるメッセージ キューのケース共有_PHP チュートリアルの実装

WBOY
リリース: 2016-07-13 10:29:30
オリジナル
733 人が閲覧しました

memche メッセージ キューの原理は、キーについて大騒ぎすることです。キーは、シリアル化後にメッセージまたはログを記録するために連続した番号とプレフィックスを作成するために使用されます。その後、スケジュールされたプログラムを通じてコン​​テンツがファイルまたはデータベースにドロップされます。

PHP を使用してメッセージキューを実装する たとえば、メールを送信するときに大量のメールを送信するのに時間がかかる場合は、キューを使用できます。
キューの実装を容易にする軽量のキューサーバーは次のとおりです:
starling は memcache プロトコルをサポートする軽量の永続サーバーです
https://github.com/starling/starling
Beanstalkd は軽量で効率的で、永続性をサポートしており、毎回処理できます2 番目 約 3,000 のキューを処理する
http://kr.github.com/beanstalkd/
Memcache/memcached を使用して、php でメッセージ キューを実装することもできます。

コードをコピーします コードは次のとおりです:

/**
* Memcache メッセージキュークラス
*/
class QMC {
const PREFIX = 'ASDFASDFFWQKE'; 
/**
* mc を初期化します
* @staticvar string $mc
* @return Memcache
*/
static private function mc_init() {
static $mc = null; 
if (is_null($mc)) {
$mc = 新しい Memcache; 
$mc->connect('127.0.0.1', 11211); 
}
$mc を返します。 
}
/**
* mc counter、カウントをインクリメントし、新しいカウントを返します
* @param string $key Counter
* @param int $offset カウントの増分。0 はカウントが変化しないことを意味します
* @param int $ time 時間
* @return int/false 失敗した場合は false を返し、成功した場合はカウンターを更新した後にカウントを返します
*/
static public function set_counter( $key, $offset, $time=0 ){
$mc = self::mc_init(); 
$val = $mc->get($key); 
if( !is_numeric($val) || $val < 0 ){
$ret = $mc->set( $key, 0, $time ); 
if( !$ret ) は false を返します。 
$val = 0; 
}
$offset = intval( $offset ); 
if( $offset > 0 ){
return $mc->increment( $key, $offset ); 
}elseif( $offset return $mc->decrement( $key, -$offset ); 
}
return $val; 
}
/**
* キューに書き込みます
* @param string $key
* @parammixed $value
* @return bool
*/
static public function input( $key, $value ){
$mc = self::mc_init(); 
$w_key = self::PREFIX.$key.'W'; 
$v_key = self::PREFIX.$key.self::set_counter($w_key, 1); 
return $mc->set( $v_key, $value ); 
}
/**
* キュー内のデータを読み取ります
* @param string $key
* @param int $max 読み取る項目の最大数
* @return array
*/
static public function Output( $key, $max=100 ){
$out = array(); 
$mc = self::mc_init(); 
$r_key = self::PREFIX.$key.'R'; 
$w_key = self::PREFIX.$key.'W'; 
$r_p = self::set_counter( $r_key, 0 );//写指针
$w_p = self::set_counter( $w_key, 0 );//写指针
if( $r_p == 0 ) $r_p = 1; 
while( $w_p >= $r_p ){
if( --$max $v_key = self::PREFIX.$key.$r_p; 
$r_p = self::set_counter( $r_key, 1 ); 
$out[] = $mc->get( $v_key ); 
$mc->delete($v_key); 
}
return $out; 
}
}
/**
使用法:
QMC::input($key, $value);//書き込みキュー
$list = QMC::output($key);//読み取りキュー
*/
?> 

PHP 共有共有内に保存されているメッセージの列:

复制代価代価如下:

/**
* 共有メモリを使用した PHP 循環メモリ キューの実装
* 複数のプロセスをサポートし、さまざまなデータ型のストレージをサポート
* 注: エンキューまたはデキュー操作が完了したら、できるだけ早く unset() を使用してクリティカル セクションを解放します
*
* @author wangbinandi@gmail.com
* @作成 2009-12-23
*/
class ShmQueue
{
private $maxQSize = 0; // キューの最大長
private $front = 0 // キューヘッドポインタ
private $rear = 0; ; // キュー末尾ポインタ
private $blockSize = 256; // ブロック サイズ (バイト)
private $memSize = 25600; // 最大共有メモリ (バイト)
private $filePtr = ' / shmq.ptr';
プライベート $semId = 0;
パブリック関数 __construct()
$shmkey = ftok(__FILE__, 't');
$this->shmop_open($shmkey, "c" , 0644, $this->memSize );
$this->memSize / $this->blockSize;
// セマフォ
$this->semId = sem_get ($ shmkey, 1);
sem_acquire($this->semId); // クリティカルセクションに入るために適用します
$this->init();
private function init()
if ( file_exists($) this->filePtr) ){
$contents = file_get_contents($this->filePtr);
$data =explode( '|', $contents );
if ( isset($data[0]) && isset( $data[1])){
$this->フロント = (int)$data[0];
$this->リア = (int)$data[1]; function getLength()
{
return (($this->rear - $this->front + $this->memSize) % ($this->memSize) )/$this->blockSize; }
public function enQueue( $value )
{
if ( $this->ptrInc($this->rear) == $this->front ){ // キューがいっぱいです
return false
}
$ data = $this->encode($value);
shmop_write($this->shmId, $data, $this->rear );
$this->rear = $this->ptrInc( $ this->rear);
return true;
public function deQueue()
{
if ( $this->front == $this->rear ){ // キューが空
return false; }
$value = shmop_read($this->shmId, $this->front, $this->blockSize-1);
$this->front = $this->ptrInc($this-> ; フロント);
return $this->decode($value);
プライベート関数 ptrInc( $ptr )
{
return ($ptr + $this->memSize) ) ;
}
プライベート関数 encode($value)
$data =
echo ''; - >blockSize -1;
echo '';
if ( strlen($data) > $this->blockSize -1 ){
throw new Exception(strlen($data)." はオーバーロード ブロック サイズです!" ) ;
}
return $data;
private 関数 decode($value )
{
$data =explode("__eof", $value)
}
public関数 __destruct()
$data = $this->front . '|'
file_put_contents($this->filePtr, $data); semId ); // クリティカルセクションを終了し、セマフォを解放します
}
}
/*
// キュー操作に入ります
$shmq = new ShmQueue()
$data = 'test data'; ;enQueue( $data);
unset($shmq);
$shmq = new ShmQueue();
unset($shmq); /
?>



メッセージ キューが大きい場合、大規模なデータベースのシリアル化と逆シリアル化を頻繁に行うのは時間がかかりすぎます。以下は、PHP を使用して実装したメッセージ キューです。末尾にデータを挿入し、末尾を操作するだけで済みます。読み取りや操作のためにメッセージ キュー全体を操作する必要はありません。ただし、このメッセージ キューはスレッドセーフではありません。競合の可能性を回避しようとしているだけです。メッセージの密度がそれほど高くない場合 (たとえば、数秒に 1 つのメッセージだけである場合)、この方法での使用を検討できます。
スレッドの安全性を実現したい場合、ファイルをロックしてから操作することをお勧めします。以下はコードです:
コードは次のとおりです:




コードをコピーします

コードは次のとおりです:

クラス Memcache_Queue
{
プライベート $memcache;  
プライベート $name;  
プライベート $プレフィックス;   construct($ maxsize、$ name、$ memcache、$ memcache、prefix = "__memcache_queue__")
{
if($ memcache == null){
スロー新しい例外( "memcacheオブジェクトはnull、null、new the object first")) ;  
}
$this->memcache = $memcache;  
$this->name = $name;  
$this->prefix = $prefix;  
$this->maxSize = $maxSize;  
$this->front = 0;  
$this->real = 0;  
$this->size = 0;  
}
function __get($name)
{
return $this->get($name);  
}
関数 __set($name, $value)
{
$this->add($name, $value);  
$this を返します。  
}
関数 isEmpty()
{
return $this->size == 0;  
}
関数 isFull()
{
return $this->size == $this->maxSize;  
}
function enQueue($data)
{
if ($this->isFull()) {
throw new Exception("Queue is Full");  
}
$this->increment("size");  
$this->set($this->real, $data);  
$this->set("real", ($this->real + 1) % $this->maxSize);  
$this を返します。  
}
function deQueue()
{
if ($this->isEmpty()) {
throw new Exception("Queue is Empty");  
}
$this->decrement("size");  
$this->delete($this->front);  
$this->set("front", ($this->front + 1) % $this->maxSize);  
$this を返します。  
}
function getTop()
{
return $this->get($this->front);  
}
function getAll()
{
return $this->getPage();  
}
function getPage($offset = 0, $limit = 0)
{
if ($this->isEmpty() || $this->size < $offset) {
return null;  
}
$keys[] = $this->getKeyByPos(($this->front + $offset) % $this->maxSize);  
$num = 1;  
for ($pos = ($this->front + $offset + 1) % $this->maxSize; $pos != $this->real; $pos = ($pos + 1) % $this ->maxSize)
{
$keys[] = $this->getKeyByPos($pos);  
$num++;  
if ($limit > 0 && $limit == $num) {
Break;  
}
}
return array_values($this->memcache->get($keys));  
}  
function makeEmpty()
{
$keys = $this->getAllKeys();  
foreach ($keys as $value) {
$this->delete($value);  
}
$this->delete("real");  
$this->delete("front");  
$this->delete("size");  
$this->delete("maxSize");  
}
プライベート関数 getAllKeys()
{
if ($this->isEmpty())
{
return array();  
}
$keys[] = $this->getKeyByPos($this->front);  
for ($pos = ($this->front + 1) % $this->maxSize; $pos != $this->real; $pos = ($pos + 1) % $this->; maxSize)
{
$keys[] = $this->getKeyByPos($pos);  
}
$keys を返します。  
}
プライベート関数 add($pos, $data)
{
$this->memcache->add($this->getKeyByPos($pos), $data);  
$this を返します。  
}
private function increment($pos)
{
return $this->memcache->increment($this->getKeyByPos($pos));  
}
private function decrement($pos)
{
$this->memcache->decrement($this->getKeyByPos($pos));  
}
プライベート関数 set($pos, $data)
{
$this->memcache->set($this->getKeyByPos($pos), $data);  
$this を返します。  
}
プライベート関数 get($pos)
{
return $this->memcache->get($this->getKeyByPos($pos));  
}
private function delete($pos)
{
return $this->memcache->delete($this->getKeyByPos($pos));  
}
プライベート関数 getKeyByPos($pos)
{
return $this->prefix 。 $this->name 。 $pos;  
}
}

www.bkjia.com本当http://www.bkjia.com/PHPjc/774998.html技術記事 memche メッセージ リストの原理は、キー上で文章を作成し、連続した数字を前に並べてメッセージやメッセージに書き込むことです。
ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
最新の問題
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート