热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Android游戏开发15:详谈OnTouchEvent()触屏事件的性能优化分享

      各位童鞋肯定都知道在模拟器中,我们的鼠标当点击一次模拟器屏幕然后释放,先触发ACTION_DOWN然后ACT

       各位童鞋肯定都知道在模拟器中,我们的鼠标当点击一次模拟器屏幕然后释放,先触发 ACTION_DOWN 然后 ACTION_UP ;如果是在屏幕上移动那么才会触发 ACTION_MOVE 的动作;OK,很对。但是你要知道,这只是模拟器!

       真机与模拟器的区别

       当我们的用户在玩我们的游戏的时候,尤其是RPG这种类型的,用户肯定需要会长时间的去触屏按我们的虚拟按键,比如我们会在屏幕上画上一个虚拟方向盘类似这样子~那么其实 ACTION_MOVE 这个事件会被Android一直在响应!

       为什么会一直响应 ACTION_MOVE 这个动作呢? 如果用户没有移动手指而是静止不动也会一直响应?

       原因有两点:第一点是因为,Android 对于触屏事件很敏感!第二点,虽然我们的手指感觉是静止没有移动,其实事实不是如此!当我们的手指触摸到手机屏幕上之后,感觉静止没动,其实手指在不停的微颤抖震动。不信你试试静止下手指,是不是微微动?嘿嘿~

       so~ 我们就要分析了,如果ACTION_MOVE此时间一直被Android os 一直不停的响应并处理,无疑对我们游戏的性能增加了不少的负担!

       比如我们游戏线程绘图时间每次用了100ms,那么当手指触摸屏幕,这短暂的0.1秒内大概会产生10个左右的MotionEvent ,并且系统会尽可能快的把这些event发给监听线程, 这样的话在这一段时间内cpu就会忙于处理onTouchEvent从而杯具点的话会造成画面一卡一卡的。

       那么我们其实根本用不着按键响应这么多次,而是需要在我们每次绘图后,或者绘图前接受一次用户按键就OK了,这样能让帧率不至于下降的太厉害不是么?!so~我们要控制这个时间,让他慢下来,随着我们的绘图时间一起来合作~这样就能减不少系统线程的负担。

       实例分析

       下面贴一段代码,一块来看一下:

Java代码
  1. package com.himi;   
  2. import android.app.Activity;   
  3. import android.os.Bundle;   
  4. import android.util.Log;   
  5. import android.view.MotionEvent;   
  6. import android.view.Window;   
  7. import android.view.WindowManager;   
  8. /**  
  9.  * @author Himi  
  10.  */  
  11. public class MainActivity extends Activity {   
  12.     private Object object;   
  13.     private final int TIME = 50;//备注1   
  14.     @Override  
  15.     public void onCreate(Bundle savedInstanceState) {   
  16.         super.onCreate(savedInstanceState);   
  17.         getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);   
  18.         this.requestWindowFeature(Window.FEATURE_NO_TITLE);   
  19.         setContentView(R.layout.main);   
  20.         object = new Object();   
  21.     }   
  22.     @Override  
  23.     public boolean onTouchEvent(MotionEvent event) {   
  24.         if (event.getAction() == MotionEvent.ACTION_DOWN) {   
  25.             Log.v("Himi""ACTION_DOWN");   
  26.         } else if (event.getAction() == MotionEvent.ACTION_UP) {   
  27.             Log.v("Himi""ACTION_UP");   
  28.         } else if (event.getAction() == MotionEvent.ACTION_MOVE) {   
  29.             Log.v("Himi""ACTION_MOVE");   
  30.         }   
  31.         synchronized (object) {//备注2   
  32.             try {   
  33.                 object.wait(TIME);   
  34.             } catch (InterruptedException e) {   
  35.                 // TODO Auto-generated catch block   
  36.                 e.printStackTrace();   
  37.             }   
  38.         }   
  39.         return true;//这里一定要返回true!原因请看【Android游戏开发9】   
  40.     }   
  41. }  

       代码很清晰了,主要就是备注2的地方。

       备注2:

       可能有的童鞋会问为什么不用sleep()的方法,其实如果我们只是想让线程休眠指定时间的话可以用sleep()函数,但是这个没有资源锁的限制。而Object的wait、notify方法通常用在时间不定的条件限制等待,并且必须写在同步代码块中。

       还有童鞋会问,为什么不用当前类的object来使用:this.wait(),而是new 一个object来?这是因为:

       synchronized 中的Object 表示Object 调用wait必须拥有该对象的监视锁,当前我们有了object的锁,就要用object调用wait~

       备注1:

       这里的变量大家都知道其实是我们设置的休眠的时间,那么这里我想拿出来跟大家说下关于这个时间的定值,在上文我也有说过我们的游戏中只要按键跟我们的绘图线程的时间一样即可,当然这里是个我们的理想时间!如果我们游戏中有人物的帧,那么我们可以来根据人物帧数来当成设定这个睡眠时间也是相当合适的,毕竟人物一帧说明逻辑执行了一遍了,呵呵~,这个还是根据大家游戏的情况而定吧。

       注意:Object.wait(long timeout)这个方法也需要慎用!

       原因是因为测试发现:这个睡眠的时间其实比你规定的时间要略微的长一些,不过我们合理控制好时间还是没问题的。

       补充:

       1、看到有童鞋问//备注2这里能不能用this,也就是当前的object,答案是可以的,但是要注意最好不要这样用,原因是如果当其他地方也需要与当前的Object进行同步的话有可能出现死锁情况!用一个新的object的原因也就是可以让代码中该干什么就干什么,互不影响,

       2、//备注2这里其实我们可以对其优化,,毕竟一个Object比较浪费,我们其实只需要一个字节就足够了,so~我们可以这样定义一个:byte[] lock = new byte[0]; 这样可以算是最优了~

