前麵一篇《NoSQL架構實踐—以NoSQL為輔》主要介紹了以NoSQL為輔助的架構,這種架構實施起來比較簡單,易於理解,由於其中也使用了傳統的關係數據庫,讓開發者更容易控製NoSQL帶來的風險。接下來我們繼續深入下去,換另外一個角度,“以NoSQL為主”來架構係統。
隻使用NoSQL作為數據存儲。
圖 4-純NoSQL架構
在一些數據結構、查詢關係非常簡單的係統中,我們可以隻使用NoSQL即可以解決存儲問題。這樣不但可以提高性能,還非常易於擴展。手機鳳凰網的前端展示係統就使用了這種方案。
在一些數據庫結構經常變化,數據結構不定的係統中,就非常適合使用NoSQL來存儲。比如監控係統中的監控信息的存儲,可能每種類型的監控信息都不太一樣。這樣可以避免經常對MySQL進行表結構調整,增加字段帶來的性能問題。
這種架構的缺點就是數據直接存儲在NoSQL中,不能做關係數據庫的複雜查詢,如果由於需求變更,需要進行某些查詢,可能無法滿足,所以采用這種架構的時候需要確認未來是否會進行複雜關係查詢以及如何應對。
非常幸運的是,有些NoSQL數據庫已經具有部分關係數據庫的關係查詢特性,他們的功能介於key-value和關係數據庫之間,卻具有key-value數據庫的性能,基本能滿足絕大部分web 2.0網站的查詢需求。比如:
MongoDB就帶有關係查詢的功能,能解決常用的關係查詢,所以也是一種非常不錯的選擇。下麵是一些MongoDB的資料:
· 《視覺中國的NoSQL之路:從MySQL到MongoDB》
· 《Choosing a non-relational database; why we migrated from MySQL to MongoDB》
· 最近的一次Mongo Beijing 開發者聚會也有一部分資料。
雖然Foursquare使用MongoDB的宕機事件的出現使人對MongoDB的自動Shard提出了質疑,但是毫無疑問,MongoDB在NoSQL中,是一個優秀的數據庫,其單機性能和功能確實是非常吸引人的。由於上麵的例子有詳細的介紹,本文就不做MongoDB的使用介紹。
Tokyo Tyrant數據庫帶有一個名為table的存儲類型,可以對存儲的數據進行關係查詢和檢索。一個table庫類似於MySQL中的一個表。下麵我們看一個小演示:
我們要存儲一批用戶信息,用戶信息包含用戶名(name),年齡(age),email,最後訪問時間(lastvisit),地區(area)。下麵為寫入的演示代碼:
<?php $tt = new TokyoTyrantTable ( "127.0.0.1", 1978 ); $tt-<vanish ();//清空 $id = $tt->genUid ();//獲取一個自增id //put方法提供數據寫入。 put ( string $key , array $columns ); $tt->put ( $id, array ("id" => $id, "name" => "zhangsan", "age" => 27, "email" => "zhangsan@gmail.com", "lastvisit" =>strtotime ( "2011-3-5 12:30:00" ), "area" => "北京" ) ); $id = $tt->genUid (); $tt->put ( $id, array ("id" => $id, "name" => "lisi", "age" => 25, "email" => "lisi@126.com", "lastvisit" => strtotime( "2011-3-3 14:40:44" ), "area" => "北京" ) ); $id = $tt->genUid (); $tt->put ( $id, array ("id" => $id, "name" => "laowang", "age" => 37, "email" => "laowang@yahoo.com", "lastvisit" =>strtotime ( "2011-3-5 08:30:12" ), "area" => "成都" ) ); $id = $tt->genUid (); $tt->put ( $id, array ("id" => $id, "name" => "tom", "age" => 21, "email" => "tom@hotmail.com", "lastvisit" =>strtotime ( "2010-12-10 13:12:13" ), "area" => "天津" ) ); $id = $tt->genUid (); $tt->put ( $id, array ("id" => $id, "name" => "jack", "age" => 21, "email" => "jack@gmail.com", "lastvisit" =>strtotime ( "2011-02-24 20:12:55" ), "area" => "天津" ) ); //循環打印數據庫的所有數據庫 $it = $tt->getIterator (); foreach ( $it as $k => $v ) { print_r ( $v ); } ?>
比如我們需要查詢年齡為21歲的所有用戶:
<?php $tt = new TokyoTyrantTable ( "127.0.0.1", 1978 ); $query = $tt->getQuery ();
//查詢年齡為21歲的用戶
$query->addCond ( “age”, TokyoTyrant::RDBQC_NUMEQ, “21” );print_r ( $query->search () ); ?>
查詢所有在2011年3月5日之後登陸的用戶:
<?php $tt = new TokyoTyrantTable ( "127.0.0.1", 1978 ); $query = $tt->getQuery (); $query->addCond ( “lastvisit”, TokyoTyrant::RDBQC_NUMGE, strtotime ( "2011-3-5 00:00:00" ) ); print_r ( $query->search () ); ?>
從上麵的示例代碼可以看出,使用起來是非常簡單的,甚至比SQL語句還要簡單。Tokyo Tyrant的表類型存儲還提供了給字段建立普通索引和倒排全文索引,大大增強了其檢索功能和檢索的性能。
所以,完全用NoSQL來構建部分係統,是完全可能的。配合部分帶有關係查詢功能的NoSQL,在開發上比MySQL數據庫更加快速和高效。