目前很多APP都用到相机相册,最常见的就是用户设置自己头像,使用的比较频繁。还有就是QQ,微信的分享图片.
package com.example.aa.cameral;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Build;
import android.provider.MediaStore;
import android.support.v4.content.FileProvider;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
public class MainActivity extends AppCompatActivity {
private ImageView img;
private Uri uriForFile;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button bt = (Button) findViewById(R.id.button);
img = (ImageView) findViewById(R.id.imageView);
bt.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
File outimage = new File(getExternalCacheDir(), "output_image.jpg");
if (outimage.exists()) {
outimage.delete();
}
try {
outimage.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
//获得URI
if (Build.VERSION.SDK_INT >= 24) {
uriForFile = FileProvider.getUriForFile(MainActivity.this, "com.example.aa.fileprovider", outimage);
} else {
uriForFile = Uri.fromFile(outimage);
}
//启动相机
Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT, uriForFile);
startActivityForResult(intent, 1);
}
});
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == 1) {
switch (requestCode) {
case 1:
try {
Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(uriForFile));
img.setImageBitmap(bitmap);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
break;
default:
break;
}
}
}
}
上述代码中的 getExternalCacheDir(),方法是获得手机的关联缓存目录,具体的路径是sdcard/Android/data/包名/cache,我们之所以使用这个关联缓存,是因为Android6.0引入的运行时权限,而SD被列为危险权限,为了跳过这一步。
//获得URI
if (Build.VERSION.SDK_INT >= 24) {
uriForFile = FileProvider.getUriForFile(MainActivity.this, "com.example.aa.fileprovider", outimage);
} else {
uriForFile = Uri.fromFile(outimage);
}
这段代码,做了一个判断,Android7.0就是用FileProvider.getUriForFile 得到一个封装的uri对象,第三个参数是任意的唯一字符串 之所以这样做,这里主要是因为Android7.0下使用本地真实路径被认为是不安全的,如果是7.0以前的版本,那么就是用fromFile得到一个uri对象,这个对象就包含了对应的文件地址。 前面的fileProvider是一个特殊的内容提供器,和内容提供者类似的机制,可以对封装的URI共享给外部访问,从而提高应用的安全性。
要在清单文件中注册内容提供者:
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="com.example.aa.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
里面的属性值 android:authorities="com.example.aa.fileprovider"必须和刚才的FileProvider.getUriForFile()第二个参数一致, 还要在里面指定一个共享路径:
android:resource="@xml/file_paths",具体代码如下
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path name="my_images" path="" />
</paths>
里面name 属性随意写,后面值不写代表整个SDCard,很多时候还有从相册选取的功能,见下一遍文章。
项目地址:https://github.com/zh2016hz/CameraDemo.git