一 图片表示原理

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

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

二 加载图片OOM异常

解决办法

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

  1. public class MainActivity extends AppCompatActivity implements View.OnClickListener {
  2.  
  3. ImageView mImageView;
  4.  
  5. @Override
  6. protected void onCreate(Bundle savedInstanceState) {
  7. super.onCreate(savedInstanceState);
  8. setContentView(R.layout.activity_main);
  9.  
  10. findViewById(R.id.load).setOnClickListener(this);
  11. mImageView = findViewById(R.id.image);
  12. }
  13.  
  14. @Override
  15. public void onClick(View view) {
  16. switch (view.getId()) {
  17. case R.id.load:
  18. load();
  19. break;
  20. }
  21. }
  22.  
  23. private void load() {
  24. try {
  25. BitmapFactory.Options option = new BitmapFactory.Options();
  26. option.inJustDecodeBounds = true; //只会解析图片的大小 不会加载图片的内容
  27. BitmapFactory.decodeStream(getAssets().open("big.jpg"), null, option);
  28. // 获取图片的宽高
  29. int width = option.outWidth;
  30. int height = option.outHeight;
  31. // 获取屏幕的宽高
  32. int screenWidth = getScreenWidth();
  33. int screenHeight = getScreenHeight();
  34. // 把图片的宽高和屏幕的宽高进行对比
  35. int scaleX = width / screenWidth;
  36. int scaleY = height / screenHeight;
  37. int scale = scaleX > scaleY ? scaleX : scaleY;
  38. option.inJustDecodeBounds = false; //加载图片的内容
  39. // 如果设置为>1 请求解码器对原始数据进行子采样 例如inSampleSize==4返回图像的宽度/高度是原始图像的1/4
  40. // 任何值<=1都与1相同
  41. option.inSampleSize = scale;
  42. Bitmap bitmap = BitmapFactory.decodeStream(getAssets().open("big.jpg"), null, option);
  43. int byteCount = bitmap.getByteCount();
  44. Log.i("HUANG", "byteCount=" + byteCount);
  45. mImageView.setImageBitmap(bitmap);
  46.  
  47. } catch (IOException e) {
  48. e.printStackTrace();
  49. }
  50. }
  51.  
  52. /** 得到设备屏幕的宽度 (像素) **/
  53. private int getScreenWidth() {
  54. return getResources().getDisplayMetrics().widthPixels;
  55. }
  56.  
  57. /** 得到设备屏幕的高度 (像素) **/
  58. private int getScreenHeight() {
  59. return getResources().getDisplayMetrics().heightPixels;
  60. }
  61.  
  62. }

三 图片处理原理

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

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

Canvas: 画布

Paint: 画笔

Matrix: 图形矩阵 3*3

Bitmap: 要绘制的图片

四 图片的旋转 平移 缩放

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

  1. public class MainActivity extends AppCompatActivity implements View.OnClickListener {
  2.  
  3. ImageView mImageView, mCopyView;
  4. Bitmap mBitmap;
  5.  
  6. @Override
  7. protected void onCreate(Bundle savedInstanceState) {
  8. super.onCreate(savedInstanceState);
  9. setContentView(R.layout.activity_main);
  10.  
  11. findViewById(R.id.change).setOnClickListener(this);
  12. mImageView = findViewById(R.id.image);
  13. mCopyView = findViewById(R.id.copy);
  14. try {
  15. mBitmap = BitmapFactory.decodeStream(getAssets().open("mm.jpg"));
  16. mImageView.setImageBitmap(mBitmap);
  17. } catch (IOException e) {
  18. e.printStackTrace();
  19. }
  20. }
  21.  
  22. @Override
  23. public void onClick(View view) {
  24. switch (view.getId()) {
  25. case R.id.change:
  26. change();
  27. break;
  28. }
  29. }
  30.  
  31. // 图片的旋转 平移 缩放
  32. // 注意: 旋转 平移 缩放 这三种效果在本案例中只能同时存在一种 分别打开注释看效果
  33. private void change() {
  34. if (null == mBitmap) return;
  35. // 新建空白的图片 要和原图的大小一样
  36. Bitmap bitmap = Bitmap.createBitmap(mBitmap.getWidth(), mBitmap.getHeight(), mBitmap.getConfig());
  37. Canvas canvas = new Canvas(bitmap); //画布 传参必须是一个空白的图片 否则报错
  38. Paint paint = new Paint(); //画笔
  39. Matrix matrix = new Matrix(); //矩阵
  40. // 旋转30度 以图片的中心为圆心
  41. matrix.setRotate(30, mBitmap.getWidth() / 2, mBitmap.getHeight() / 2);
  42. // X轴平移80
  43. //matrix.setTranslate(80, 0);
  44. // Y轴缩为原来的0.5
  45. //matrix.setScale(1F, 0.5F, mBitmap.getWidth() / 2, mBitmap.getHeight() / 2);
  46. canvas.drawColor(Color.WHITE); //绘制背景为白色
  47. canvas.drawBitmap(mBitmap, matrix, paint); //绘制图片
  48. mCopyView.setImageBitmap(bitmap);
  49. }
  50.  
  51. }

