首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 数据库 > 其他数据库 >

Mongodb运用GridFS存储文件

2012-06-26 
Mongodb使用GridFS存储文件先来看一个简单的例子:public class Mongofs {private static final String USE

Mongodb使用GridFS存储文件
先来看一个简单的例子:

public class Mongofs {private static final String USER = "rechard";private static final String PASS = "root";public static void main(String[] args) throws Exception {Mongo m = new Mongo("localhost");DB db = m.getDB("newdb");boolean auth = db.authenticate(USER, PASS.toCharArray());if (auth) {DBCollection coll = db.getCollection("testColl");db.requestStart();String file, newFilename, bucket, saveTo;bucket = "fs";newFilename = "mm";file = "/home/rechard/Pictures/105066115253.jpg";saveTo = "/home/rechard/Desktop/1.jpg";storeFile(db, bucket, file, newFilename);findSingleFileAndSaveAs(db, bucket, newFilename, saveTo);remove(db, bucket, newFilename);db.requestDone();} }/** * 存储文件到mongo * @param db * @param bucket * @param file * @param fn * @throws Exception */static void storeFile(DB db, String bucket, String file, String fn)throws Exception{File files = new File(file);// 创建一个GridFS实例GridFS gfs = new GridFS(db, bucket);GridFSInputFile gfsInput = gfs.createFile(files);// 指定一个GridFS实体的名字gfsInput.setFilename(fn);gfsInput.save();}/** * 查找单个文件并保存 * @param db * @param bucket * @param fn * @param saveTo * @throws Exception */static void findSingleFileAndSaveAs(DB db, String bucket, String fn, String saveTo) throws Exception{GridFS gfs = new GridFS(db, bucket);GridFSDBFile dbFile = gfs.findOne(fn);//System.out.println(dbFile);if (dbFile != null) System.out.println("file size:" + dbFile.writeTo(saveTo));}/** * 从mongodb删除文件 * @param db * @param bucket * @param fn */static void remove(DB db, String bucket, String fn){GridFS gfs = new GridFS(db, bucket);gfs.remove(fn);}}


以上代码仅供测试和抛砖引玉,不作具体解决方法,详细信息请参考http://www.mongodb.org/

1.关于GridFS的构造函数参数bucket,我认为是一个标识符,即在newdb这个数据库下,不同的bucket代表不同的表(类型),而一个类型又可以包含多个文件。当然,bucket也不是必须要传的,如果觉得麻烦可以使用另一个Constructor-GridFS(DB db),默认bucket是"fs".

2.一个文件在mongodb里存储的文件,看起来是大概是这样的:

详细的解释如下:
{  "_id" : <unspecified>,                  // 文件的唯一标示id,由mongo自动创建  "length" : data_number,                 // 文件的大小  "chunkSize" : data_number,              // 块大小,默认是256k  "uploadDate" : data_date,               // 存储时间  "md5" : data_string                     // 文件的md5码}


3.关于GridFSDBFile的writeTo方法有一个陷阱:
该方法名的三个重载方法实际上都存在返回值(long),他表示了这个文件的物理字节大小。如图:
当然你也可以不返回而直接调用这个方法。但请注意,前面最好加上GridFSDBFile对象不为空的判断,否则程序将有可能抛出异常。

4.GridFS的remove方法,意味着从mongodb中删除某个文件,返回值为void。这也正是让我警惕的地方。如果因为传参错误,remove方法接受了不正确的value或者空串,从而导致删除失败。想象一下,如果一张本该删除的图片却出现在首页上,但程序却没有给你哪怕一点儿提示!

ps:因为32位机器有2G的data size的限制,所以玩玩就行了。不过真正的服务器是不会有这个限制的,并且配合Nginx存取速度也十分理想,网上也已经有类似的测试结果。对于同类产品来,mongo的性能还是比较高的(虽然是以牺牲磁盘空间作为前提),而且也支持类似mysql的master-master replic双机热备份。缺点是,对于原子性的生产环境仍然不如传统的关系型数据库,这点很无奈。同时mongo也不支持事务。

总的来说,mongodb还是很不错的。也许未来几年,关系型数据库和NoSQL还会长期的并存下去。但谁又能说的准呢,就让我们拭目以待吧。
1 楼 dadadada2x 2011-08-23   我想给gridfs文件非空验证
validates_presence_of :result_file, :message => "不能为空。"
出现
Could not open file matching {"_id"=>BSON::ObjectId('4e53422d64de3709dc000002')} {}
错误请问一下这是怎么回事 2 楼 GreenLife 2011-08-27   dadadada2x 写道我想给gridfs文件非空验证
validates_presence_of :result_file, :message => "不能为空。"
出现
Could not open file matching {"_id"=>BSON::ObjectId('4e53422d64de3709dc000002')} {}
错误请问一下这是怎么回事

不知道你是指存进去,还是取出来的时候,看错误信息似乎是id不匹配造成的。
如果存进去的话,可以使用File类的exists()判断;取文件的话,楼主已经提到了,可以用dbFile != null进行判断

热点排行