当将位图保存到磁盘时,实心路径会显示伪影。

5
我有一个应用程序,使用Paths绘制简单的2D几何图形。这些形状都是纯色的,有时不透明度小于255,并且可能带有线条装饰。在绘制几何图形的视图中,绘图方式从未出现过任何问题。但是,当我使用相同的代码绘制位图并将其保存为JPEG(质量为100)或PNG时,输出文件的纯色区域总是存在着相同的伪影。这是一种通常与JPEG压缩相关联的斑驳效果。
视图截图:Activity截图 保存的图像:保存的图像文件 放大伪影:放大伪影 我尝试了以下方法:
- 保存为PNG和JPEG格式 - 开启和关闭抖动和反锯齿 - 提高位图的DPI,并允许位图使用其默认API - 将我用作相机的矩阵应用于几何表示,而不是将其应用于位图的画布 - 全局启用和禁用硬件加速 - 使用第三方库将位图保存为.bmp文件
所有方法都产生了相同的伪影,没有使情况变得更糟或更好。
public class MainActivity extends AppCompatActivity {
Context context;

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    this.context = getApplicationContext();
}

// button OnClick listener
public void saveImage(View view) {
    new saveBitmapToDisk().execute(false);
}

public Bitmap getBitmap() {
    final int bitmapHeight = 600, bitmapWidth = 600;
    Bitmap bitmap = Bitmap.createBitmap(bitmapWidth, bitmapHeight, Bitmap.Config.ARGB_8888);
    Canvas bitmapCanvas = new Canvas(bitmap);

    float[] triangle = new float[6];
    triangle[0] = bitmapWidth / 2;
    triangle[1] = 0;
    triangle[2] = 0;
    triangle[3] = bitmapHeight / 2;
    triangle[4] = bitmapWidth / 2;
    triangle[5] = bitmapHeight / 2;

    Path solidPath = new Path();
    Paint solidPaint = new Paint();
    solidPaint.setStyle(Paint.Style.FILL);

    solidPath.moveTo(triangle[0], triangle[1]);

    for(int i = 2; i < triangle.length; i += 2)
        solidPath.lineTo(triangle[i], triangle[i+1]);

    solidPath.close();

    solidPaint.setColor(Color.GREEN);
    bitmapCanvas.drawPath(solidPath, solidPaint);
    return bitmap;
}

private class saveBitmapToDisk extends AsyncTask<Boolean, Integer, Uri> {
    Boolean toShare;

    @Override
    protected Uri doInBackground(Boolean... shareFile) {
        this.toShare = shareFile[0];
        final String appName = context.getResources().getString(R.string.app_name);
        final String IMAGE_SAVE_DIRECTORY = String.format("/%s/", appName);
        final String fullPath = Environment.getExternalStorageDirectory().getAbsolutePath() + IMAGE_SAVE_DIRECTORY;
        File dir, file;

        try {
            dir = new File(fullPath);
            if (!dir.exists())
                dir.mkdirs();

            OutputStream fOut;

            file = new File(fullPath, String.format("%s.png", appName));

            for (int suffix = 0; file.exists(); suffix++)
                file = new File(fullPath, String.format("%s%03d.png", appName, suffix));

            file.createNewFile();
            fOut = new FileOutputStream(file);

            Bitmap saveBitmap = getBitmap();
            saveBitmap.compress(Bitmap.CompressFormat.PNG, 100, fOut);
            fOut.flush();
            fOut.close();
            MediaStore.Images.Media.insertImage(context.getContentResolver(), file.getAbsolutePath(), file.getName(), file.getName());

        } catch (OutOfMemoryError e) {
            Log.e("MainActivity", "Out of Memory saving bitmap; bitmap is too large");
            return null;
        } catch (Exception e) {
            Log.e("MainActivity", e.getMessage());
            return null;
        }

        return Uri.fromFile(file);
    }

    @Override
    protected void onPostExecute(Uri uri) {
        super.onPostExecute(uri);
        Toast.makeText(context, "Image saved", Toast.LENGTH_SHORT).show();
    }
}
}
3个回答

3
我用PNG测试了你的程序,文件没有出现伪像。 这些伪像是JPEG压缩的结果。
编辑: 需要编辑的行为

MediaStore.Images.Media.insertImage(context.getContentResolver(), file.getAbsolutePath(), file.getName(), file.getName());

一些问题导致了转换为JPEG格式的问题。

正确的保存图片方法是:

ContentValues values = new ContentValues();
values.put(Images.Media.DATE_TAKEN, System.currentTimeMillis());
values.put(Images.Media.MIME_TYPE, "image/png");
values.put(MediaStore.MediaColumns.DATA, file.getAbsolutePath());
context.getContentResolver().insert(Images.Media.EXTERNAL_CONTENT_URI, values);