五 图片的涂鸦操作

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

  1. public class MainActivity extends AppCompatActivity implements View.OnTouchListener {
  2.  
  3. ImageView mImageView;
  4. Bitmap mNewBitmap;
  5. Canvas mCanvas;
  6. Paint mPaint;
  7. Matrix mMatrix;
  8. int mStartX, mStartY; //按下点的坐标
  9.  
  10. @Override
  11. protected void onCreate(Bundle savedInstanceState) {
  12. super.onCreate(savedInstanceState);
  13. setContentView(R.layout.activity_main);
  14.  
  15. mImageView = findViewById(R.id.image);
  16. try {
  17. Bitmap bitmap = BitmapFactory.decodeStream(getAssets().open("mm.jpg"));
  18. // 不能直接在原图上进行绘制 必须新建空白的图片
  19. mNewBitmap = Bitmap.createBitmap(bitmap.getWidth(), bitmap.getHeight(), bitmap.getConfig());
  20. mCanvas = new Canvas(mNewBitmap);
  21. mPaint = new Paint();
  22. mPaint.setColor(Color.YELLOW);
  23. mMatrix = new Matrix();
  24. // 把原图绘制在空白的图片上
  25. mCanvas.drawBitmap(bitmap, mMatrix, mPaint);
  26. mImageView.setImageBitmap(mNewBitmap);
  27. mImageView.setOnTouchListener(this); //设置触摸监听
  28.  
  29. } catch (IOException e) {
  30. e.printStackTrace();
  31. }
  32. }
  33.  
  34. @Override
  35. public boolean onTouch(View v, MotionEvent event) {
  36. switch (event.getAction()) {
  37. case MotionEvent.ACTION_DOWN: //按下
  38. mStartX = (int) event.getX();
  39. mStartY = (int) event.getY();
  40. break;
  41.  
  42. case MotionEvent.ACTION_MOVE: //移动
  43. // 获取移动点的坐标
  44. int moveX = (int) event.getX();
  45. int moveY = (int) event.getY();
  46. // 画线
  47. mCanvas.drawLine(mStartX, mStartY, moveX, moveY, mPaint);
  48. // 把新图设置给ImageView
  49. mImageView.setImageBitmap(mNewBitmap);
  50. // 把移动点置为开始点
  51. mStartX = moveX;
  52. mStartY = moveY;
  53. break;
  54.  
  55. case MotionEvent.ACTION_UP: //弹起
  56. break;
  57. }
  58. return true; //事件自己来处理
  59. }
  60.  
  61. }

六 图片的颜色处理

图片是有颜色

