Android_(游戏)打飞机01:前言
(游戏)打飞机01:前言 传送门
(游戏)打飞机02:游戏背景滚动 传送门
(游戏)打飞机03:控制玩家飞机 传送门
(游戏)打飞机04:绘画敌机、添加子弹 传送门
(游戏)打飞机05:处理子弹,击中敌机,添加计分板 传送门
(游戏)打飞机06:后续 传送门
程序已放到Github上托管 : 传送门
打飞机游戏效果
游戏中的二级缓存
android之surfaceView学习:传送门
制作游戏时,为了时刻渲染游戏场景,涉及到一个实时画图的问题了,对于实时更新UI这个问题,android的UI更新都需要在主线程中更新,但是如果将一个实时绘图的操作放在主线程,必定会出现阻塞主线程的问题,即便是不阻塞主线程,也会降低程序运行的速度
surfaceView提供了UI线程。可以自己更新UI,因此,这样我们在surfaceView中进行实时的绘画,然后通过更改其中的绘画的数据,既可以实现我们想要的实时的更新UI的这个问题了,并且消耗较小的资源
surfaceCreated:创建时需要执行的操作
surfaceView:大小改变时需要执行的操作
surfaceDestroyed:销毁时进行的操作
//视图创建时通知
public void surfaceCreated(SurfaceHolder holder) {
this.holder = holder;
runState = true;
//视图创建时开始线程
new Thread(this).start();
} //界面发生改变的时候通知
public void surfaceChanged(SurfaceHolder holder, int i, int i1, int i2) { } //销毁时通知
public void surfaceDestroyed(SurfaceHolder holder) {
runState = false;
}
线程中的方法
public void run() {
Random ran = new Random(); try{
while(true) {
//获得绘画的画布
Canvas canvas = holder.lockCanvas(); Paint p = new Paint();
p.setColor(Color.rgb(ran.nextInt(255),ran.nextInt(255),ran.nextInt(255))); canvas.drawLine(ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),p); //把绘画好的内容提交上去
holder.unlockCanvasAndPost(canvas);//解锁
Thread.sleep(1000);
}
}catch (Exception e){ }
}
锁住画布时,开始编辑要绘画的内容
//锁住画布
Canvas canvas = holder.lockCanvas();
内容绘画好后,开始解锁
//把绘画好的内容提交上去
holder.unlockCanvasAndPost(canvas);//解锁
制作游戏时,为了时时渲染游戏画面,线程中常用死循环
while(true) {
//锁住画布
Canvas canvas = holder.lockCanvas(); Paint p = new Paint();
p.setColor(Color.rgb(ran.nextInt(255),ran.nextInt(255),ran.nextInt(255))); canvas.drawLine(ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),p); //把绘画好的内容提交上去
holder.unlockCanvasAndPost(canvas);//解锁
Thread.sleep(1000);
}
实现效果:
package com.example.administrator.myapplication; import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.view.SurfaceHolder;
import android.view.SurfaceView; import java.util.Random; /**
* Created by Administrator on 2018/8/9.
*/ public class GameView extends SurfaceView implements Runnable, SurfaceHolder.Callback{ public GameView(Context context) {
super(context); getHolder().addCallback(this); //注册回调方法
} private boolean runState = false;
private SurfaceHolder holder = null; @Override
public void run() {
Random ran = new Random(); try{
while(true) {
//获得绘画的画布
Canvas canvas = holder.lockCanvas(); Paint p = new Paint();
p.setColor(Color.rgb(ran.nextInt(255),ran.nextInt(255),ran.nextInt(255))); canvas.drawLine(ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),p); //把绘画好的内容提交上去
holder.unlockCanvasAndPost(canvas);//解锁
Thread.sleep(1000);
}
}catch (Exception e){ }
} //视图创建时通知
public void surfaceCreated(SurfaceHolder holder) {
this.holder = holder;
runState = true;
new Thread(this).start(); } //界面发生改变的时候通知
public void surfaceChanged(SurfaceHolder holder, int i, int i1, int i2) { } //销毁时通知
public void surfaceDestroyed(SurfaceHolder holder) {
runState = false;
} }
GameView.java
package com.example.administrator.myapplication; import android.support.v7.app.AppCompatActivity;
import android.os.Bundle; public class MainActivity extends AppCompatActivity{ @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(new GameView(this)); } }
MainActivity.java
随之而来有一个问题。产生新线条的时候会让原本存在线条也会发生位置、颜色的改变,这样绘制游戏时这样不仅会产生大量的游戏内存,也容易使玩家造成晕眩效果
制作游戏常用二级缓存来解决这个问题(减少游戏运行时内存开销!!!)
二级缓存工作机制
所谓二级缓存实际上并不复杂,当Android端需要获得数据时比如获取网络中的图片,我们首先从内存中查找(按键查找),内存中没有的再从磁盘文件或sqlite中去查找,若磁盘中也没有才通过网络获取;当获得来自网络的数据,就以key-value对的方式先缓存到内存(一级缓存),同时缓存到文件或sqlite中(二级缓存)。注意:内存缓存会造成堆内存泄露,所有一级缓存通常要严格控制缓存的大小,一般控制在系统内存的1/4。
二级缓存
二级缓存:当游戏需要产生新动画时,不改变原场景游戏画面,只要把新加入的绘画内容先放到二级缓存中,绘画好后从二级缓存取出来绘画到游戏界面上
运用Bitmap实现二级缓存
private Bitmap gameBitmap = null; gameBitmap = Bitmap.createBitmap(500,500, Bitmap.Config.ARGB_8888);
public void run() {
Random ran = new Random(); try{
while(true) {
//获得绘画的画布
Canvas canvas = holder.lockCanvas();
Paint p = new Paint();
Canvas c = new Canvas(gameBitmap); // p.setColor(Color.WHITE);
// c.drawRect(new Rect(0,0,500,500),p); //白色背景 p.setColor(Color.rgb(ran.nextInt(255),ran.nextInt(255),ran.nextInt(255))); c.drawLine(ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),p); canvas.drawBitmap(gameBitmap,0,0,new Paint()); //把绘画好的内容提交上去
holder.unlockCanvasAndPost(canvas);//解锁
Thread.sleep(1000);
}
}catch (Exception e){ }
实现效果:
产生新线条时,原本绘画线条不会发生改变
二级缓存加载图片
package com.example.administrator.myapplication; import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Rect;
import android.view.SurfaceHolder;
import android.view.SurfaceView; import java.util.ArrayList;
import java.util.List;
import java.util.Random; /**
* Created by Administrator on 2018/8/9.
*/ public class GameView extends SurfaceView implements Runnable, SurfaceHolder.Callback{ public GameView(Context context) {
super(context); getHolder().addCallback(this); //注册回调方法 gameBitmap = Bitmap.createBitmap(1200,1024, Bitmap.Config.ARGB_8888); bitmaps.add(BitmapFactory.decodeResource(getResources(),R.drawable.a3));
bitmaps.add(BitmapFactory.decodeResource(getResources(),R.drawable.a4));
bitmaps.add(BitmapFactory.decodeResource(getResources(),R.drawable.a5));
} private List<Bitmap> bitmaps = new ArrayList<Bitmap>(); private boolean runState = false;
private SurfaceHolder holder = null;
private Bitmap gameBitmap = null; @Override
public void run() {
Random ran = new Random();
int index = 0;
try{
while(true) {
//获得绘画的画布
Canvas canvas = holder.lockCanvas();
Paint p = new Paint();
Canvas c = new Canvas(gameBitmap); c.drawBitmap( bitmaps.get(index++),0,0,p); if(index==bitmaps.size()){
index=0;
}
// p.setColor(Color.WHITE);
// c.drawRect(new Rect(0,0,500,500),p); //白色背景
// p.setColor(Color.rgb(ran.nextInt(255),ran.nextInt(255),ran.nextInt(255)));
//c.drawLine(ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),p); canvas.drawBitmap(gameBitmap,0,0,new Paint()); //把绘画好的内容提交上去
holder.unlockCanvasAndPost(canvas);//解锁
Thread.sleep(1000);
}
}catch (Exception e){ }
} //视图创建时通知
public void surfaceCreated(SurfaceHolder holder) {
this.holder = holder;
runState = true;
new Thread(this).start(); } //界面发生改变的时候通知
public void surfaceChanged(SurfaceHolder holder, int i, int i1, int i2) { } //销毁时通知
public void surfaceDestroyed(SurfaceHolder holder) {
runState = false;
} }
GameView.java
package com.example.administrator.myapplication; import android.support.v7.app.AppCompatActivity;
import android.os.Bundle; public class MainActivity extends AppCompatActivity{ @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(new GameView(this)); } }
MainActivity.java
添加图片进二级缓存中
private List<Bitmap> bitmaps = new ArrayList<Bitmap>();
bitmaps.add(BitmapFactory.decodeResource(getResources(),R.drawable.a3));
bitmaps.add(BitmapFactory.decodeResource(getResources(),R.drawable.a4));
bitmaps.add(BitmapFactory.decodeResource(getResources(),R.drawable.a5));
运行程序时
public void run() {
Random ran = new Random();
int index = 0;
try{
while(true) {
//获得绘画的画布
Canvas canvas = holder.lockCanvas();
Paint p = new Paint();
Canvas c = new Canvas(gameBitmap); c.drawBitmap( bitmaps.get(index++),0,0,p); if(index==bitmaps.size()){
index=0;
}
// p.setColor(Color.WHITE);
// c.drawRect(new Rect(0,0,500,500),p); //白色背景
// p.setColor(Color.rgb(ran.nextInt(255),ran.nextInt(255),ran.nextInt(255)));
//c.drawLine(ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),ran.nextInt(1000),p); canvas.drawBitmap(gameBitmap,0,0,new Paint()); //把绘画好的内容提交上去
holder.unlockCanvasAndPost(canvas);//解锁
Thread.sleep(1000);
}
}catch (Exception e){ }
}
Android_(游戏)打飞机01:前言的更多相关文章
- Android_(游戏)打飞机06:后续
(游戏)打飞机01:前言 传送门 (游戏)打飞机02:游戏背景滚动 传送门 (游戏)打飞机03:控制玩家飞机 传送门 (游戏)打飞机04:绘画敌机.添加子弹 传送门 (游戏)打飞机05:处理子弹, ...
- Android_(游戏)打飞机05:处理子弹,击中敌机,添加计分板
(游戏)打飞机01:前言 传送门 (游戏)打飞机02:游戏背景滚动 传送门 (游戏)打飞机03:控制玩家飞机 传送门 (游戏)打飞机04:绘画敌机.添加子弹 传送门 (游戏)打飞机05:处理子弹, ...
- Android_(游戏)打飞机04:绘画敌机、添加子弹
(游戏)打飞机01:前言 传送门 (游戏)打飞机02:游戏背景滚动 传送门 (游戏)打飞机03:控制玩家飞机 传送门 (游戏)打飞机04:绘画敌机.添加子弹 传送门 (游戏)打飞机05:处理子弹, ...
- Android_(游戏)打飞机03:控制玩家飞机
(游戏)打飞机01:前言 传送门 (游戏)打飞机02:游戏背景滚动 传送门 (游戏)打飞机03:控制玩家飞机 传送门 (游戏)打飞机04:绘画敌机.添加子弹 传送门 (游戏)打飞机05:处理子弹, ...
- Android_(游戏)打飞机02:游戏背景滚动
(游戏)打飞机01:前言 传送门 (游戏)打飞机02:游戏背景滚动 传送门 (游戏)打飞机03:控制玩家飞机 传送门 (游戏)打飞机04:绘画敌机.添加子弹 传送门 (游戏)打飞机05:处理子弹, ...
- Python之游戏开发-飞机大战
Python之游戏开发-飞机大战 想要代码文件,可以加我微信:nickchen121 #!/usr/bin/env python # coding: utf-8 import pygame impor ...
- Python小游戏之 - 飞机大战美女 !
用Python写的"飞机大战美女"小游戏 源代码如下: # coding=utf-8 import os import random import pygame # 用一个常量来存 ...
- 01 前言/基础设施 - DevOps之路
01 前言/基础设施 - DevOps之路 文章Github地址,欢迎start:https://github.com/li-keli/DevOps-WiKi 简介 基础架构采用DevOps设计思想, ...
- 微信小游戏 demo 飞机大战 代码分析(四)(enemy.js, bullet.js, index.js)
微信小游戏 demo 飞机大战 代码分析(四)(enemy.js, bullet.js, index.js) 微信小游戏 demo 飞机大战 代码分析(一)(main.js) 微信小游戏 demo 飞 ...
随机推荐
- 学习python基础规则
前面应该是记流水账的方式,毕竟学习的内容不多无法产出什么有效的内容. 这两天从开始下载Python开始学习,一路顺畅冒的问题,直到开始学习python的游戏规则,严格缩进.注释及‘’的使用等感觉还不错 ...
- Codeforces 1201E2. Knightmare (hard)
传送门 看到棋盘先黑白染色冷静一下 然后分析发现,如果初始时两只马在同色的格子,那么一定是后手吃先手 反之一定是先手吃后手 所以分类讨论一下,如果初始在同色的格子,并且后手到达终点的步数更少,那么后手 ...
- 导入excel-uploadify+npoi
总结: 导入文件时一定要给database设置栏位 虚拟表的栏位名称可以与数据库表不一致,但顺序一定要一致,因为它是按照顺序依次插入的 sheet.FirstRowNum = 0; sheet.Fir ...
- Vue开发——实现吸顶效果
因为项目需求,最近开始转到微信公众号开发,接触到了Vue框架,这个效果的实现虽说是基于Vue框架下实现的,但是同样也可以借鉴到其他地方,原理都是一样的. 进入正题,先看下效果图: 其实js做这个效果还 ...
- Django框架——基础之路由系统(urls.py)11111111
1.URL路由系统前言 URL是Web服务的入口,用户通过浏览器发送过来的任何请求,都是发送到一个指定的URL地址,然后被响应. 在Django项目中编写路由,就是向外暴露我们接收哪些URL的请求,除 ...
- 第二篇 HTML 常用元素及属性值
常用元素及属性值 先和同学了解下,一部分常用的元素,区别以及属性,常用在哪里. 标签是由左右尖括号抱起来的,由开始标签开始,再由结束标签结束,里面内容则是元素,比如:<div>< ...
- Seeker:一款可获取高精度地理和设备信息的工具分析
Seeker是一款可以获取高精度地理和设备信息的工具.其利用HTML5,Javascript,JQuery和PHP来抓取设备信息,以及Geolocation接口实现对设备高精度地理位置的获取. See ...
- 通用mapper将另外一个同名的表生成在同一个实体及mapper中
今天遇见了一个在网上都搜索不到的错误,使用通过mapper生成实体及mapper文件时会将另外一个数据库的同名文件生成在一个实体及mapper中,这样就会造成一个实体和mapper中有两个表的字段,经 ...
- pat (B)_1002
1002 写出这个数 (20 分) 读入一个正整数 n,计算其各位数字之和,用汉语拼音写出和的每一位数字. 输入格式: 每个测试输入包含 1 个测试用例,即给出自然数 n 的值.这里保证 n 小于 1 ...
- 小程序UI设计(2)-符合视觉规范-字体规范
下图是微信小程序官方要求字体规范 根据此要求小程序设计工具定制了符合规范的组件.如下图 工具使用时,将左侧组件拖拽到设计区域即可.字体大小和颜色都是按照规范设置的.在使用时根据微信要求在不同位置摆放即 ...