MainActivity
package com.example.cameraalbumtest;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.content.FileProvider;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.provider.MediaStore;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
public class MainActivity extends AppCompatActivity {
public static final int TAKE_PHOTO = 1;
private ImageView picture;
private Uri imageUri;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Button takePhoto = (Button)findViewById(R.id.take_photo);
picture = (ImageView)findViewById(R.id.picture);
takePhoto.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//创建File对象,用于存储拍照后的图片
/*给图片命名,并将它存放在手机SD卡的应用关联缓存目录下
* 应用关联缓存目录就是SD卡专门用于存放当前应用缓存数据的位置
* 调用getExternalCacheDir()可以得到这个目录
* 具体的路径是/sdcard/Android/data/<package name>/cache
* 因为读写SD卡的其他目录都要进行运行时权限处理,而这个会跳过这一步
* */
File outputImage = new File(getExternalCacheDir(), "output_image.jpg");
try {
if (outputImage.exists()) {
outputImage.delete();
}
outputImage.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
if(Build.VERSION.SDK_INT >= 24){
/*调用FileProvider的getUriForFile()方法将File对象转换成一个封装过的Uri对象
* getUriForFile()方法接收三个参数
* 第一个参数:要求传入Context对象
* 第二个参数:可以是任意唯一的字符串
* 第三个参数:刚创建的File对象
*
*
* 之所以要进行一层这样的转换,是因为直接使用本地真是路径的Uri被认为是不安全的
* 会抛出一个FileUriExposedException异常
* 而FileProvider则是一种特殊的内容提供器
* 它使用了和内容提供器类似的机制来对数据进行保护
* 可以选择性地将封装过的Uri共享给外部,从而提高了应用的安全性
* */
imageUri = FileProvider.getUriForFile(MainActivity.this,"com.example.cameraalbumtest.fileprovider",outputImage);
}
else{
imageUri = Uri.fromFile(outputImage);
}
//启动相机程序
/*构建出一个Intent对象
* 将这个Intent的action指定为android.media.action.IMAGE_GAPTURE
* 再调用Intent的putExtra()方法指定图片的输出地址
* 这里填入的是刚刚得到的Uri对象
* 最后再调用startActivityForResult()来启动活动
* 由于这里使用的是隐式Intent,系统会找出能够响应这个Intent的活动去启动
* 这样照相机的程序就会被打开,拍下的照片会输出到output_image.jpg中
* */
Intent intent = new Intent("android.media.action.IMAGE_GAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT,imageUri);
startActivityForResult(intent,TAKE_PHOTO);
}
});
}
/* 刚使用的是startActivityForResult()来启动活动
* 因此拍完照后会有结果返回到onActivityResult()方法中
* 如果发现拍照成功,就可以调用BitmapFactory的decodeStream()
* 将output_image.jpg这张照片解析为Bitmap对象,然后把它设置到ImageView中显示出来
* */
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
switch (requestCode) {
case TAKE_PHOTO:
if (requestCode == RESULT_OK) {
try {
//将拍摄的照片显示出来
Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));
picture.setImageBitmap(bitmap);
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
break;
default:
break;
}
}
}
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical" android:layout_width="match_parent"
android:layout_height="match_parent">
<Button
android:id="@+id/take_photo"
android:text="Take photo"
android:layout_width="match_parent"
android:layout_height="wrap_content"/>
<ImageView
android:id="@+id/picture"
android:layout_gravity="center_horizontal"
android:layout_width="wrap_content"
android:layout_height="wrap_content"/>
</LinearLayout>
file_paths.xml文件,res->xml目录下
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path name = "my_images" path="Android/data/com.example.cameraalbumtest.fileprovider" />
</paths>
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.cameraalbumtest">
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
/*
android:authorities属性的值必须要和刚才FileProvider.getUriForFile()方法中的第二个参数一致
使用<meta-data></meta-data>来指定Uri的共享路径,并引用了一个xml/file_paths资源
*/
<provider
android:authorities="com.example.cameraalbumtest.fileprovider"
android:name="androidx.core.content.FileProvider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths"/>
</provider>
</application>
</manifest>
源程序下载