详解Android Bitmap的使用

一 图片表示原理

图片是由每个像素点来组成 像素点就是小方块

图片的大小等于 宽*高*每个像素点的大小

二 加载图片OOM异常

解决办法

其中big.jpg是一张21.2MB的高清图

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

  ImageView mImageView;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    findViewById(R.id.load).setOnClickListener(this);
    mImageView = findViewById(R.id.image);
  }

  @Override
  public void onClick(View view) {
    switch (view.getId()) {
      case R.id.load:
        load();
        break;
    }
  }

  private void load() {
    try {
      BitmapFactory.Options option = new BitmapFactory.Options();
      option.inJustDecodeBounds = true; //只会解析图片的大小 不会加载图片的内容
      BitmapFactory.decodeStream(getAssets().open("big.jpg"), null, option);
      // 获取图片的宽高
      int width = option.outWidth;
      int height = option.outHeight;
      // 获取屏幕的宽高
      int screenWidth = getScreenWidth();
      int screenHeight = getScreenHeight();
      // 把图片的宽高和屏幕的宽高进行对比
      int scaleX = width / screenWidth;
      int scaleY = height / screenHeight;
      int scale = scaleX > scaleY ? scaleX : scaleY;
      option.inJustDecodeBounds = false; //加载图片的内容
      // 如果设置为>1 请求解码器对原始数据进行子采样 例如inSampleSize==4返回图像的宽度/高度是原始图像的1/4
      // 任何值<=1都与1相同
      option.inSampleSize = scale;
      Bitmap bitmap = BitmapFactory.decodeStream(getAssets().open("big.jpg"), null, option);
      int byteCount = bitmap.getByteCount();
      Log.i("HUANG", "byteCount=" + byteCount);
      mImageView.setImageBitmap(bitmap);

    } catch (IOException e) {
      e.printStackTrace();
    }
  }

  /** 得到设备屏幕的宽度 (像素) **/
  private int getScreenWidth() {
    return getResources().getDisplayMetrics().widthPixels;
  }

  /** 得到设备屏幕的高度 (像素) **/
  private int getScreenHeight() {
    return getResources().getDisplayMetrics().heightPixels;
  }

}

三 图片处理原理

Android里面所有的显示效果都是绘制出来的

用Android封装好的绘图类去绘制图片

Canvas: 画布

Paint: 画笔

Matrix: 图形矩阵 3*3

Bitmap: 要绘制的图片

四 图片的旋转 平移 缩放

其中mm.jpg是一张57KB的图 属于正常范围 不需要额外处理

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

  ImageView mImageView, mCopyView;
  Bitmap mBitmap;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    findViewById(R.id.change).setOnClickListener(this);
    mImageView = findViewById(R.id.image);
    mCopyView = findViewById(R.id.copy);
    try {
      mBitmap = BitmapFactory.decodeStream(getAssets().open("mm.jpg"));
      mImageView.setImageBitmap(mBitmap);
    } catch (IOException e) {
      e.printStackTrace();
    }
  }

  @Override
  public void onClick(View view) {
    switch (view.getId()) {
      case R.id.change:
        change();
        break;
    }
  }

  // 图片的旋转 平移 缩放
  // 注意: 旋转 平移 缩放 这三种效果在本案例中只能同时存在一种 分别打开注释看效果
  private void change() {
    if (null == mBitmap) return;
    // 新建空白的图片 要和原图的大小一样
    Bitmap bitmap = Bitmap.createBitmap(mBitmap.getWidth(), mBitmap.getHeight(), mBitmap.getConfig());
    Canvas canvas = new Canvas(bitmap); //画布 传参必须是一个空白的图片 否则报错
    Paint paint = new Paint(); //画笔
    Matrix matrix = new Matrix(); //矩阵
    // 旋转30度 以图片的中心为圆心
    matrix.setRotate(30, mBitmap.getWidth() / 2, mBitmap.getHeight() / 2);
    // X轴平移80
    //matrix.setTranslate(80, 0);
    // Y轴缩为原来的0.5
    //matrix.setScale(1F, 0.5F, mBitmap.getWidth() / 2, mBitmap.getHeight() / 2);
    canvas.drawColor(Color.WHITE); //绘制背景为白色
    canvas.drawBitmap(mBitmap, matrix, paint); //绘制图片
    mCopyView.setImageBitmap(bitmap);
  }

}

五 图片的涂鸦操作

其中mm.jpg是一张57KB的图 属于正常范围 不需要额外处理

public class MainActivity extends AppCompatActivity implements View.OnTouchListener {

