Bitmap 是图形类,Android 系统支持的图片格式有 png、jpg、bmp 等。

对位图操作在游戏中是很重要的知识点,比如游戏中需要两张除了大小之外其他完全相同的图,那么如果会对位图进行缩放操作,很容易就节约了一张图片资源;这样既节约了美工的时间,更节约游戏安装包的大小;当然除了缩放,还有很多操作,例如对位图进行旋转、镜像、设置透明度等等操作都会节约很大的资源。

首先创建一个位图实例。位图的实例不能通过 new,如果想通过一张图片资源文件创建一个位图,则要通过位图工厂来索引图片资源文件,从而生成一张位图实例,如下所示:

BitmapFactory.decodeResource(Resources res,int Id)

作用:通过资源文件生成一张位图

第一个参数:资源实例

第二个参数:资源ID

懂得如何通过图片资源创建位图实例后,下面就来详细介绍如何操作位图。创建项目“BitmapProject”,游戏框架为 SurfaceView 游戏框架。具体步骤参照“11.游戏开发基础(SurfaceView 游戏框架、View 和 SurfaceView 的区别)”。

我这个准备了一张图片pic01.png:

修改 MySurfaceView.java ,代码中 bmp 是由图片资源生成一个位图实例。

Bitmap bmp = BitmapFactory.decodeResource(this.getResources(),R.drawable.pic01);

下面讲解位图常用的操作函数:

1.绘制位图

                /**
* 在画布上绘制一张位图
* 第一个参数:位图实例
* 第二、三个参数:位图的X,Y坐标
* 第四个参数:画笔实例
*/
canvas.drawBitmap(bmp,0,0,paint);

代码执行效果如下:

2.旋转位图

                /**
* 旋转画布
* 第一个参数:画布旋转的角度
* 第二、三个参数:画布的旋转点
* 如果旋转的角度大于0,顺时针旋转;旋转的角度小于0,则逆时针旋转
*
* Canvas 中旋转画布还有一个函数:rotate(float degress),参数传入的是旋转画布的角度,此种方法无法设置旋转点,默认旋转点为屏幕中心点。
*/
canvas.rotate(30,bmp.getWidth()/2,bmp.getHeight()/2);
canvas.drawBitmap(bmp,0,0,paint);

代码执行效果如下:

如果想让位图进行旋转,那么通过这种旋转画布的方法即可实现。但是当使用此种实现画布旋转时,需要注意两点:

• 如果希望图片的旋转是以图片中心点进行旋转,那么在使用 rotate 旋转画布函数对画布进行旋转时,其旋转点坐标应该设置为图片的中心点坐标;

• rotate 函数是对整个画布进行旋转操作,也就是意味着,画布上所有绘制的元素都会因画布的旋转而进行对应的旋转。

例如代码修改为:

                canvas.rotate(30,bmp.getWidth()/2,bmp.getHeight()/2);
canvas.drawBitmap(bmp,0,0,paint);
canvas.drawBitmap(bmp,100,0,paint);

当旋转画布后,绘制两个位图,效果图如下所示:

通过图片可以明显看出,第二次绘制的图片也被旋转!当然这并不是想要的效果,那么如果只想对一张位图进行旋转操作该如何实现呢?

Canvas 类中有两个很重要的函数 save() 与 restore():

• save():作用是用于保存当前画布的状态;

• restore():作用是恢复上次保存的画布状态。

这两个函数是配对出现的。 注意 save() 函数出现的次数不能大于 restore() 函数。

对位图旋转代码进行修改:

                //先保持画布的状态
canvas.save();
canvas.rotate(30,bmp.getWidth()/2,bmp.getHeight()/2);
canvas.drawBitmap(bmp,0,0,paint);
//再将画布恢复状态
canvas.restore();
canvas.drawBitmap(bmp,100,0,paint);

项目运行效果如下:

所以,当对画布进行缩放、旋转和位移操作时,为了保证其他绘制的元素不受影响,应该利用 save() 与 restore() 对画布进行适当的保持与恢复操作。

除此之外 还有一种对位图进行旋转的方式,就是利用矩阵 Matrix 来实现,代码如下:

                //首先创建一个矩阵实例
Matrix mx = new Matrix();
//然后对矩阵进行旋转缩放
mx.postRotate(30,bmp.getWidth()/2,bmp.getHeight()/2);
//最后使用画布绘制位图时,将矩阵信息作为参数传入
canvas.drawBitmap(bmp,mx,paint);

效果如下:

Matrix 类的 postRotate 函数与 Canvas 中的 rotate 函数的作用相同,参数表示的函数也都一致。

