Android 图片、视频存储和读取

1. 应用专属文件

1.1 文件路径

        应用专属文件的访问与更新不需要任何权限。

        应用专属文件包括应用内部空间和应用外部空间。以下是内外部空间对应的目录获取方式。其中`getExternalFilesDir`可以设置对应目录下的文件夹,我们可以使用Environment中设置的变量来指定文件夹,防止搞混。

        这些文件会在应用卸载的时候删除。

        内部空间中的文件其他应用无法访问。外部空间中的文件,其他应用在适当权限下可以访问。

Context.getExternalCacheDir() 
/storage/emulated/0/Android/data/com.example.myapplication/cache

Context.getExternalFilesDir(null)
/storage/emulated/0/Android/data/com.example.myapplication/files

Context.getExternalFilesDir("abcdefg")
/storage/emulated/0/Android/data/com.example.myapplication/files/abcdefg

Context.getCacheDir()
/data/user/0/com.example.myapplication/cache

Context.getFilesDir()
/data/user/0/com.example.myapplication/files

Environment.DIRECTORY_MUSIC = "Music"
Environment.DIRECTORY_PICTURES = "Pictures"
Environment.DIRECTORY_DOWNLOADS = "Download"
Environment.DIRECTORY_DOCUMENTS = "Documents"
Environment.DIRECTORY_SCREENSHOTS = "Screenshots"
Environment.DIRECTORY_MOVIES = "Movies"

         还可以获取外部存储的根目录,对文件进行操作。

Environment.getExternalStorageDirectory();
/storage/emulated/0

Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES) // 必须带个目录名字符串
/storage/emulated/0/Pictures

1.2 文件访问

  1. 使用File API
  2. context.openFileOutput(filename, Context.MODE_...) 获取FileOutStream
  3. context.openFileInput() 获取FileInputStream
  4. 使用文件描述符来访问文件。(如bitmap的生成就可以使用文件描述符)
    ParcelFileDescriptor parcelFileDescriptor =
    getContentResolver().openFileDescriptor(uri, "r");
    FileDescriptor fileDescriptor = parcelFileDescriptor.getFileDescriptor();

2. 共享空间

        共享空间主要使用ContentResolver + MediaStore API来进行访问。

2.1 权限请求

        对于Android10(sdk 29) 以上系统并且开启了分区存储的情况下,如果只是读写自身创建的文件,则不需要请求任何权限。如果需要访问其他应用创建的文件,则需要请求以下权限。(一定不要请求WRITE_EXTERNAL_STORAGE权限,因为对于Android10以上的系统并没有这个权限。)

Manifest.permission.READ_EXTERNAL_STORAGE

        对于Android9及以下的系统,则需要请求下个两个权限。

Manifest.permission.WRITE_EXTERNAL_STORAGE
Manifest.permission.READ_EXTERNAL_STORAGE

 2.2 MediaStoreAPI 访问媒体

2.2.1 查询 

Cursor query = contentResolver.query(
                MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
                new String[]{
                        MediaStore.Images.Media._ID,
                        MediaStore.Images.Media.DATA,
                        MediaStore.Images.Media.DISPLAY_NAME,
                        MediaStore.Images.Media.SIZE
                },
                MediaStore.Images.Media.DATA + " LIKE ?",
                new String[]{"%Screenshot_2022-06-18-10-24%"},
                null
);

while (query.moveToNext()) {
    String id = query.getString(query.getColumnIndex(MediaStore.Images.Media._ID));
    String DATA = query.getString(query.getColumnIndex(MediaStore.Images.Media.DATA));
    String DISPLAY_NAME = query.getString(query.getColumnIndex(MediaStore.Images.Media.DISPLAY_NAME));
    String EXPOSURE_TIME = query.getString(query.getColumnIndex(MediaStore.Images.Media.SIZE));
    uri = ContentUris.withAppendedId(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, Long.parseLong(id));
}

2.2.2  插入

        值得注意的是,使用contentResolve.openOutputStream、contentResolve.openInputStream来根据提供的uri来对文件进行读写。

// 1. 插入个空文件占个坑
ContentValues values = new ContentValues();
values.put(MediaStore.Images.Media.DISPLAY_NAME, "hello.jpg");
values.put(MediaStore.Images.Media.AUTHOR, "李嘉浩");

Uri insert = contentResolver.insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);

try {
    // 2. 向文件写入数据
    OutputStream outputStream = contentResolver.openOutputStream(insert);
    InputStream inputStream = contentResolver.openInputStream(uri); // 这个uri是上面查询部分获取的某一张图片uri。
    byte[] buffer = new byte[1024];
    int byteNum = 0;
    while ((byteNum = inputStream.read(buffer, 0, buffer.length)) != -1) {
        outputStream.write(buffer, 0, byteNum);
    }
} catch (FileNotFoundException e) {
    e.printStackTrace();
} catch (IOException e) {
    e.printStackTrace();
}

参考

官网https://developer.android.com/training/data-storage

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值