这是我简化后的测试程序,可以直接发送生成的文件。
public class Test2Activity extends Activity {
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    new saveBitmapToDisk().execute();
  }

  public Bitmap getBitmap() {
    final int bitmapHeight = 600, bitmapWidth = 600;
    Bitmap bitmap = Bitmap.createBitmap(bitmapWidth, bitmapHeight, Bitmap.Config.ARGB_8888);
    Canvas bitmapCanvas = new Canvas(bitmap);

    Paint solidPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
    solidPaint.setStyle(Paint.Style.FILL);
    solidPaint.setColor(Color.RED);
    bitmapCanvas.drawCircle(300, 300, 200, solidPaint);

    return bitmap;
  }

  private class saveBitmapToDisk extends AsyncTask<Void, Void, Uri> {
    Boolean toShare;

    @Override
    protected Uri doInBackground(Void... shareFile) {
      Context context = Test2Activity.this;
      try {
        File file = new File(context.getExternalFilesDir(null), "test.png");
        FileOutputStream fOut = new FileOutputStream(file);

        Bitmap saveBitmap = getBitmap();
        saveBitmap.compress(Bitmap.CompressFormat.PNG, 100, fOut);
        fOut.flush();
        fOut.close();
        return Uri.fromFile(file);
      } catch (OutOfMemoryError e) {
        Log.e("MainActivity", "Out of Memory saving bitmap; bitmap is too large");
        return null;
      } catch (Exception e) {
        Log.e("MainActivity", e.getMessage());
        return null;
      }

    }

    @Override
    protected void onPostExecute(Uri uri) {
      Context context = Test2Activity.this;
      Toast.makeText(context, "Image saved", Toast.LENGTH_SHORT).show();

      final Intent intent = new Intent(android.content.Intent.ACTION_SEND);
      intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
      intent.putExtra(Intent.EXTRA_STREAM, uri);
      intent.setType("image/png");
      Test2Activity.this.startActivity(intent);
    }
  }
}

我刚在一台新设备上进行了双重检查,我肯定仍然遇到了伪影问题,而且它绝对不是旧文件。也不是缩略图;进入“信息”部分会显示正确的尺寸。这可能是构建环境中的某些问题吗? - Bob Liberatore
@项目 我在回复中添加了一个简化的活动,你应该测试一下。它在透明背景上创建了一个圆形图像,并打开共享意图以发送此文件,请在您的设备上运行它,并通过电子邮件将生成的图像发送给自己。如果图像具有透明背景,则编码为PNG,如果不透明,则编码为JPEG。 - yoah
测试了你的示例,它具有透明背景和没有伪影。好奇可能是我保存到的目录或类似的东西...喝完咖啡后会仔细查看。 - Bob Liberatore
1
结果证明问题出现在MediaStore.Images.Media.insertImage(...)这一行代码上。我将其更改为此问题的接受答案并且终于可以工作了。看起来我使用的代码与相机有关。感谢您帮助我解决这个问题;如果您想更新您的回答,我可以将其标记为接受的答案。 - Bob Liberatore
@项目完成。我自己也学到了一些新东西。 - yoah

1

像这样的瑕疵是JPEG压缩的自然而不可避免的后果。

它们不应该出现在PNG压缩中。如果您在创建PNG文件时出现了这样的瑕疵,我敢打赌您根本没有创建PNG流,而是在具有PNG扩展名的文件中创建了JPEG流。没有一个好的解码器会依赖于文件扩展名。


这种情况发生在saveBitmap.compress(Bitmap.CompressFormat.JPEG,100,fOut);和saveBitmap.compress(Bitmap.CompressFormat.PNG,100,fOut);两种情况下。 - Bob Liberatore
更明确地说,我很长一段时间以来都将其命名为CompressFormat.PNG,其中我得到的图像是这样的。这里之所以是CompressFormat.JPEG,仅仅是因为我正在尝试改变代码的某些内容,看看是否会得到不同的结果。 - Bob Liberatore

0

我在你的代码中注意到了两件事:

1)你保存的文件名是String.format("%s.jpg", appName)String.format("%s%03d.png", appName, suffix),与实际编码无关。

2)你保存的位图的密度由prefs.saveImageDensity().get()决定,因此它可能与屏幕上看到的位图的实际密度不同。

也许你被1)搞混了,或者2)导致了你看到的压缩伪影?


  1. 我编辑帖子的函数可能有误。
  2. 我应该表述得更清楚;我已经尝试过各种密度,包括本机密度。我刚刚回去确认文件扩展名为PNG,并注释掉设置DPI的部分。问题仍然存在。
- Bob Liberatore
文档甚至提到保存为PNG是无损操作,但也许存在一个错误。你应该尝试编写一个单独的测试程序,加载和保存位图,并查看是否出现问题。这将显示保存为PNG是否真正无损。 - hkBst
我用 Android API 替换了使用 这个库 保存为位图,但是我得到了相同的伪影。因此,它似乎与压缩无关,而与实际绘制位图有关。这让我感到非常奇怪。 - Bob Liberatore
只是一个更新:我一直在排除我使用绘制方法时是否有任何奇怪的事情。除了调用canvasView.cluster.drawCached(bitmapCanvas);,我试过直接在位图上画圆形、正方形和路径。但还是出现了这些奇怪的伪像。所以不确定还有什么可以检查的。看起来我必须创建一个测试项目来解决这个问题。 - Bob Liberatore
我已经编辑了这个问题,并创建了一个活动来在新项目中测试它。从这个活动保存时仍然出现了问题。 - Bob Liberatore

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接