核心原理就是重绘图片

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

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

  1. public class MainActivity extends AppCompatActivity implements View.OnClickListener {
  2.  
  3. ImageView mImageView;
  4. Bitmap mBitmap, mNewBitmap;
  5. Canvas mCanvas;
  6. Paint mPaint;
  7.  
  8. @Override
  9. protected void onCreate(Bundle savedInstanceState) {
  10. super.onCreate(savedInstanceState);
  11. setContentView(R.layout.activity_main);
  12.  
  13. mImageView = findViewById(R.id.image);
  14. try {
  15. mBitmap = BitmapFactory.decodeStream(getAssets().open("mm.jpg"));
  16. mImageView.setImageBitmap(mBitmap);
  17. mNewBitmap = Bitmap.createBitmap(mBitmap.getWidth(), mBitmap.getHeight(), mBitmap.getConfig());
  18. mCanvas = new Canvas(mNewBitmap);
  19. mPaint = new Paint();
  20. findViewById(R.id.change).setOnClickListener(this);
  21.  
  22. } catch (IOException e) {
  23. e.printStackTrace();
  24. }
  25. }
  26.  
  27. @Override
  28. public void onClick(View view) {
  29. switch (view.getId()) {
  30. case R.id.change:
  31. int randomR = (int) (Math.random() * 256); //0-255 随机数
  32. int randomG = (int) (Math.random() * 256); //0-255 随机数
  33. int randomB = (int) (Math.random() * 256); //0-255 随机数
  34. int randomA = (int) (Math.random() * 256); //0-255 随机数
  35. float colorR = (255 - randomR) / (float) 255;
  36. float colorG = (255 - randomG) / (float) 255;
  37. float colorB = (255 - randomB) / (float) 255;
  38. float colorA = (255 - randomA) / (float) 255;
  39. Log.i("HUANG", "randomR=" + randomR);
  40. Log.i("HUANG", "randomG=" + randomG);
  41. Log.i("HUANG", "randomB=" + randomB);
  42. Log.i("HUANG", "randomA=" + randomA);
  43. Log.i("HUANG", "colorR=" + colorR);
  44. Log.i("HUANG", "colorG=" + colorG);
  45. Log.i("HUANG", "colorB=" + colorB);
  46. Log.i("HUANG", "colorA=" + colorA);
  47.  
  48. ColorMatrix matrix = new ColorMatrix(); //颜色矩阵 5*4
  49. matrix.set(new float[]{
  50. colorR, 0, 0, 0, 0, //red
  51. 0, colorG, 0, 0, 0, //green
  52. 0, 0, colorB, 0, 0, //blue
  53. 0, 0, 0, colorA, 0 //alpha
  54. });
  55. ColorMatrixColorFilter filter = new ColorMatrixColorFilter(matrix);
  56. mPaint.setColorFilter(filter);
  57. mCanvas.drawBitmap(mBitmap, new Matrix(), mPaint);
  58. mImageView.setImageBitmap(mNewBitmap);
  59. break;
  60. }
  61. }
  62.  
  63. }

七 内存泄漏和内存溢出

