本文简单研究一下MediaScanner生成及保存thumbnail的方式,并给出代码快速查询图片的thumbnail,

希望可以抛块儿砖,引出一大堆玉来~

1. 手动执行Dev Tools /Media Scanner,或插入SD卡的时候,会启动MediaScannerService,把媒体文件更新到media数据库,

如果是图片文件会同时生成thumbnail

这部分代码在 packages\providers\MediaProvider

2. 数据库中图片及thumbnail的表结构

adb shell进入目录 /data/data/com.android.providers.media/databases

可以看到两个数据库internal.db external-xx.db,其中external.db中保存着sd卡中的媒体信息

我们用sqlit查看一下 

sqlite3 external-xx.db

.table

可以看到有很多媒体文件相关的表,我们需要关心p_w_picpaths跟thumbnails这个表

p_w_picpaths表结构:

view plaincopy to clipboardprint?
p_w_picpaths (_id INTEGER PRIMARY KEY,_data TEXT,_s 
ize INTEGER,_display_name TEXT,mime_type TEXT,title TEXT,date_added INTEGER,date 
_modified INTEGER,description TEXT,picasa_id TEXT,isprivate INTEGER,latitude DOU 
BLE,longitude DOUBLE,datetaken INTEGER,orientation INTEGER,mini_thumb_magic INTE 
GER,bucket_id TEXT,bucket_display_name TEXT) 


thumbnails表结构:

view plaincopy to clipboardprint?
thumbnails (_id INTEGER PRIMARY KEY,_ 
data TEXT,p_w_picpath_id INTEGER,kind INTEGER,width INTEGER,height INTEGER) 


我们看一下在我的模拟器上实际的数据:

select * from p_w_picpaths;

view plaincopy to clipboardprint?
4|/mnt/sdcard/Android/1289887991860.jpg|40449|1289887991860.jpg|p_w_picpath/jpeg|12898 
87991860|1290054484|1290064872||||||1290064872000|0|-5876966875320966333|1801299 
020|Android 
8|/mnt/sdcard/20101126062805_hahaha.bmp|65590|20101126062805_hahaha.bmp|p_w_picpath/x- 
ms-bmp|20101126062805_hahaha|1290752887|1290752885||||||1290752885000||536666747 
8445081544|-1595679508|sdcard 
9|/mnt/sdcard/20101126064005_20101126062805_hahaha.bmp|65590|20101126064005_2010 
1126062805_hahaha.bmp|p_w_picpath/x-ms-bmp|20101126064005_20101126062805_hahaha|129075 
3607|1290753606||||||1290753606000||2418262411059016544|-1595679508|sdcard 


select * from thumbnails;

view plaincopy to clipboardprint?
4|/mnt/sdcard/DCIM/.thumbnails/1290054484766.jpg|4|1|400|300 
6|/mnt/sdcard/DCIM/.thumbnails/1290752887313.jpg|8|1|128|128 
7|/mnt/sdcard/DCIM/.thumbnails/1290753608349.jpg|9|1|128|128 


3. 现在可以看到sd卡下面的 DCIM/.thumbnails目录做什么用了,MediaScanner把生成的thumbnail放到这里,

并在数据库中来维护thumbnail跟图片的对应关系:

p_w_picpaths表中第一列是图片的id, 对应的thumbnail表中的p_w_picpath_id项

p_w_picpaths表中第二列式图片的路径,thumbnail表中第二列是thumbnail的路径

4. 从这个结果我们可以这样来查询某张图片的thumbnail (我自己想的,或许有更好的方法)

当前文件路径 ---> 查询p_w_picpaths表得到 _id ---> 用 _id = p_w_picpath_id查询thumbnials表 得到thumbnial 的路径

5. 根据文件路径file_path的到_id:

view plaincopy to clipboardprint?
String[] projection2 = { 
"_id", 
//"_data" 
}; 
Uri uri2 = MediaStore.Images.Media.getContentUri("external"); 
String where = String.format( 
"_data = '%s' ", 
file_path); 
Cursor c2 = Media.query(getContentResolver(), uri2, 
projection2, where , null); 
long p_w_picpath_id = c2.getLong(c2.getColumnIndexOrThrow("_id")); 


6. 根据p_w_picpath_id查询thumbnail路径

view plaincopy to clipboardprint?
String[] projection = { 
"_data" , 
//"p_w_picpath_id" 
}; 


Cursor c = Thumbnails.queryMiniThumbnail(getContentResolver(), p_w_picpath_id, 
Thumbnails.MINI_KIND, projection); 


thumbnail_path = c.getString(c.getColumnIndexOrThrow("_data")); 



7. 这样就从当前文件得到了对应的thumbnail,前提是MediaScanner帮你扫描过了,当然你也可以在代码中自己启动扫描

然后把thumnail路径传递给你用来显示的ImageView就可以了