我正在研究从web服务器下载图像的功能,在屏幕上显示它,如果用户希望保持图像,将其保存在SD卡的某个文件夹中。有没有一种简单的方法来获取位图并将其保存到我选择的SD卡文件夹中?

我的问题是,我可以下载图像,在屏幕上显示为位图。我能够找到的将图像保存到特定文件夹的唯一方法是使用FileOutputStream,但这需要一个字节数组。我不知道如何转换(如果这是正确的方式)从位图到字节数组,所以我可以使用FileOutputStream写入数据。

我的另一个选择是使用MediaStore:

MediaStore.Images.Media.insertImage(getContentResolver(), bm,
    barcodeNumber + ".jpg Card Image", barcodeNumber + ".jpg Card Image");

它可以很好地保存到SD卡,但不允许您自定义文件夹。


当前回答

下面是保存位图到文件的示例代码:

public static File savebitmap(Bitmap bmp) throws IOException {
    ByteArrayOutputStream bytes = new ByteArrayOutputStream();
    bmp.compress(Bitmap.CompressFormat.JPEG, 60, bytes);
    File f = new File(Environment.getExternalStorageDirectory()
            + File.separator + "testimage.jpg");
    f.createNewFile();
    FileOutputStream fo = new FileOutputStream(f);
    fo.write(bytes.toByteArray());
    fo.close();
    return f;
}

现在调用这个函数将位图保存到内部内存中。

文件newfile = savebitmap(位图);

我希望它能帮助你。 快乐编码生活。

其他回答

有些格式,比如无损的PNG,会忽略质量设置。

try (FileOutputStream out = new FileOutputStream(filename)) {
    bmp.compress(Bitmap.CompressFormat.PNG, 100, out); // bmp is your Bitmap instance
    // PNG is a lossless format, the compression factor (100) is ignored
} catch (IOException e) {
    e.printStackTrace();
}

在Android 4.4 Kitkat之后,截至2017年,Android 4.4及以下的份额约为20%并且正在下降,使用File类和getExternalStorageDirectory()方法保存到SD卡是不可能的。这个方法返回你的设备内部内存和图像保存为每个应用程序可见。你也可以只保存你的应用程序私有的图像,当用户使用openFileOutput()方法删除你的应用程序时删除。

从Android 6.0开始,您可以将SD卡格式化为内部内存,但只能对您的设备私有。(如果你格式化SD车作为内部存储器,只有你的设备可以访问或看到它的内容)你可以使用其他答案保存到SD卡,但如果你想使用一个可移动的SD卡,你应该阅读下面我的回答。

您应该使用存储访问框架获取uri到activity的onActivityResult方法来获取用户选择的文件夹,并添加检索持久权限,以便用户重新启动设备后能够访问文件夹。

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);

    if (resultCode == RESULT_OK) {

        // selectDirectory() invoked
        if (requestCode == REQUEST_FOLDER_ACCESS) {

            if (data.getData() != null) {
                Uri treeUri = data.getData();
                tvSAF.setText("Dir: " + data.getData().toString());
                currentFolder = treeUri.toString();
                saveCurrentFolderToPrefs();

                // grantUriPermission(getPackageName(), treeUri,
                // Intent.FLAG_GRANT_READ_URI_PERMISSION |
                // Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

                final int takeFlags = data.getFlags()
                        & (Intent.FLAG_GRANT_READ_URI_PERMISSION | Intent.FLAG_GRANT_WRITE_URI_PERMISSION);
                // Check for the freshest data.
                getContentResolver().takePersistableUriPermission(treeUri, takeFlags);

            }
        }
    }
}

现在,保存保存文件夹到共享首选项,而不是每次你想保存图像时都要求用户选择文件夹。

你应该使用DocumentFile类来保存你的图像,而不是File或ParcelFileDescriptor,要了解更多信息,你可以检查这个线程保存图像到SD卡压缩(CompressFormat.JPEG, 100, out);方法和DocumentFile类。

您想要保存位图到您选择的目录。我做了一个ImageWorker库,使用户能够加载,保存和转换位图/绘图/base64图像。

最小SDK - 14

先决条件

保存文件需要WRITE_EXTERNAL_STORAGE权限。 检索文件需要READ_EXTERNAL_STORAGE权限。

储蓄位图/ Drawable Base64

ImageWorker.to(context).
    directory("ImageWorker").
    subDirectory("SubDirectory").
    setFileName("Image").
    withExtension(Extension.PNG).
    save(sourceBitmap,85)

加载位图

val bitmap: Bitmap? = ImageWorker.from(context).
    directory("ImageWorker").
    subDirectory("SubDirectory").
    setFileName("Image").
    withExtension(Extension.PNG).
    load()

实现

添加依赖关系

在项目级别Gradle

allprojects {
        repositories {
            ...
            maven { url 'https://jitpack.io' }
        }
    }

在应用级Gradle

dependencies {
            implementation 'com.github.ihimanshurawat:ImageWorker:0.51'
    }

你可以在https://github.com/ihimanshurawat/ImageWorker/blob/master/README.md上阅读更多

一些新设备不保存位图,所以我解释了一点。

请确保您在下面添加了权限

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

并在XML文件夹名为provider_paths.xml下创建一个XML文件

<?xml version="1.0" encoding="utf-8"?>
<paths>
    <external-path
        name="external_files"
        path="." />
</paths>

and in AndroidManifest under <provider android:name="android.support.v4.content.FileProvider" android:authorities="${applicationId}.provider" android:exported="false" android:grantUriPermissions="true"> <meta-data android:name="android.support.FILE_PROVIDER_PATHS" android:resource="@xml/provider_paths"/> </provider> then simply call saveBitmapFile(passYourBitmapHere) public static void saveBitmapFile(Bitmap bitmap) throws IOException { File mediaFile = getOutputMediaFile(); FileOutputStream fileOutputStream = new FileOutputStream(mediaFile); bitmap.compress(Bitmap.CompressFormat.JPEG, getQualityNumber(bitmap), fileOutputStream); fileOutputStream.flush(); fileOutputStream.close(); } where File getOutputMediaFile() { File mediaStorageDir = new File( Environment.getExternalStorageDirectory(), "easyTouchPro"); if (mediaStorageDir.isDirectory()) { // Create a media file name String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss") .format(Calendar.getInstance().getTime()); String mCurrentPath = mediaStorageDir.getPath() + File.separator + "IMG_" + timeStamp + ".jpg"; File mediaFile = new File(mCurrentPath); return mediaFile; } else { /// error handling for PIE devices.. mediaStorageDir.delete(); mediaStorageDir.mkdirs(); galleryAddPic(mediaStorageDir); return (getOutputMediaFile()); } } and other methods public static int getQualityNumber(Bitmap bitmap) { int size = bitmap.getByteCount(); int percentage = 0; if (size > 500000 && size <= 800000) { percentage = 15; } else if (size > 800000 && size <= 1000000) { percentage = 20; } else if (size > 1000000 && size <= 1500000) { percentage = 25; } else if (size > 1500000 && size <= 2500000) { percentage = 27; } else if (size > 2500000 && size <= 3500000) { percentage = 30; } else if (size > 3500000 && size <= 4000000) { percentage = 40; } else if (size > 4000000 && size <= 5000000) { percentage = 50; } else if (size > 5000000) { percentage = 75; } return percentage; } and void galleryAddPic(File f) { Intent mediaScanIntent = new Intent( "android.intent.action.MEDIA_SCANNER_SCAN_FILE"); Uri contentUri = Uri.fromFile(f); mediaScanIntent.setData(contentUri); this.sendBroadcast(mediaScanIntent); }