  ImageView mImageView;
  Bitmap mNewBitmap;
  Canvas mCanvas;
  Paint mPaint;
  Matrix mMatrix;
  int mStartX, mStartY; //按下点的坐标

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    mImageView = findViewById(R.id.image);
    try {
      Bitmap bitmap = BitmapFactory.decodeStream(getAssets().open("mm.jpg"));
      // 不能直接在原图上进行绘制 必须新建空白的图片
      mNewBitmap = Bitmap.createBitmap(bitmap.getWidth(), bitmap.getHeight(), bitmap.getConfig());
      mCanvas = new Canvas(mNewBitmap);
      mPaint = new Paint();
      mPaint.setColor(Color.YELLOW);
      mMatrix = new Matrix();
      // 把原图绘制在空白的图片上
      mCanvas.drawBitmap(bitmap, mMatrix, mPaint);
      mImageView.setImageBitmap(mNewBitmap);
      mImageView.setOnTouchListener(this); //设置触摸监听

    } catch (IOException e) {
      e.printStackTrace();
    }
  }

  @Override
  public boolean onTouch(View v, MotionEvent event) {
    switch (event.getAction()) {
      case MotionEvent.ACTION_DOWN: //按下
        mStartX = (int) event.getX();
        mStartY = (int) event.getY();
        break;

      case MotionEvent.ACTION_MOVE: //移动
        // 获取移动点的坐标
        int moveX = (int) event.getX();
        int moveY = (int) event.getY();
        // 画线
        mCanvas.drawLine(mStartX, mStartY, moveX, moveY, mPaint);
        // 把新图设置给ImageView
        mImageView.setImageBitmap(mNewBitmap);
        // 把移动点置为开始点
        mStartX = moveX;
        mStartY = moveY;
        break;

      case MotionEvent.ACTION_UP: //弹起
        break;
    }
    return true; //事件自己来处理
  }

}

六 图片的颜色处理

图片是有颜色

核心原理就是重绘图片

改变图片的颜色就是对画笔进行操

其中mm.jpg是一张57KB的图 属于正常范围 不需要额外处理

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

  ImageView mImageView;
  Bitmap mBitmap, mNewBitmap;
  Canvas mCanvas;
  Paint mPaint;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    mImageView = findViewById(R.id.image);
    try {
      mBitmap = BitmapFactory.decodeStream(getAssets().open("mm.jpg"));
      mImageView.setImageBitmap(mBitmap);
      mNewBitmap = Bitmap.createBitmap(mBitmap.getWidth(), mBitmap.getHeight(), mBitmap.getConfig());
      mCanvas = new Canvas(mNewBitmap);
      mPaint = new Paint();
      findViewById(R.id.change).setOnClickListener(this);

    } catch (IOException e) {
      e.printStackTrace();
    }
  }

  @Override
  public void onClick(View view) {
    switch (view.getId()) {
      case R.id.change:
        int randomR = (int) (Math.random() * 256); //0-255 随机数
        int randomG = (int) (Math.random() * 256); //0-255 随机数
        int randomB = (int) (Math.random() * 256); //0-255 随机数
        int randomA = (int) (Math.random() * 256); //0-255 随机数
        float colorR = (255 - randomR) / (float) 255;
        float colorG = (255 - randomG) / (float) 255;
        float colorB = (255 - randomB) / (float) 255;
        float colorA = (255 - randomA) / (float) 255;
        Log.i("HUANG", "randomR=" + randomR);
        Log.i("HUANG", "randomG=" + randomG);
        Log.i("HUANG", "randomB=" + randomB);
        Log.i("HUANG", "randomA=" + randomA);
        Log.i("HUANG", "colorR=" + colorR);
        Log.i("HUANG", "colorG=" + colorG);
        Log.i("HUANG", "colorB=" + colorB);
        Log.i("HUANG", "colorA=" + colorA);

        ColorMatrix matrix = new ColorMatrix(); //颜色矩阵 5*4
        matrix.set(new float[]{
            colorR, 0, 0, 0, 0, //red
            0, colorG, 0, 0, 0, //green
            0, 0, colorB, 0, 0, //blue
            0, 0, 0, colorA, 0 //alpha
        });
        ColorMatrixColorFilter filter = new ColorMatrixColorFilter(matrix);
        mPaint.setColorFilter(filter);
        mCanvas.drawBitmap(mBitmap, new Matrix(), mPaint);
        mImageView.setImageBitmap(mNewBitmap);
        break;
    }
  }

}

七 内存泄漏和内存溢出

内存泄漏(MemoryLeak)

有些对象只有有限的生命周期 当它们的任务完成之后 它们将被回收 如果在对象的生命周期本该结束的时候 这个对象还被一系列的引用 这就会导致内存泄漏

随着泄漏的累积 App将消耗完内存 内存泄漏最终会导致内存溢出

内存泄漏的原因

1. 资源对象没关闭(Cursor File...)

2. 没有及时调用recycle()释放不再使用的Bitmap

3. 广播注册没取消

4. ...

