說到php連mongoDB,不得不先介紹一下php的官方手冊,網址在:http://us.php.net/manual/en/book.mongo.php
在php的mongo擴充中,提供了4類介面(對象):
1,針對mongoDB串連的操作:Mongo
http://us.php.net/manual/en/class.mongo.php
2,針對mongoDB中資料庫的操作:MongoDB
http://us.php.net/manual/en/class.mongodb.php
3,針對mongoDB中collection的操作:MongoCollection
http://us.php.net/manual/en/class.mongocollection.php
4,針對查詢結果集的操作:MongoCursor
http://us.php.net/manual/en/class.mongocursor.php
與mongoDB建立串連:
直接執行個體化mongo類+建立串連:
$mo = new Mongo();//得到一個Mongo連線物件
執行個體化了一個Mongo類,並且與預設的localhost:27017連接埠的mongoDB建立串連。
如果想串連到其他的主機,可以這樣寫:$mongo = new Mongo("mongodb://username:password@192.168.1.22:12345");
另外一種方式,執行個體化mongo類,再手動建立串連:(http://www.my400800.cn
)
$mongo = new Mongo("mongodb://username:password@192.168.1.22:12345",array('connect'=>false));//初始化類
$mongo->connect();//建立串連
Mongo類中有用的一些方法:
Mongo::listDBs()
http://us.php.net/manual/en/mongo.listdbs.php
返回一個包含當前mongo服務上的庫(DB)資訊的數組。
$mo = new Mongo();
$dbs = $mo->listDBs();//獲得一個包含db資訊的數組
Mongo::selectCollection($db,$coll)
http://us.php.net/manual/en/mongo.selectcollection.php
返回一個當前串連下的某db中的collection對象。
$mo = new Mongo();
$coll = $mo->selectCollection(’db’,'mycoll’);//得到一個collection對象
選擇想要的資料庫(Mongo類):
一種方式:
http://us.php.net/manual/en/mongo.get.php
$mongo = new Mongo();
$db = $mongo->foo;//得到一個MongoDB對象
另一種方式:
http://us.php.net/manual/en/mongo.selectdb.php
$mongo = new Mongo();
$db = $mongo->selectDB(’foo’);//得到一個MongoDB對象
MongoDB中有用的函數:
建立一個MongoDB對象
http://us.php.net/manual/en/mongodb.construct.php
$mo = new Mongo();
$db = new MongoDB($mo,’dbname’);//通過建立方式獲得一個MongoDB對象
刪除當前DB
http://us.php.net/manual/en/mongodb.drop.php
$db = $mo->dbname;
$db->drop();
獲得當前資料庫名
http://us.php.net/manual/en/mongodb.–tostring.php
$db = $mo->dbname;
$db->_tostring();
選擇想要的collection:
A:
$mo = new Mongo();
$coll = $mo->dbname->collname;//獲得一個collection對象
B:
$db = $mo->selectDB(’dbname’);
$coll = $db->collname;
C:
$db = $mo->dbname;
$coll = $db->selectCollectoin(’collname’);//獲得一個collection對象
插入資料(MongoCollection對象):
http://us.php.net/manual/en/mongocollection.insert.php
MongoCollection::insert(array $a,array $options)
array $a 要插入的數組
array $options 選項
safe 是否返回操作結果資訊
fsync 是否直接插入到物理硬碟
常式:
$coll = $mo->db->foo;
$a = array(’a'=>’b');
$options = array(’safe’=>true);
$rs =$coll->insert($a,$options);
$rs為一個array型的數組,包含操作資訊
刪除資料庫中的記錄(MongoCollection對象):
http://us.php.net/manual/en/mongocollection.remove.php
MongoCollection::remove(array $criteria,array $options)
array $criteria 條件
array $options 選項
safe 是否返回操作結果
fsync 是否是直接影響到物理硬碟
justOne 是否隻影響一條記錄
常式:
$coll = $mo->db->coll;
$c = array(’a'=>1,’s’=>array(’$lt’=>100));
$options = array(’safe’=>true);
$rs = $coll->remove($c,$options);
$rs為一個array型的數組,包含操作資訊
更新資料庫中的記錄(MongoCollection對象):
http://us.php.net/manual/en/mongocollection.update.php
MongoCollection::update(array $criceria,array $newobj,array $options)
array $criteria 條件
array $newobj 要更新的內容
array $options 選項
safe 是否返回操作結果
fsync 是否是直接影響到物理硬碟
upsert 是否沒有匹配資料就添加一條新的
multiple 是否影響所有合格記錄,預設隻影響一條
常式:
$coll = $mo->db->coll;
$c = array(’a'=>1,’s’=>array(’$lt’=>100));
$newobj = array(’e'=>’f',’x'=>’y');
$options = array(’safe’=>true,’multiple’=>true);
$rs = $coll->remove($c,$newobj,$options);
$rs為一個array型的數組,包含操作資訊
查詢collection獲得單條記錄(MongoCollection類):
http://us.php.net/manual/en/mongocollection.findone.php
array MongoCollection::findOne(array $query,array $fields)
array $query 條件
array $fields 要獲得的欄位
常式:
$coll = $mo->db->coll;
$query = array(’s’=>array(’$lt’=>100));
$fields = array(’a'=>true,’b'=>true);
$rs = $coll->findOne($query,$fields);
如果有結果就返回一個array,如果沒有結果就返回NULL
查詢collection獲得多條記錄(MongoCollection類):
http://us.php.net/manual/en/mongocollection.find.php
MongoCursor MongoCollection::find(array $query,array $fields)
array $query 條件
array $fields 要獲得的欄位
常式:
$coll = $mo->db->coll;
$query = array(’s’=>array(’$lt’=>100));
$fields = array(’a'=>true,’b'=>true);
$cursor = $coll->find($query,$fields);
返回一個遊標記錄對象MongoCursor。
針對遊標對象MongoCursor的操作(MongoCursor類):
http://us.php.net/manual/en/class.mongocursor.php
迴圈或的結果記錄:
$cursor = $coll->find($query,$fields);
while($cursor->hasNext()){
$r = $cursor->getNext();
var_dump($r);
}
或者
$cursor = $coll->find($query,$fields);
foreache($cursor as $k=>$v){
var_dump($v);
}
或者
$cursor = $coll->find($query,$fields);
$array= iterator_to_array($cursor);
一個血的教訓:
http://us.php.net/manual/en/mongocursor.snapshot.php
在
我們做了find()操作,獲得$cursor遊標之後,這個遊標還是動態,也就是
在我獲得遊標到我迴圈操作完成對應記錄的過程中,預設情況下,這對合格記錄如果增加,結果集也會自動增加。換句話說,在我find()之後,到我的
遊標迴圈完成這段時間,如果再有合格記錄被插入到collection,那麼這些記錄也會被$cursor獲得。
如果你想在獲得$cursor之後的結果集不變化,需要這樣做:
$cursor = $coll->find($query,$fields);
$cursor->snapshot();
foreache($cursor as $k=>$v){
var_dump($v);
}
下面讓我們使用group操作,根據group_id分組,匯總計算count:
<?php
ini_set('mongo.native_long', 1);
$instance = new Mongo();
$instance = $instance->selectCollection('test', 'test');
$keys = array('group_id' => 1);
$initial = array('count' => 0);
$reduce = ' function(obj, prev) { prev.count += obj.count; } ';
$result = $instance->group($keys, $initial, $reduce);
var_dump($result);
?>
|
結果和預想的有出入,count沒有實現累加,而是變成了[object Object],目前,如果必須使用group操作,那麼有兩種方法可以緩解這個問題:
ini_set('mongo.native_long', 0);
|
$initial = array('count' => (float)0);
|
這兩種方法都是治標不治本的權宜之計,既然當前PHP驅動裡group的實現有問題,那我們就繞開它,用其它的方式實現同樣的功能,這個方式就是MapReduce:
<?php
ini_set('mongo.native_long', 1);
$instance = new Mongo();
$instance = $instance->selectDB('test');
$map = ' function() { emit(this.group_id, this.count); } ';
$reduce = ' function(key, values) { var sum = 0;
for (var index in values) { sum += values[index]; }
return sum; } ';
$result = $instance->command(array( 'mapreduce' => 'test', 'map' => $map, 'reduce' => $reduce ));
$result = iterator_to_array($instance->{$result['result']}->find());
var_dump($result);
?>
|