Android 保存图片并刷新相册(无需权限)

文章介绍了在Android11及以后版本中,如何在不使用MANAGE_EXTERNAL_STORAGE敏感权限的情况下,通过兼容方式将图片保存到相册,以及利用ACTION_MEDIA_SCANNER_SCAN_FILE通知相册刷新的两种方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

一、背景

之前的保存图片吧,在Android11之前,我会去申请WIRITE权限,11之后,我又去申请MANAGE_EXTERNAL_STORAGE这个权限,该权限将授权读写所有共享存储内容,这也将同时包含非媒体类型的文件。但是在google上架的时候,MANAGE_EXTERNAL_STORAGE属于敏感权限,需要向谷歌做声明,做了,没过,谷歌说可以不要这个权限也可以保存,看来是我无知了,于是去研究了一下下,有了下面的方法。

二、实现

    @SuppressLint("UseCompatLoadingForDrawables")
    @AfterPermissionGranted(200)
    fun saveFreshAppImageToGallery(context: PreViewActivity, imageResId: Int) {
        val drawable = context.getDrawable(imageResId)

        if (drawable is BitmapDrawable) {
            val bitmap = drawable.bitmap
            val displayName = "${System.currentTimeMillis()}.jpg"

            val values = ContentValues().apply {
                put(MediaStore.Images.Media.DISPLAY_NAME, displayName)
                put(MediaStore.Images.Media.MIME_TYPE, "image/jpeg")
                put(MediaStore.Images.Media.DATE_ADDED, System.currentTimeMillis() / 1000)
            }

            val contentResolver = context.contentResolver
            val collection =
                MediaStore.Images.Media.getContentUri(MediaStore.VOLUME_EXTERNAL_PRIMARY)
            val item = contentResolver.insert(collection, values)

            item?.let { uri ->
                try {
                    val outputStream = contentResolver.openOutputStream(uri)
                    bitmap.compress(Bitmap.CompressFormat.JPEG, 100, outputStream)
                    outputStream?.close()
                    Toast.makeText(context, "Save successfully!", Toast.LENGTH_SHORT).show()
                    val mediaScanIntent = Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE)
                    mediaScanIntent.data = uri
                    context.sendBroadcast(mediaScanIntent)
                } catch (e: Exception) {
                    Toast.makeText(context, "Save Failed!", Toast.LENGTH_SHORT).show()
                    e.printStackTrace()
                }
            }
        }
    }

其中:通知相册刷新的逻辑是下面代码

val mediaScanIntent = Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE)
mediaScanIntent.data = uri
context.sendBroadcast(mediaScanIntent)

或者下面的方法

    private fun saveToGallery(photoFile: File) {
        val contentValues = ContentValues().apply {
            put(MediaStore.Images.Media.DISPLAY_NAME, "photo.jpg")
            put(MediaStore.Images.Media.MIME_TYPE, "image/jpeg")
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
                put(MediaStore.Images.Media.IS_PENDING, 1)
            }
        }

        val contentResolver = contentResolver
        val collectionUri = if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
            MediaStore.Images.Media.getContentUri(MediaStore.VOLUME_EXTERNAL_PRIMARY)
        } else {
            MediaStore.Images.Media.EXTERNAL_CONTENT_URI
        }

        val imageUri = contentResolver.insert(collectionUri, contentValues)

        imageUri?.let { uri ->
            try {
                contentResolver.openOutputStream(uri)?.use { outputStream ->
                    val inputStream = FileInputStream(photoFile)
                    inputStream.copyTo(outputStream)
                    inputStream.close()
                    outputStream.close()
                }
                if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
                    contentValues.clear()
                    contentValues.put(MediaStore.Images.Media.IS_PENDING, 0)
                    contentResolver.update(uri, contentValues, null, null)
                }
            } catch (e: IOException) {
            }
        }
    }

原理都是一样的

### Android 开发中保存图片至本地相册的方法 在Android开发过程中,将图片保存到本地相册是一个常见需求。此过程涉及几个重要环节:申请必要权限、获取目标图片(通常来自网络)、执行保存操作以及通知媒体扫描器刷新图库。 #### 权限声明 对于API级别低于29的应用程序,在`AndroidManifest.xml`文件内需声明读写外部存储的权限: ```xml <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> ``` 而对于API 29及以上版本,则推荐采用分区存储方案请求访问特定类型的共享集合权限[^3]。 #### 下载保存图片 下面给出了一段完整的Java代码片段用于说明如何从指定URL下载图片将其存入公共照片目录下: ```java import java.io.*; import okhttp3.*; public class ImageDownloader { private static final OkHttpClient client = new OkHttpClient(); public static void downloadImage(String url, Context context) throws IOException { Request request = new Request.Builder().url(url).build(); try (Response response = client.newCall(request).execute()) { if (!response.isSuccessful()) throw new IOException("Unexpected code " + response); File path; if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q){ ContentValues values = new ContentValues(); values.put(MediaStore.Images.Media.DISPLAY_NAME, System.currentTimeMillis()); values.put(MediaStore.Images.Media.MIME_TYPE, "image/jpeg"); ContentResolver resolver = context.getContentResolver(); Uri collection = MediaStore.Images.Media.EXTERNAL_CONTENT_URI; Uri item = resolver.insert(collection, values); if(item != null){ try(InputStream inputStream = response.body().byteStream(); OutputStream outputStream = resolver.openOutputStream(item)){ byte[] buffer = new byte[4096]; int bytesRead; while ((bytesRead = inputStream.read(buffer)) != -1) { outputStream.write(buffer, 0, bytesRead); } outputStream.flush(); } catch(IOException e){ Log.e("TAG", "Failed to write image data.", e); } } }else{ path = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES); File file = new File(path, String.valueOf(System.currentTimeMillis())); try(FileOutputStream fos = new FileOutputStream(file); InputStream is = response.body().byteStream()){ fos.write(is.readAllBytes()); } } } } } ``` 这段代码首先创建了一个HTTP GET请求来获取远程服务器上的图片资源;接着根据不同安卓版本选择合适的路径进行保存——对于较新的系统版本(Q+)会利用ContentProvider机制直接向媒体数据库插入记录,而旧版则简单地把文件放置于Pictures文件夹里[^4]。 最后提醒一点,完成上述动作之后应当调用MediaScannerConnection.scanFile方法告知操作系统有新内容加入以便及时更新UI界面中的缩略图显示。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

&岁月不待人&

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值