神器: LeakCanary 内存泄露检测工具(https://github.com/square/leakcanary)

内存溢出(OutOfMemoryError OOM)

内存溢出是指当对象的内存占用已经超出分配内存的空间大小

内存溢出的原因

1. Bitmap过大

2. 内存泄露导致

3. ...

八 ImageView中scaleType属性值含义

以上就是详解Android Bitmap的使用的详细内容,更多关于Android Bitmap的资料请关注我们其它相关文章!

(0)

相关推荐

  • Android Bitmap的加载与缓存

    Android系统中图片一般用Bitmap对象表示,它支持png,jpg等常见格式.通常情况下图片的体积都比较大,单个应用允许使用的内存又是有限的,所以我们需要采取一些手段减少内存占用并提高加载速度. 1.图片加载 SDK提供了BitmapFactory类供我们加载图片,常用的方法有这么几个: BitmapFactory.decodeFile :从文件加载. BitmapFactory.decodeByteArray :从字节数组加载. BitmapFactory.decodeStream :从

  • Android 实现将Bitmap 保存到本地

    Overview 图片是一个可以使你程序变得比较的美观,所以我们会在我们的软件中使用图片.但是对于图片的操作也是比较的复杂.今天,我们学习一下如是将我们的图片保存到我们的本地. 开发环境 Android Studio 3.6 Android 11 Mac OS 10.15 模拟机 Google Pixel3 API R 然后学习一下如何来完成我们的功能 按照国际惯例,我们先来看一下我们的代码: /** * Bitmap 帮助类之一 */ class BitmapUtils { /** * Sav

  • Android 实现把bitmap图片的某一部分的颜色改成其他颜色

    把bitmap图片的某一部分的颜色改成其他颜色 private Bitmap ChangeBitmap(Bitmap bitmap){ int bitmap_h; int bitmap_w; int mArrayColorLengh; int[] mArrayColor; int count = 0; mArrayColorLengh = bitmap.getWidth() * bitmap.getHeight(); mArrayColor = new int[mArrayColorLengh]

  • Android中Bitmap、File与Uri之间的简单记录

    简介: 感觉Uri .File.bitmap 比较混乱,这里进行记载,方便以后查看.下面话不多说了,来一起看看详细的介绍吧 Bitmap.File与Uri 1.将一个文件路径path转换成File String path ; File file = new File(path) 2.讲一个Uri转换成一个path 以选择一张图片为例: String path = FileTools.getRealPathFromUri(content,uri); //自定义方法在下面 public static

  • Android中Bitmap常见的一些操作:缩放、裁剪、旋转和偏移

    前言 Bitmap相信对各位Android开发者们来说都不陌生,用它可以获取图片信息,进行图片剪切.平移.旋转.缩放等操作,并可以指定格式保存图片文件.本文将对它的一些常见操作进行总结,下面话不多说了,来一起看看详细的介绍吧 Android Bitmap 相关操作 常见的几个操作:缩放,裁剪,旋转,偏移 很多操作需要 Matrix 来支持:Matrix 通过矩阵来处理位图,计算出各个像素点的位置,从而把bitmap显示出来. matrix里有一个3x3的矩阵,用于图像处理: MSCALE_X M

  • Android中的Bitmap序列化失败的解决方法

    之前写了个User类(实现了Serializable接口),类变量里有Bitmap类型的头像图片,Bitmap导致序列化不成功,报 "android.graphics.Bitmap"相关错误 解决方法之一:把Bitmap对象替换成byte数组来表示间接表示图片,在需要Bitmap的时候再讲byte数组转换成Bitmap对象.这是因为byte数组和Bitmap之间的可以转化,实现也比较方便. 附byte数组与Bitmap的相互转换方法: Bitmap转换成byte数组 private b

  • Android BitmapUtils工具类使用详解

    本文实例为大家分享了Android BitmapUtils工具类的具体代码,供大家参考,具体内容如下 public final class BitmapUtils { public static final String TAG = "BitmapUtil"; private static int sShotScreenWidth = 480; private static int sShotScreenHeight = 720; private static int sShotScr

  • android view转Bitmap生成截图的方法

    项目中经常会用到分享的功能,有分享链接也有分享图片,其中分享图片有的需要移动端对屏幕内容进行截取分享,说白了就是将view 转成bitmap 再到图片分享,还有一种情况是将不可见的view 转成bitmap ,这种view是没有直接显示在界面上的,需要我们使用inflate 进行创建的view. 第一种 先看通过 DrawingCache 方法来截取普通的view,获取它的视图(Bitmap). private Bitmap createBitmap(View view) { view.buil

  • android获取图片尺寸的两种方式及bitmap的缩放操作

    我就废话不多说了,大家还是直接看代码吧~ //Uri.parse("file://"+result.getImage().getCompressPath())) String path=uri.getPath(); Log.e("图片路径",path+""); SpannableString spannableString=new SpannableString(path); //方法一:通过uri把图片转化为bitmap的方法 Bitmap b

  • 详解Android Bitmap的常用压缩方式

    一.前言 已经好久没有更新博客,大概有半年了,主要是博主这段时间忙于找工作,Android岗位的工作真的是越来越难找,好不容易在广州找到一家,主要做海外产品,公司研发实力也不错,所以就敲定了三方协议.现在已经在公司实习了一个月多,目前主要是负责公司某个产品的内存优化,刚好就总结了一下Android Bitmap常用的优化方式. Android中的图片是以Bitmap方式存在的,绘制的时候也是Bitmap,直接影响到app运行时的内存,在Android,Bitmap所占用的内存计算公式是:图片长度

随机推荐