Reids是一個比較高級的開源key-value存儲系統,采用ANSI C實現。其與memcached類似,但是支持持久化數據存儲入隊操作
創新互聯是少有的網站制作、成都網站制作、營銷型企業網站、小程序開發、手機APP,開發、制作、設計、賣鏈接、推廣優化一站式服務網絡公司,公司2013年成立,堅持透明化,價格低,無套路經營理念。讓網頁驚喜每一位訪客多年來深受用戶好評
復制代碼 代碼如下:
?php
$redis = new Redis();
$redis-connect('127.0.0.1',6379);
while(True){
try{
$value = 'value_'.date('Y-m-d H:i:s');
$redis-LPUSH('key1',$value);
sleep(rand()%3);
echo $value."\n";
}catch(Exception $e){
echo $e-getMessage()."\n";
}
}
?
出隊操作
復制代碼 代碼如下:
?php
$redis = new Redis();
$redis-pconnect('127.0.0.1',6379);
while(True){
try{
echo $redis-LPOP('key1')."\n";
}catch(Exception $e){
echo $e-getMessage()."\n";
}
sleep(rand()%3);
}?
如何使用Redis 做隊列操作
Reids是一個比較高級的開源key-value存儲系統,采用ANSI C實現。其與memcached類似,但是支持持久化數據存儲,同時value支持多種類型:字符串 (同memcached中的value),列表 ,集合 (Set),有序集合 (OrderSet)和Hash 。所有的值類型均支持原子操作,如列表中追加彈出元素,集合中插入移除元素等。Rdids的數據大部分位于內存中,其讀寫效率非常高,其提供AOF(追加 式操作記錄文件)和DUMP(定期數據備份)兩種持久化方式。Redis支持自定義的VM(虛擬內存)機制,當數據容量超過內存時,可以將部分Value 存儲到文件中。同時Redis支持Master-Slave機制,可以進行數據復制。
可以把Redis的list結構當隊列來用.
從上面Redis的場景和作用來說,對于我們現在的開發活動,究竟能把Redis引入在那些場景,而不是把這么好的東東演變成“為了使用Redis,而Redis”的慘烈局面呢?當然,具體問題具體分析,這個真的很重要哈。
緩存?分布式緩存?
隊列?分布式隊列?
某些系統應用(例如,電信、銀行和大型互聯網應用等)都會使用到,當然,現在大行其道的memcache就是很好的證明;但從某一方面來說,memcache是否能把兩張囊括其中,而且能做到更好(沒有實際的應用過,所以只是拋出)。但從Redis身上,我就能感覺到,Redis,就能把隊列和緩存兩張都囊括其中,而且都不會產生并發環境下的困擾,因為Redis中的操作都是原子操作來著。
至于評論兩者的孰好孰壞就免了,存在就是理由,選擇適合的就是最好的。
下面開始玩玩Redis中的隊列(分布式)設計YY吧,請大蝦們多多指點。
狀況場景:
現在的項目,都是部署在多個服務器,或者多個IP上,而且前臺經由F5分發,所以用戶的請求究竟落在那一臺的服務器上,是無法確定的。對于項目中,有一秒殺設計,剛開始沒有考慮到這種部署,同時也是使用最容易處理的方式,直接給數據庫表鎖行記錄(Oracle上的)。可以說,對于不同的應用部署,而只有一臺數據庫服務器來說,很“輕松”的就解決了這個并發的問題。所以現在考慮一下,是不是挪到應用上,避免數據庫服務器也摻雜到業務上。
比如,現在有2臺應用服務器,1臺數據庫服務器。想法是,把Redis部署在數據庫服務器上,兩臺服務器在操作并發緩存或者隊列時,先從Redis服務器上,取得在兩臺應用服務器的代理對象,再做入列出列的操作。
看代碼實現(PHP)
入隊列操作文件 list_push.php
復制代碼 代碼如下:
?php
$redis = getRedisInstance();//從Redis服務器拿到redis實例$redis-connect('Redis服務器IP', 6379);
while (true) {
$redis-lPush('list1', 'A_'.date('Y-m-d H:i:s'));sleep(rand()%3);
}
?
執行# php list_push.php
出隊列操作 list_pop.php文件
復制代碼 代碼如下:
?php
$redis = getRedisInstance();//從Redis服務器拿到redis實例$redis-pconnect('Redis服務器IP', 6379);
while(true) {
try {
var_export( $redis-blPop('list1', 10) );} catch(Exception $e) {
//echo $e;
}
}
實現方法(Python)
1.入隊列(write.py)
復制代碼 代碼如下:
#!/usr/bin/env python
import time
from redis import Redis
redis = Redis(host='127.0.0.1', port=6379)while True:
now = time.strftime("%Y/%m/%d %H:%M:%S")
redis.lpush('test_queue', now)
time.sleep(1)
2.出隊列(read.py)
復制代碼 代碼如下:
#!/usr/bin/env python
import sys
from redis import Redis
redis = Redis(host='127.0.0.1', port=6379)while True:
res = redis.rpop('test_queue')
if res == None:
pass
else:
print str(res)
最近遇到一個批量發送短信的需求,短信接口是第三方提供的。剛開始想到,獲取到手機號之后,循環調用接口發送不就可以了嗎?
但很快發現問題:當短信數量很大時,不僅耗時,而且成功率很低。
于是想到,用PHP和MySQL實現一個消息隊列,一條一條的發送短信。下面介紹具體的實現方法:
首先,建立一個數據表sms,包含以下字段:
id,
phone, //手機號
content //短信內容
將需要發送的短信和手機號存入sms表中。
接下來,需要用PHP實現一個定時器,定時讀取一條記錄,并發送短信:
?php
$db = new Db();
$sms = new Sms();
while(true){
$item = $db-getFirstRecord(); //獲取數據表第一條記錄
if(!$item){
//如果隊列中沒有數據,則結束定時器
break;
}
$res = $sms-send($item['phone'],$item['content']); //發送短信
if($res){
$db-deleteFristRecord(); //刪除發送成功的記錄
echo $item['phone'].'發送成功';
}else{
echo $item['phone'].'發送失敗,稍后繼續嘗試';
}
sleep(10); //每隔十秒循環一次
}
echo '發送完畢!';
?
將代碼保存為timer_sms.php,打開命令行,執行定時器:
php timer_sms.php
好了,php定時器將會根據設定的時間間隔(這里設的是10秒),自動完成發送短信的任務。任務完成后將自動退出定時器,不再占用服務器資源。
根據我的測試,PHP定時器占用資源并不多,不會對服務器造成壓力。而且是異步訪問數據庫,也不會影響數據庫的運行。
這種方式的優點是:
1、后臺運行,前臺無需等待
2、成功率高,失敗的記錄會自動重發,直到成功
在PHP中,我們可以用array來簡單模擬隊列的功能,array_push出列,array_pop入列,但是更高效的用法還是要使用SplQueue這個官方提供的隊列類來做,記錄一下簡單的用法。
我們注意到,SplQueue中還有push和pop方法,它們與enqueue和dequeue有什么區別呢?
push和enqueue沒有區別,但是pop等同于棧的pop(FILO),每次彈出最后壓入隊列的元素。
而隊列是先進先出(FIFO)的,所以每次彈出的都是剩下的節點中先入列節點
以上是PHP隊列的簡單用法,下面是一些迭代器指針相關的操作
隊列這種數據結構更簡單,就像我們生活中排隊一樣,它的特性是先進先出(FIFO)。
PHP
SPL中SplQueue類就是實現隊列操作,和棧一樣,它也可以繼承雙鏈表(SplDoublyLinkedList)輕松實現。
SplQueue類摘要如下:
SplQueue簡單使用如下:
復制代碼
代碼如下:
$queue
=
new
SplQueue();
/**
*
可見隊列和雙鏈表的區別就是IteratorMode改變了而已,棧的IteratorMode只能為:
*
(1)SplDoublyLinkedList::IT_MODE_FIFO
|
SplDoublyLinkedList::IT_MODE_KEEP
(默認值,迭代后數據保存)
*
(2)SplDoublyLinkedList::IT_MODE_FIFO
|
SplDoublyLinkedList::IT_MODE_DELETE
(迭代后數據刪除)
*/
$queue-setIteratorMode(SplDoublyLinkedList::IT_MODE_FIFO
|
SplDoublyLinkedList::IT_MODE_DELETE);
//SplQueue::enqueue()其實就是
SplDoublyLinkedList::push()
$queue-enqueue('a');
$queue-enqueue('b');
$queue-enqueue('c');
//SplQueue::dequeue()其實就是
SplDoublyLinkedList::shift()
print_r($queue-dequeue());
foreach($queue
as
$item)
{
echo
$item
.
PHP_EOL;
}
print_r($queue);
而優先隊列SplPriorityQueue是基于堆(后文介紹)實現的。
SplPriorityQueue的類摘要如下:
SplPriorityQueue簡單使用:
$pq
=
new
SplPriorityQueue();
$pq-insert('a',
10);
$pq-insert('b',
1);
$pq-insert('c',
8);
echo
$pq-count()
.PHP_EOL;
//3
echo
$pq-current()
.
PHP_EOL;
//a
/**
*
設置元素出隊模式
*
SplPriorityQueue::EXTR_DATA
僅提取值
*
SplPriorityQueue::EXTR_PRIORITY
僅提取優先級
*
SplPriorityQueue::EXTR_BOTH
提取數組包含值和優先級
*/
$pq-setExtractFlags(SplPriorityQueue::EXTR_DATA);
while($pq-valid())
{
print_r($pq-current());
//a
c
b
$pq-next();
}
新聞名稱:php實現數據庫隊列 php中數據庫怎么設計
轉載注明:http://vcdvsql.cn/article44/ddccpee.html
成都網站建設公司_創新互聯,為您提供外貿建站、服務器托管、網站策劃、網頁設計公司、移動網站建設、網站排名
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