使用矩阵对位图进行操作时,可以免去对画布的状态保持和恢复,因为矩阵就是针对单独位图进行的操作,所以不会影响画布其他元素的绘制。

3.平移位图

                //创建位图实例
Bitmap bmp = BitmapFactory.decodeResource(this.getResources(), R.drawable.pic01); canvas.save();
/**
* 平移图片
* 第一个参数:在X轴上平移画布的距离
* 第二个参数:在Y轴上平移画布的距离
*/
canvas.translate(100, 0);
canvas.drawBitmap(bmp, 0, 0, paint);
canvas.restore(); canvas.save();
canvas.translate(-90, 280);
canvas.drawBitmap(bmp, 0, 0, paint);
canvas.restore(); canvas.save();
//利用矩阵也可以完成对位图的平移操作
Matrix maT = new Matrix();
maT.postTranslate(380,-90);
canvas.drawBitmap(bmp, maT, paint);
canvas.restore();

代码执行效果如下:

4.缩放位图

                //创建位图实例
Bitmap bmp = BitmapFactory.decodeResource(this.getResources(), R.drawable.pic01); canvas.save();
canvas.drawText("原图:", 0, 20, paint);
canvas.drawBitmap(bmp, 0, 40, paint);
canvas.restore(); canvas.save();
/**
* 对画布进行缩放
* 第一个参数:对画布X轴的缩放比例
* 第二个参数:对画布Y轴的缩放比例
* 此方法无法设置缩放起始点,默认缩放起始点为屏幕的(0,0)点
*/
//缩小到0.5倍
canvas.scale(0.5f, 0.5f);
canvas.drawBitmap(bmp, 600, 40, paint);
canvas.restore(); canvas.save();
/**
* 对画布进行缩放
* 第一个参数:对画布X轴的缩放比例
* 第二个参数:对画布Y轴的缩放比例
* 第三、四个参数:对画布进行缩放的起始点
*/
//放大为2倍
canvas.scale(2f, 2f, 0, 0);
canvas.drawBitmap(bmp, 0, 220, paint);
canvas.restore(); canvas.save();
//,利用矩阵实现对位图的缩放操作,当前为缩小到0.75倍
Matrix ma = new Matrix();
ma.postTranslate(700, 40);
ma.postScale(0.75f, 0.75f);
canvas.drawBitmap(bmp,ma, paint);
canvas.restore();

代码执行效果如下:

5.镜像反转位图

                //创建位图实例
Bitmap bmp = BitmapFactory.decodeResource(this.getResources(), R.drawable.pic01); canvas.save();
canvas.drawText("原图:", 10, 20, paint);
canvas.drawBitmap(bmp, 10, 40, paint);
canvas.restore(); //X 轴镜像
canvas.save();
canvas.drawText("X 轴镜像:", 10, 320, paint);
//当X轴的比例值小于0时,其实是对画布进行X轴的镜像后的缩放。
canvas.scale(-1, 1, 10+bmp.getWidth()/2, 340+bmp.getHeight()/2);
canvas.drawBitmap(bmp, 10, 340,paint);
canvas.restore(); //Y 轴镜像
canvas.save();
canvas.drawText("Y 轴镜像:", 10, 620, paint);
//利用矩阵实现对位图的镜像操作
Matrix ma = new Matrix();
ma.postTranslate(10, 640);
ma.postScale(1, -1, bmp.getWidth()/2+10,bmp.getHeight()/2+640);
canvas.drawBitmap(bmp,ma, paint);
canvas.restore(); //XY 轴镜像
canvas.save();
canvas.drawText("XY轴镜像:", 10, 920, paint);
canvas.scale(-1, -1, 10+bmp.getWidth()/2, 940+bmp.getHeight()/2);
canvas.drawBitmap(bmp, 10, 940,paint);
canvas.restore();

代码执行效果如下:

在游戏开发中,一般最经常用的是 png 格式的图片,原因在于 png 格式的图片支持透明度,当然这不是必须的。