-上面就是Android游戏开发15:详谈OnTouchEvent() 触屏事件的性能优化分享全部内容. 如果需要了解更多android开发及android游戏开发内容关注<编程笔记>


推荐阅读
  • Android 渐变圆环加载控件实现
    本文介绍了如何在 Android 中创建一个自定义的渐变圆环加载控件,该控件已在多个知名应用中使用。我们将详细探讨其工作原理和实现方法。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文介绍如何在 Android 中通过代码模拟用户的点击和滑动操作,包括参数说明、事件生成及处理逻辑。详细解析了视图(View)对象、坐标偏移量以及不同类型的滑动方式。 ... [详细]
  • 本文详细介绍了Java中org.neo4j.helpers.collection.Iterators.single()方法的功能、使用场景及代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
  • Explore how Matterverse is redefining the metaverse experience, creating immersive and meaningful virtual environments that foster genuine connections and economic opportunities. ... [详细]
  • Java 类成员初始化顺序与数组创建
    本文探讨了Java中类成员的初始化顺序、静态引入、可变参数以及finalize方法的应用。通过具体的代码示例,详细解释了这些概念及其在实际编程中的使用。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 1.如何在运行状态查看源代码?查看函数的源代码,我们通常会使用IDE来完成。比如在PyCharm中,你可以Ctrl+鼠标点击进入函数的源代码。那如果没有IDE呢?当我们想使用一个函 ... [详细]
  • 本文详细介绍了Java中org.eclipse.ui.forms.widgets.ExpandableComposite类的addExpansionListener()方法,并提供了多个实际代码示例,帮助开发者更好地理解和使用该方法。这些示例来源于多个知名开源项目,具有很高的参考价值。 ... [详细]
  • 深入解析Spring Cloud Ribbon负载均衡机制
    本文详细介绍了Spring Cloud中的Ribbon组件如何实现服务调用的负载均衡。通过分析其工作原理、源码结构及配置方式,帮助读者理解Ribbon在分布式系统中的重要作用。 ... [详细]
  • 本文深入探讨了 Java 中的 Serializable 接口,解释了其实现机制、用途及注意事项,帮助开发者更好地理解和使用序列化功能。 ... [详细]
  • 本文详细介绍了Akka中的BackoffSupervisor机制,探讨其在处理持久化失败和Actor重启时的应用。通过具体示例,展示了如何配置和使用BackoffSupervisor以实现更细粒度的异常处理。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
author-avatar
好天气
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有