由于MongoDB的文檔結構為BJSON格式(BJSON全稱:Binary JSON),而BJSON格式本身就支持保存二進制格式的數(shù)據(jù),因此可以把文件的二進制格式的數(shù)據(jù)直接保存到MongoDB的文檔結構中。但是由于一個 BJSON的最大長度不能超過4M,所以限制了單個文檔中能存入的最大文件不能超過4M。為了提供對大容量文件存取的支持,samus驅動提供了 “GridFS”方式來支持,“GridFS”方式文件操作需要引入新的程序集“MongoDB.GridFS.dll”。下面我們分別用兩種方式來實 現(xiàn)。
一、在文檔對象中存取文件
當文件大小較小的時候,直接存入文檔對象實現(xiàn)起來更簡潔。比如大量圖片文件的存取等,一般圖片文件都不會超過4M。我們先實現(xiàn)一個上傳圖片存入數(shù)據(jù)庫,再取出來寫回頁面的例子:
1. 把圖片存到BJSON中
復制代碼 代碼如下:
/// <summary>
/// 把圖片存到BJSON中
/// </summary>
public void SaveImgBJSON(byte[] byteImg)
{
Document doc = new Document();
doc["ID"] = 1;
doc["Img"] = byteImg;
mongoCollection.Save(doc);
}
2. 獲取BJSON方式存儲的圖片字節(jié)數(shù)據(jù)
復制代碼 代碼如下:
/// <summary>
/// 獲取BJSON方式存儲的圖片字節(jié)數(shù)據(jù)
/// </summary>
public byte[] GetImgBJSON()
{
Document doc= mongoCollection.FindOne(new Document { { "ID", 1 } });
return doc["Img"] as Binary;
}
上面兩段代碼是在對MongoDB相關操作進行BLL封裝類中添加的兩個方法,封裝方式查看上節(jié)內容。下面看看在webform中如何調用:
在界面拖出一個FileUpload控件和一個Button控件,頁面cs類加如下方法:
復制代碼 代碼如下:
protected void Button1_Click(object sender, EventArgs e)
{
ImgBLL imgBll = new ImgBLL();
imgBll.DeleteAll();
imgBll.SaveImgBJSON(FileUpload1.FileBytes);
Response.BinaryWrite(imgBll.GetImgBJSON());
}
二、用GridFS方式存取文件
在實現(xiàn)GridFS方式前我先講講它的原理,為什么可以存大文件。驅動首先會在當前數(shù)據(jù)庫創(chuàng)建兩個集合:"fs.files" 和"fs.chunks"集合,前者記錄了文件名,文件創(chuàng)建時間,文件類型等基本信息;后者分塊存儲了文件的二進制數(shù)據(jù)(并支持加密這些二進制數(shù)據(jù))。分 塊的意思是把文件按照指定大小分割,然后存入多個文檔中。"fs.files"怎么知道它對應的文件二進制數(shù)據(jù)在哪些塊呢?那是因為 在"fs.chunks"中有個"files_id"鍵,它對應"fs.files"的"_id"。"fs.chunks"還有一個鍵(int 型)"n",它表明這些塊的先后順序。這兩個集合名中的"fs"也是可以通過參數(shù)自定義的。
如果你只是想知道怎么用,可以忽略上面這段話,下面將用法:
1. GridFS方式的文件新建,讀取,刪除
復制代碼 代碼如下:
private string GridFsSave(byte[] byteFile)
{
string filename = Guid.NewGuid().ToString();
//這里GridFile構造函數(shù)有個重載,bucket參數(shù)就是用來替換那個創(chuàng)建集合名中默認的"fs"的。
GridFile gridFile = new GridFile(mongoDatabase);
using (GridFileStream gridFileStream = gridFile.Create(filename))
{
gridFileStream.Write(byteFile, 0, byteFile.Length);
}
return filename;
}
private byte[] GridFsRead(string filename)
{
GridFile gridFile = new GridFile(mongoDatabase);
GridFileStream gridFileStream = gridFile.OpenRead(filename);
byte[] bytes = new byte[gridFileStream.Length];
gridFileStream.Read(bytes, 0, bytes.Length);
return bytes;
}
private void GridFsDelete(string filename)
{
GridFile gridFile = new GridFile(mongoDatabase);
gridFile.Delete(new Document("filename", filename));
}
2. 再次封裝GridFS操作,新文檔只存儲文件名稱,相當于只是一個鍵,新文檔還可以有除“文件名”之外其他的鍵。
復制代碼 代碼如下: