memche メッセージ キューの原理は、キーについて大騒ぎすることです。キーは、シリアル化後にメッセージまたはログを記録するために連続した番号とプレフィックスを作成するために使用されます。その後、スケジュールされたプログラムを通じてコンテンツがファイルまたはデータベースにドロップされます。
/**
* 共有メモリを使用した 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;
}
}
http://www.bkjia.com/PHPjc/774998.htmlwww.bkjia.com本当http://www.bkjia.com/PHPjc/774998.html技術記事 memche メッセージ リストの原理は、キー上で文章を作成し、連続した数字を前に並べてメッセージやメッセージに書き込むことです。