我们最常常遇到的也就是保存图片到SD卡,然后打开系统的图库可以看得到,可是为什么微信QQ之类的应用可以,我们保存到SD卡就看不到呢?原因就是保存到SD卡后我们还要让系统扫描图片到多媒体库,因为系统的图库读的其实是一个多媒体库的数据库,所以我们保存之后如果没有通知系统扫描的画是看不到的
这里要声明一点,后边的扫描要用到获取文件的Mimetype,不知道怎么获取的人看这里:
Demo源码连接在文章尾部给出。
首先,保存图片到SD卡,当然方法不止一种,我这里例举一种:
/**
* 保存图片
* @author YOLANDA
* @param bitmap 图片
* @return
*/
private String saveImg(Bitmap bitmap){
File myappDir = new File(Environment.getExternalStorageDirectory(), "yolanda");
if(myappDir.exists() && myappDir.isFile()) {
myappDir.delete();
}
if (!myappDir.exists()) {
myappDir.mkdir();
}
String fileName = System.currentTimeMillis() + ".png";
File file = new File(myappDir, fileName);
if(file.exists()) {
file.delete();
}
try {
FileOutputStream fos = new FileOutputStream(file);
bitmap.compress(CompressFormat.PNG, 100, fos);
fos.flush();
fos.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return file.getAbsolutePath();
}
那么还有一个问题是,我们怎么保存图片到系统图库呢?为什么要单独说这个问题呢,因为有的手机保存的路径在DCIM/100ANDRo下,有的手机在DCIM/Camera下,所以很难做到兼容,所以我们用到系统的一个方法:
MediaStore.Images.Media.insertImage(getContentResolver(), file.getAbsolutePath(), fileName, null);
这里做个参数说明,第一个参数不用说了,第二个参数是文件在SD卡的路径,第三个是文件保存的名字,第四个是文件描述,可有可无;
那么现在完成的代码就是:
/**
* 保存图片到SD卡
* @author YOLANDA
* @param isInsertGallery 是否保存到图库
* @return
*/
private String saveImg(Bitmap bitmap, boolean isInsertGallery){
File myappDir = new File(Environment.getExternalStorageDirectory(), "yolanda");
if(myappDir.exists() && myappDir.isFile()) {
myappDir.delete();
}
if (!myappDir.exists()) {
myappDir.mkdir();
}
String fileName = System.currentTimeMillis() + ".png";
File file = new File(myappDir, fileName);
if(file.exists()) {
file.delete();
}
try {
FileOutputStream fos = new FileOutputStream(file);
bitmap.compress(CompressFormat.PNG, 100, fos);
fos.flush();
fos.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
if(isInsertGallery) {
try {
MediaStore.Images.Media.insertImage(getContentResolver(), file.getAbsolutePath(), fileName, null);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
return file.getAbsolutePath();
}
这样就可以做到既保存到我们自定义的目录中,又保存在系统图库中了。现在打开图库就直接可以看到图片了。
看似好像解决了我们刚开始说的保存图片到SD打开图库可以看到的问题了,那么有人就问了,QQ和微信人家也不是直接存到图库的啊,人家咋可以保存到自己的目录打开图库就可以看到呢?好,下面我们就来解决这个问题
其实,我们保存图片到SD卡时,要通知系统来扫描,这里分为两种情况:
1、在Androd4.4出来之前,我们都可以发送一个系统广播出去,这个广播既可以扫描一个文件夹:
sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED, Uri.parse("file://"+ Environment.getExternalStorageDirectory())));
又可以扫描一个文件:
sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.parse("file://" + filePath)));
所以在Android4.4之前的的系统中最终的代码是:
/**
* 保存后用广播扫描,Android4.4以下使用这个方法
* @author YOLANDA
*/
private void saveBroadcast(){
String filePath = saveImg(true);
sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.parse("file://" + filePath)));
Toast.makeText(this, "保存成功:" + filePath, Toast.LENGTH_LONG).show();
}
/**
* 保存图片到SD卡
* @author YOLANDA
* @param isInsertGallery 是否保存到图库
* @return
*/
private String saveImg(boolean isInsertGallery){
File myappDir = new File(Environment.getExternalStorageDirectory(), "yolanda");
if(myappDir.exists() && myappDir.isFile()) {
myappDir.delete();
}
if (!myappDir.exists()) {
myappDir.mkdir();
}
String fileName = System.currentTimeMillis() + ".png";
File file = new File(myappDir, fileName);
if(file.exists()) {
file.delete();
}
try {
FileOutputStream fos = new FileOutputStream(file);
bitmap.compress(CompressFormat.PNG, 100, fos);
fos.flush();
fos.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
if(isInsertGallery) {
try {
MediaStore.Images.Media.insertImage(getContentResolver(), file.getAbsolutePath(), fileName, null);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
return file.getAbsolutePath();
}
为什么说是Android4.4之前呢?因为Android4.4开始,Android把系统广播的权限回收了,只有系统应用才可以发送系统广播,那么这还怎么活啊???不要着急,车道山前必有路,还有一个MediaScannerConnection,这个玩意儿可强大了,不仅仅可以搞定图片,只要是媒体文件都可以用他来扫描,不过要知道文件的MimeType,不知道怎么获取文件MimeType的看这里:
2、在Android4.4和之后用MediaScannerConnection来扫描。
这里讲解一下怎么用MediaScannerConnection来扫描文件,我封装好了一个类出来:
/**
* 实现MediaScannerConnection.MediaScannerConnectionClient
* @author YOLANDA
* @Time 2015年4月8日 上午9:03:54
*/
public class MediaScanner implements MediaScannerConnection.MediaScannerConnectionClient {
/**
* 扫描对象
*/
private MediaScannerConnection mediaScanConn = null;
public MediaScanner(Context context) {
//实例化
mediaScanConn = new MediaScannerConnection(context, this);
}
/**文件路径集合**/
private String[] filePaths;
/**文件MimeType集合**/
private String[] mimeTypes;
/**
* 扫描文件
* @author YOLANDA
* @param filepaths
* @param mimeTypes
*/
public void scanFiles(String[] filePaths, String[] mimeTypes) {
this.filePaths = filePaths;
this.mimeTypes = mimeTypes;
mediaScanConn.connect();//连接扫描服务
}
/**
* @author YOLANDA
*/
@Override
public void onMediaScannerConnected() {
for (int i = 0; i < filePaths.length; i++) {
mediaScanConn.scanFile(filePaths[i], mimeTypes[i]);//服务回调执行扫描
}
filePaths = null;
mimeTypes = null;
}
private int scanTimes = 0;
/**
* 扫描一个文件完成
* @author YOLANDA
* @param path
* @param uri
*/
@Override
public void onScanCompleted(String path, Uri uri) {
scanTimes ++;
if(scanTimes == filePaths.length) {//如果扫描完了全部文件
mediaScanConn.disconnect();//断开扫描服务
scanTimes = 0;//复位计数
}
}
}
这个类可以单个文件,也可以扫描多个文件,用法就是直接new一个MediaScannerConnection对象,这个对象需要一个实现了MediaScannerConnection.MediaScannerConnectionClient接口的对象,我们调用的时候步骤是:
1、new一个MediaScannerConnection对象
2、传要扫描的文件路径和文件MimeType进来
3、调用MediaScannerConnection对象的connect()方法,会触发MediaScannerConnection.MediaScannerConnectionClient接口的onMediaScannerConnected方法
4、在MediaScannerConnection.MediaScannerConnectionClient接口的onMediaScannerConnected()方法内调用用MediaScannerConnection.scanFile(String path, String mimeType)来扫描文件
5、扫描结束后会触发MediaScannerConnection.MediaScannerConnectionClient接口的onScanCompleted方法,在这个方法里调用MediaScannerConnection.disconnect()断开连接
所以我们完成的代码是:
public class MainActivity extends Activity implements View.OnClickListener {
private Bitmap bitmap = null;
private ImageView imgUserHead;//假如这是用户头像
private Button btnSaveBoradcase;//保存并广播扫描
private Button btnSaveScan;//保存并扫描到媒体库
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
imgUserHead = (ImageView) findViewById(R.id.img_userhead);
btnSaveBoradcase = (Button) findViewById(R.id.btn_broadcast);
btnSaveScan = (Button) findViewById(R.id.btn_scan);
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.scan);
imgUserHead.setImageBitmap(bitmap);
btnSaveBoradcase.setOnClickListener(this);
btnSaveScan.setOnClickListener(this);
}
/**
* @author YOLANDA
* @param v
*/
@Override
public void onClick(View v) {
if(v == btnSaveBoradcase) {
saveBroadcast();
}
if(v == btnSaveScan) {
saveScan();
}
}
/**
* 保存后用广播扫描,Android4.4以下使用这个方法
* @author YOLANDA
*/
private void saveBroadcast(){
String filePath = saveImg(true);
sendBroadcast(new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, Uri.parse("file://" + filePath)));
Toast.makeText(this, "保存成功:" + filePath, Toast.LENGTH_LONG).show();
}
/**
* 保存后用MediaScanner扫描,通用的方法
* @author YOLANDA
*/
private void saveScan(){
String filePath = saveImg(false);
MediaScanner mediaScanner = new MediaScanner(this);
String[] filePaths = new String[]{filePath};
String[] mimeTypes = new String[]{MimeTypeMap.getSingleton().getMimeTypeFromExtension("png")};
mediaScanner.scanFiles(filePaths, mimeTypes);
Toast.makeText(this, "保存成功:" + filePath, Toast.LENGTH_LONG).show();
}
/**
* 保存图片到SD卡
* @author YOLANDA
* @param isInsertGallery 是否保存到图库
* @return
*/
private String saveImg(boolean isInsertGallery){
File myappDir = new File(Environment.getExternalStorageDirectory(), "yolanda");
if(myappDir.exists() && myappDir.isFile()) {
myappDir.delete();
}
if (!myappDir.exists()) {
myappDir.mkdir();
}
String fileName = System.currentTimeMillis() + ".png";
File file = new File(myappDir, fileName);
if(file.exists()) {
file.delete();
}
try {
FileOutputStream fos = new FileOutputStream(file);
bitmap.compress(CompressFormat.PNG, 100, fos);
fos.flush();
fos.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
if(isInsertGallery) {
try {
MediaStore.Images.Media.insertImage(getContentResolver(), file.getAbsolutePath(), fileName, null);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
return file.getAbsolutePath();
}
/**
* @author YOLANDA
*/
@Override
protected void onDestroy() {
super.onDestroy();
if(bitmap != null) {
bitmap.recycle();
bitmap = null;
}
}
}