`
love19820823
  • 浏览: 935289 次
文章分类
社区版块
存档分类
最新评论

Java游戏开发中应始终坚持的10项基本原则

 
阅读更多

文章关键字:Java 优化 效率 AWT SWT SWING 精简 绘图 监听 游戏开发 GC DRAW

关于文章中涉及的两个杜撰概念:

一、绘图器:众所周知,Java GUI以paint进行绘图,以repaint进行图像刷新,而完成repaint及paint这一连贯过程中所用到绘图组件,我将其称为绘图器。就我个人的体会,绘图器的调用时机应始终处于repaint之后paint之前,即通过repaint触发刷新后执行,当其中的具体逻辑完成其对应的图像绘制后,再通过统一接口将其图像插入paint中,为了匹配需要,绘图器应始终以接口方式实现。

二、监听器:这里所说的监听器,并不是特指某个Listener组件,而是包括Java游戏中所需的所有监听器集合。由于Java游戏中很可能会切换不同的游戏模式,而不同模式游戏中需要处理的鼠标或键盘事件也不尽相同。所以在Java游戏开发中,我们需要一个可替换的监听器集合,用以变更不同游戏模式下的不同监听事件,为了匹配需要,监听器应始终以接口方式实现。

正文,关于Java游戏开发中应始终坚持的10项基本原则:

1、始终保持画布的唯一性。

现实生活中,人类通过口腔及消化道摄取的营养物质可以被心、肝、脾、肺、肾等内脏吸收,却没有人会想给自己的心、肝、脾、肺、肾上也弄个嘴,因为一致性的功能实现只要有一个就足够了。但是,有时我们不经意的在游戏中add、remove不同panel或canvas以求转换画面的行为,无异于是想给游戏的心、肝、脾、肺、肾上装嘴的不智之举,切忌Java的GUI都是画出来的,重绘就好,没有切换组件的必要,否则费力不讨好。


2、始终以接口方式转换监听及处理图像绘制,务必将视图及逻辑层分开。

针对一些混合类型的游戏,比如SLG+AVG、RPG+STG,我们将面临不同模式下游戏的监听器及绘图器切换问题。

这时最简单的抉择莫过于为每一个游戏类型都订制一个对应的面板进行切换,这样虽表面上方省心,但却也是最费力而不讨好的,且不说闪烁问题需要单独解决,资源占用问题,光冗余代码就够人头痛了。

其次就是在一个面板中针对不同游戏类型使用switch判断以切换监听及绘图,事件数量少时固然可以,效率也不错,但稍微多一点恐怕就不那么简单,更多时则仅余郁闷,同样不建议使用。

就我个人所见,解决这一问题的最好方法莫过于沿用MVC模式,以接口方式构建绘图器及监听器,当游戏出现变更时,我们仅仅需要切换监听及绘图接口,就可以迅速转变游戏内容,而无需区别对待不同的实现,这样即避免了组件切换的闪烁及延迟,也精简了代码,更有利于开发时的模块划分。

3、始终以静态方式加载游戏常用资源,缓存常用对象,并及时释放无用资源。

即使历史发展到今天,Java依旧没有彻底摆脱其系统资源杀手的可憎面目,GC机制也导致我们无法适时地释放资源,new的越多,系统也变得越慢,这对于大量使用图形资源的游戏来讲尤其要命。所以我们要尽一切可能令常用资源静态化为唯一实例以避免反复调用,而将一些调用后不会再使用或很少使用的资源迅速null以等待GC自动回收。否则,你将发现你的游戏距离内存溢出是那样的近……

4、始终以循环方式展开游戏,利用线程控制游戏流程,避免出现僵直现象。

事实上所谓的游戏开发,在某种程度上不过是由程序员制作出的一种夹杂着各类图形算法,用以适时地展示各种资源的幻灯程序;唯一的区别在于,普通幻灯程序中人机交互性较弱,而游戏的人机交互性较强罢了。

我们都知道,幻灯程序在展示中无论如何跳转展示页,也必然有其固定的begin与end页面,而且也势必能重复从头至尾顺序循环其begin与end,以此构成一个幻灯片。

实际上游戏制作也一样,无论游戏流程如何转变,游戏都会有也必然会有一个主流程,或者说一个主循环体,这样我们才能由游戏开始进行到游戏结束,而不是从一个结束到另一个结束,也就是说无论游戏中细节分支有多少,它的主流程处理及判定也必然是顺序的。针对这一特性,决定了我们应将游戏主体代码至于一个大的循环体之内,再利用线程控制循环体中的游戏进度,从而更好的顺应这一流程。简单的说,我们应将循环体中每一个使用到的绘图器都当作于flash中的一桢,而线程的各种控制当作时间轴,用以调节不同桢的播放速度及调用时机,以此完成各种不同的事件交互。

5、始终在处理复杂绘图时直接准备贴图而非由程序绘制。


我们都知道Java绘图事实上是GDI实现,因此其绘制复杂画面的效率也就可想而知。通常强况下,除非当前的效果非编程不能实现,或者其所造成的资源损耗确实微小到可以忽略不计,否则最好的方法就是用空间换效率,准备好图片直接贴上去吧,宁可增加些程序体积,也不要让玩家因等待的愤怒而问候你祖宗八辈。

6、始终保证repaint仅刷新需要部分,避免无谓的全局重绘。

每repaint一次,事实上就是将paint中的图形打印到窗体上一次,窗体越大,处理的图像越复杂,repaint所造成的资源损耗也势必越多,运行效率也势必越低。但反过来说,由于Java允许我们限定repaint的范围,因而我们可以将刷新限定在某一特定区域内,更准确地说我们可以仅在需要变更画面的位置上才进行刷新,以此将损耗降低到最低限度,总体上说,即使我们会因为计算刷新区域额外花费些许时间,总体上讲也比全局repaint要快得多。

7、始终双缓冲游戏图像避免闪烁现象发生。

对于Java绘图而言,每次调用repaint方法时都会清除整个屏幕,然后paint才显示画面。而万一系统速度不够,在清除背景和绘制图像间的短暂间隔内被用户看见,就出现了所谓的闪烁现象;简单来讲闪烁的成因就是运算效率不足,使得repaint与paint不连贯造成的。

针对这种情况,我们需要利用双缓冲技术加以解决。

双缓冲实现其实简单至极,主要过程就是先创建一个等大小于希望绘制图形的Image,而后取得其Graphics,每当paint绘图时我们不直接将图像绘制于paint函数的Graphics上,而是绘制于我们创建的缓冲图像的Graphics上,当绘制完成后再调用paint函数提供的drawImage方法,将整个后台图像一次画到屏幕上去。这种方法的优点在于大部分绘制是在后台进行的。将后台绘制的图像一次绘制到屏幕上。

这时只要系统速度正常,我们所看到的绘图将不再有闪烁现象发生。

8、始终在自绘组件的桌面游戏中应用AWT或SWT而非Swing。

众所周知,Swing(JFC)的GUI是以AWT为基础在本地窗体绘制而成,相较AWT虽然提供了更为丰富的组件,但也意味着它占用了更多的资源。而事实上,大多数Java桌面游戏组件是由开发者所针对性绘制,并非Swing库提供,也不需要Swing库支持,我们完全可以放弃Swing而选择AWT或SWT(SWT绘图与AWT/Swing绘图在方法上略有区别,但本质一样)这种直接Native而来的界面,实在不需劳动Swing他老人家,平白的耗费掉那些本就因使用Java应用而稀缺的系统资源。

9、始终别忘了在Graphics处理完毕后dispose。

Graphics的dispose与数据库Connection的close可谓异曲同工。为此我特意做了一个实验,在死循环中无间隔无优化的反复repaint一幅2000X2000的大图,应用dispose时虽然刷新很慢并伴随闪烁但总体讲正常,而去掉dispose运行大约一分钟后万恶的溢出大神降临……

当然,就像Connection应在全部操作完成后才close一样,Graphics也仅在全部绘图完毕后才需要dispose,也就是当最后一个paint最后一次draw后,别忘了留个dispose关门,除非你很想看见溢出大神……

10、始终以运算效率为第一优先,可适当放弃代码可读性,可适当违背OO原则。

以Java进行游戏开发,最大的问题莫过于系统资源的损耗,在关键问题上,就别死抱着OO不放了。


——————华丽的分割线——————


若你能始终坚持以上十点,虽然别指望就此超越C/C++游戏的运行效率,但已能与Delphi游戏争锋而无愧色,傲视于vb6、Flash、rmxp、rmvx等工具开发的游戏而鄙夷之(-_-||| )。

但说时容易做时难,有些部分笔者也没有做到。惰性使然,诸君切勿学我……

分享到:
评论

相关推荐

    《Java程序设计案例教程》课程标准.doc

    通过本教材的学习,学生将了解有关Java的各种先进技术,掌握网络环境下应 用的开发机制,熟练掌握使用Java开发独立的桌面应用程序;同时了解Java安全应用等 方面的知识,为进一步学习信息安全技术方向的课程打好基础...

    Solid-Design-Principles-in-Java:SOLID设计原则及其在Java 9中的实现

    Java 9中的实体设计原则 SOLID是由Robert C. Martin开发的重要编程实践的集合。 SOLID包含5种编码实践: SRP-单一责任原则OCP-打开/关闭原理LSP-Liskov替代原理ISP-接口隔离原则DIP-依赖倒置原则SRP-单一责任原则SRP...

    几个小小的Java案例

    不只是服务端开发能从中受益,任何Java系统开发都能从Spring的简单、可测试和松耦合特征中得到好处。 简单地说,Spring是一个轻量级的IoC和AOP容器框架。也许这个描述并不简单,但它确实概括出了Spring的功能。为了...

    java 工程师简历

    •技术方面:对Struts、Spring、Hibernate、Log4J、JDom、Memcache、Quartz、jQuery、JSON等技术能熟练使用,尤其是SSH的整和开发,js ajax的高用户体验的效果,项目的框架设计及OO原则的重要性,代码重构与代码的可维护...

    JBob平台系统下载

    平台从设计到开发都坚持简约实用和约定优于配置的原则,和其他许多平台相比,JBob致力于精简的配置、高效的开发模式和稳定的性能,尽可能地减少开发成本,让你集中精力解决客户需求,快乐地开发WEB项目。 用户和...

    practice.in.java:天下武功,唯快不破

    主要代码和核心测试类要有清晰的注解,可解题过程,Java代码编写规范,请参照Java的样板公司,编写规范《 xxxbaba Java开发规范》。 核心测试类要求用测试代码来覆盖,测试用例的设计,进行覆盖覆盖,如果一味的参照...

    程序员开发工程师简历模板.docx

    本人工作认真细心、责任心强、为人正直、敢于坚持原则,自我学习能力强,有较强的学习能力和程序开发能力,工作积极主动,具备较强的责任感和解决复杂问题的能力,有良好的团队协作精神、良好的沟通能力和服务意识,...

    后端开发开发技巧总结与入门常用的技巧总结.docx

    数据库优化:合理设计数据库表结构,尽量减少JOIN操作,使用索引...测试驱动开发:坚持单元测试、集成测试,保证代码质量。 持续集成/持续部署(CI/CD):构建自动化部署流水线,提高迭代效率。 容器化部署:利用Dock

    RSS案例视频,RSS阅读器

    但是这项工作没有与UserLand公司进行有效的沟通,UserLand公司也不承认RSS 1.0的有效性,并坚持按照自己的设想进一步开发出RSS的后续版本,到2002年9月发布了最新版本RSS 2.0,UserLand公司将RSS定义为“Really ...

Global site tag (gtag.js) - Google Analytics