内存泄漏(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的更多相关文章

  1. [翻译]开发文档:android Bitmap的高效使用

    内容概述 本文内容来自开发文档"Traning > Displaying Bitmaps Efficiently",包括大尺寸Bitmap的高效加载,图片的异步加载和数据缓存 ...

  2. Android Bitmap 和 ByteArray的互相转换

    Android Bitmap 和 ByteArray的互相转换 移动平台图像处理,需要将图像传给native处理,如何传递?将bitmap转换成一个 byte[] 方便传递也方便cpp代码直接处理图像 ...

  3. Android——BitMap(位图)相关知识总结贴

    Android中文API(136) —— Bitmap http://www.apkbus.com/android-54644-1-1.html Android 4.0 r1 API—Bitmap(S ...

  4. Android Bitmap 全面解析(四)图片处理效果对比 ...

    对比对象: UIL Volley 官方教程中的方法(此系列教程一里介绍的,ImageLoader的处理方法和官方的差不多) -------------------------------------- ...

  5. 36、Android Bitmap 全面解析

    Android Bitmap 全面解析(一)加载大尺寸图片 http://www.eoeandroid.com/thread-331669-1-1.html Android Bitmap 全面解析(二 ...

  6. Android bitmap图片处理

    一.View转换为Bitmap         在Android中所有的控件都是View的直接子类或者间接子类,通过它们可以组成丰富的UI界面.在窗口显示的时候Android会把这些控件都加载到内存中 ...

  7. Android Bitmap 载入与像素操作

    Android Bitmap 载入与像素操作 一:载入与像素读写 在Android SDK中,图像的像素读写能够通过getPixel与setPixel两个Bitmap的API实现. Bitmap AP ...

  8. android bitmap compress(图片压缩)

    android bitmap compress android的照相功能随着手机硬件的发展,变得越来越强大,能够找出很高分辨率的图片. 有些场景中,需要照相并且上传到服务,但是由于图片的大小太大,那么 ...

  9. Android Bitmap 常见的几个操作:缩放,裁剪,旋转,偏移

    Android Bitmap 相关操作 常见的几个操作:缩放,裁剪,旋转,偏移      很多操作需要 Matrix 来支持:Matrix 通过矩阵来处理位图,计算出各个像素点的位置,从而把bitma ...

  10. Android Bitmap变迁与原理解析(4.x-8.x)

    App开发不可避免的要和图片打交道,由于其占用内存非常大,管理不当很容易导致内存不足,最后OOM,图片的背后其实是Bitmap,它是Android中最能吃内存的对象之一,也是很多OOM的元凶,不过,在 ...

随机推荐

  1. PHP Redis 全部操作方法 转载

    PHP Redis 全部操作方法   Classes and methods Usage Class Redis Class RedisException Predefined constants C ...

  2. swift的Hashable

    Conforming to the Hashable Protocol To use your own custom type in a set or as the key type of a dic ...

  3. 【转】 iOS播放视频时候,忽略设备静音按钮

    用户有时会在静音模式下观看视频,如果不主动设置的话,视频是没有声音的,通过AVAudioSession可开启以视频为主导的播放模式, 首先需要导入,AVFoundtion.framework,在控制播 ...

  4. ceph 分布式存储安装

    [root@localhost ~]# rm -rf /etc/yum.repos.d/*.repo 下载阿里云的base源 [root@localhost ~]# wget -O /etc/yum. ...

  5. virtualbox+vagrant学习-2(command cli)-2-vagrant cloud命令--有问题

    Cloud https://www.vagrantup.com/docs/cli/cloud.html 命令: vagrant cloud 这是用来管理与vagrant相关的任何东西的命令. 该命令的 ...

  6. Jenkins构建Python项目失败

     Console Output 提示:'Python' 不是内部或外部命令,也不是可运行的程序 定位原因:python.exe 不在jenkins执行用户的PATH里面 解决:构建的时候Python命 ...

  7. git 代码分支合并merge提交新修改远程以及本地分支

    第一步:创建本地分支 点击右键选择TortoiseGit,选择Create Branch…,在Branch框中填写新分支的名称(若选中”switch to new branch”则直接转到新分支上,省 ...

  8. 移动端适配之sprite雪碧图背景定位

    移动端适配一般我会使用rem进行适配,大致操作就是按照一定尺寸设计稿进行制作,最后将所有px值转换为rem,但是一些手机背景精灵图(cssSprite)就会出现一些误差(1px左右),如果公司要求不高 ...

  9. P1437 [HNOI2004]敲砖块

    题目描述 在一个凹槽中放置了 n 层砖块.最上面的一层有n 块砖,从上到下每层依次减少一块砖.每块砖 都有一个分值,敲掉这块砖就能得到相应的分值,如下图所示. 14 15 4 3 23 33 33 7 ...

  10. 第7章 使用寄存器点亮LED灯—零死角玩转STM32-F429系列

    第7章     使用寄存器点亮LED灯 全套200集视频教程和1000页PDF教程请到秉火论坛下载:www.firebbs.cn 野火视频教程优酷观看网址:http://i.youku.com/fir ...