一、首先在AndroidManifest.xml中加入权限
<!--读取存储卡中的内容 修改或删除存储卡中的内容--> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
因为这属于危险权限,需要在java代码中动态申请。
二、activity_save_image.xml 布局一个要保存的图片,一个保存按钮
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" tools:context=".SaveImageActivity"> <ImageView android:layout_width="match_parent" android:layout_height="0dp" android:layout_weight="1" android:scaleType="fitCenter" android:src="@drawable/image1" /> <LinearLayout android:layout_width="match_parent" android:layout_height="60dp" android:gravity="center"> <Button android:id="@+id/btn_saveImage" android:layout_width="150dp" android:layout_height="50dp" android:text="保存图片到相册" android:textColor="@color/black" /> </LinearLayout> </LinearLayout>
三、SaveImageActivity中我会把Android系统10之前的写法,和最新的写法都写出来,供大家参考学习。
public class SaveImageActivity extends AppCompatActivity implements View.OnClickListener { private Button btn_saveImage; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_save_image); btn_saveImage = findViewById(R.id.btn_saveImage); btn_saveImage.setOnClickListener(this); } @Override public void onClick(View v) { if (v.getId() == R.id.btn_saveImage) {//对于共享区间写入的权限,在API29 Android系统10之前是需要申请的 //在API29及之后是不需要申请的,默认是允许的 if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Q && ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED) { ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE}, 0); } else { //保存图片到相册 SaveImage(); } } } //请求权限后的结果回调 @Override public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) { super.onRequestPermissionsResult(requestCode, permissions, grantResults); if (requestCode == 0) { if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) { //保存图片到相册 SaveImage(); } else { Toast.makeText(this, "你拒绝了该权限,无法保存图片!", Toast.LENGTH_SHORT).show(); } } } private void SaveImage() { //获取要保存的图片的位图 // Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.image1); //API 29之前可用. API29之后该方法已经被弃用了 //MediaStore 相当于管理媒体资源的一个管理器,类似于一个数据库,对媒体资源的一个索引(包括图片 音频 视频),在里面都有索引 // if (MediaStore.Images.Media.insertImage(getContentResolver(), bitmap, "", "") == null) { // Toast.makeText(this, "保存失败!", Toast.LENGTH_SHORT).show(); // } else { // Toast.makeText(this, "保存成功!", Toast.LENGTH_SHORT).show(); // } //创建一个子线程,将耗时任务在子线程中完成,防止主线程被阻塞 new Thread(new Runnable() { @Override public void run() { try { //获取要保存的图片的位图 Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.image1); //创建一个保存的Uri Uri saveUri = getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, new ContentValues()); if (TextUtils.isEmpty(saveUri.toString())) { Looper.prepare(); Toast.makeText(SaveImageActivity.this, "保存失败!", Toast.LENGTH_SHORT).show(); Looper.loop(); return; } OutputStream outputStream = getContentResolver().openOutputStream(saveUri); //将位图写出到指定的位置 //第一个参数:格式JPEG 是可以压缩的一个格式 PNG 是一个无损的格式 //第二个参数:保留原图像90%的品质,压缩10% 这里压缩的是存储大小 //第三个参数:具体的输出流 if (bitmap.compress(Bitmap.CompressFormat.JPEG, 90, outputStream)) { Looper.prepare(); Toast.makeText(SaveImageActivity.this, "保存成功!", Toast.LENGTH_SHORT).show(); Looper.loop(); } else { Looper.prepare(); Toast.makeText(SaveImageActivity.this, "保存失败!", Toast.LENGTH_SHORT).show(); Looper.loop(); } } catch (Exception e) { e.printStackTrace(); } } }).start(); } }
通知图库进行刷新的方法
Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE); Uri uri = Uri.fromFile(file); intent.setData(uri); sendBroadcast(intent);
博主已测试该功能,在Android 10 和Android 9的手机上面均能正常保存图片~ 如有不当之处,还望指正出来!