【读书笔记《Android游戏编程之从零开始》】14.游戏开发基础(Bitmap 位图的渲染与操作)的更多相关文章

  1. Windows游戏编程之从零开始d

    Windows游戏编程之从零开始d I'm back~~恩,几个月不见,大家还好吗? 这段时间真的好多童鞋在博客里留言说或者发邮件说浅墨你回来继续更新博客吧. woxiangnifrr童鞋说每天都在来 ...

  2. Java并发编程的艺术读书笔记(2)-并发编程模型

    title: Java并发编程的艺术读书笔记(2)-并发编程模型 date: 2017-05-05 23:37:20 tags: ['多线程','并发'] categories: 读书笔记 --- 1 ...

  3. Java并发编程的艺术读书笔记(1)-并发编程的挑战

    title: Java并发编程的艺术读书笔记(1)-并发编程的挑战 date: 2017-05-03 23:28:45 tags: ['多线程','并发'] categories: 读书笔记 --- ...

  4. 读书笔记--Android Gradle权威指南(下)

    前言 最近看了一本书<Android Gradle 权威指南>,收获挺多,就想着来记录一些读书笔记,方便后续查阅. 本篇内容是基于上一篇:读书笔记--Android Gradle权威指南( ...

  5. 《Essential C++》读书笔记 之 C++编程基础

    <Essential C++>读书笔记 之 C++编程基础 2014-07-03 1.1 如何撰写C++程序 头文件 命名空间 1.2 对象的定义与初始化 1.3 撰写表达式 运算符的优先 ...

  6. 【读书笔记《Android游戏编程之从零开始》】18.游戏开发基础(碰撞检测)

    1.矩形碰撞 所谓矩形碰撞,就是利用两个矩形之间的位置关系来进行判断,如果矩形的像素在另外一个矩形之中,或者之上都可以认为这两个矩形发生了碰撞. 如果单纯的去考虑哪些情况会判定两个矩形发生碰撞,倒不如 ...

  7. 【读书笔记《Android游戏编程之从零开始》】19.游戏开发基础(游戏音乐与音效)

    在一款游戏中,除了华丽的界面 UI 直接吸引玩家外,另外重要的就是游戏的背景音乐与音效:合适的背景音乐以及精彩的音效搭配会令整个游戏上升一个档次. 在 Android 中.常用于播放游戏背景音乐的类是 ...

  8. 【读书笔记《Android游戏编程之从零开始》】16.游戏开发基础(动画)

    1. Animation动画   在Android 中,系统提供了动画类 Animation ,其中又分为四种动画效果: ● AlphaAnimation:透明度渐变动画 ● ScaleAnimati ...

  9. 【读书笔记《Android游戏编程之从零开始》】12.游戏开发基础(Canvas 画布)

    1.Canvas 画布 画布类 Canvas 封装了图形和图片绘制等内容,此类常用的函数说明如下: drawColor(int color) 作用:绘制颜色覆盖画布,常用于刷屏 参数:颜色值,也可用十 ...

随机推荐

  1. android 6.0 httpclient

    Apache HTTP Client RemovalAndroid 6.0 release removes support for the Apache HTTP client. If your ap ...

  2. Scala on Visual Studio Code

    Download and install Scala Download a scala installation package from here. Then install it. Linux s ...

  3. PHP控制前台弹出对话框

    应用场景: 微信授权登录过程中,需要用户确认,故衍生此需求: 相应的逻辑不放在前端的原因是,此部分逻辑属于偏功能业务,所以放在后端,方便统一管理. 解决办法: 通过php echo出javascrip ...

  4. 初学Node(六)搭建一个简单的服务器

    搭建一个简单的服务器 通过下面的代码可以搭建一个简单的服务器: var http = require("http"); http.createServer(function(req ...

  5. MSCRM 2015 新功能(一)

    MS官网信息:http://www.microsoft.com/en-us/dynamics/crm-customer-center/what-s-new.aspx MS官网中提到的新功能主要以下几块 ...

  6. SharePoint 2013 手动删除爬网项目

    本文介绍如何手动删除某些搜索项目,其实删除搜索项目并不常用,主要还是在刚刚完成爬网,就删除了某些项目,然后有比较敏感需要马上删除的时候.下面,就跟着图文简单了解下手动删除已爬网的项目吧. 1.配置好搜 ...

  7. SharePoint 2010 文档管理系列

    前言,这是自己第一次写一个系列的文档,本来想使用SharePoint 2013版本,但是碍于SharePoint 2013对于硬件要求过高,自己的笔记本无法承受,所以退而求其次选择了在SharePoi ...

  8. 浅谈PopupWindow弹出菜单

    实现将一个View显示在某一位置,而且是浮于当前窗口 首先要有一个要显示的view的布局,可以是任意View,包括ViewGroup <?xml version="1.0" ...

  9. Modernizr的介绍和使用

    传统浏览器目前不会被完全取代,令你难以将最新的 CSS3 或 HTML5 功能嵌入你的网站. Modernizr 正是为解决这一难题应运而生,作为一个开源的 JavaScript 库,Moderniz ...

  10. Visual Studio发布Web项目报错:Unable to add 'xxx' to the Web site. Unable to add file 'xxx'. The specified file could not be encrypted.

    背景 Visual Studio下的Web项目 现象 发布时遇到Unable to add 'xxx' to the Web site.  Unable to add file 'xxx'. The ...