我花了一些时间学习如何在 Android anno 2016 上创建 2D 渲染游戏循环。
我想实现以下目标:
关于 SurfaceView 的神话:
首先有几个帖子推荐SurfaceView .乍一看,这似乎是个好主意,因为它使用了单独的渲染线程,但事实证明,从 SurfaceHolder 返回的 Canvas 不能是hardware accelerated。 !!在具有 QuadHD (2560x1440) 分辨率的设备上使用带有软件渲染的 SurfaceView 效率极低。
因此我的选择是扩展一个基本 View 并覆盖 onDraw()。为每次更新调用 invalidate()。
流畅的动画:
我的下一个挑战是流畅的动画。事实证明,在 onDraw() 中读取 System.nanoTime() 是一个坏主意,因为它不会以 1/60 秒的间隔被调用,从而在我的 Sprite 上产生不稳定的运动。因此我使用了 Choreographer为我提供每个 VSYNC 的帧时间。这很好用。
目前进展:
我觉得我已经很接近了,但我在旧设备上仍然偶尔会遇到一些延迟。内存使用率非常低,所以我不认为 GC 是幕后黑手……似乎我的回调偶尔会错过/跳转 aoad 帧。
我会发布我的代码,期待您的评论和建议。
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.drawable.Drawable;
import android.support.v4.content.res.ResourcesCompat;
import android.util.AttributeSet;
import android.view.Choreographer;
import android.view.View;
public class MiniGameView extends View implements Choreographer.FrameCallback
{
private final float mDisplayDensity;
private long mFrameTime = System.nanoTime();
private final Drawable mBackground;
private final Drawable mMonkey;
public MiniGameView(Context context)
{
this(context, null);
}
public MiniGameView(Context context, AttributeSet attrs)
{
super(context, attrs);
mDisplayDensity = getResources().getDisplayMetrics().density;
// Load graphics
mBackground = ResourcesCompat.getDrawable(getResources(), R.drawable.background, null);
mMonkey = ResourcesCompat.getDrawable(getResources(), R.drawable.monkey, null);
Choreographer.getInstance().postFrameCallback(this);
}
// Receive time in nano seconds at last VSYNC. Use this frameTime for smooth animations!
@Override
public void doFrame(long frameTimeNanos)
{
mFrameTime = frameTimeNanos;
Choreographer.getInstance().postFrameCallback(this);
invalidate();
}
// Draw game here
@Override
protected void onDraw(Canvas canvas)
{
drawBackground(canvas);
drawSprites(canvas);
}
private void drawBackground(Canvas canvas)
{
mBackground.setBounds(0, 0, canvas.getWidth(), canvas.getHeight());
mBackground.draw(canvas);
}
private void drawSprites(Canvas canvas)
{
double t = mFrameTime * 0.00000001;
int width = canvas.getWidth();
int height = canvas.getHeight();
for(int i=0;i<8;i++)
{
double x = width * (1 + Math.sin(-0.181 * t)) * 0.5;
double y = height * (1 - Math.cos(0.153 * t)) * 0.5;
int size = (int)Math.round((80 + 40 * Math.cos(0.2 * t)) * mDisplayDensity);
drawSprite(canvas, mMonkey, (int) x, (int) y, size, size);
t += 0.8;
}
}
private void drawSprite(final Canvas canvas, final Drawable sprite, int x, int y, int w2, int h2)
{
sprite.setBounds(x - w2, y - h2, x + w2, y + h2);
sprite.draw(canvas);
}
}
我还创建了一个系统跟踪 file .
最佳答案
关于 SurfaceView 确实没有“神话”。它是高分辨率快速动画的最佳选择......但你必须使用 OpenGL ES。 Surfaces 上的 Canvas 渲染——SurfaceView、TextureView 等等——不是硬件加速的,并且随着像素数量的增加变得越来越昂贵。
SurfaceView 的一个有用功能是您可以将 Surface 设置为固定大小,并让显示硬件按比例放大。对于某些类型的游戏,这可以产生足够的性能。缩放比例的一个例子是 here ;请注意渲染使用 GLES。
有关游戏循环的一般建议可以在 this appendix 中找到.你似乎在做正确的事情。您可能需要考虑添加一个丢帧计数器,以查看您的动画故障是否与丢帧有关。 (如果 Choreographer 报告的连续时间从 16.7 毫秒跳到 33 毫秒,你就知道你丢了一个。)
追踪动画故障的最佳方法是使用 systrace .这些跟踪可以非常容易地准确查看所有线程在做什么,并确定暂停的因果关系。
关于android - 在 Android 上创建无延迟的 2D 游戏循环,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36802850/
出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits
关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。
我脑子里浮现出一些关于一种新编程语言的想法,所以我想我会尝试实现它。一位friend建议我尝试使用Treetop(Rubygem)来创建一个解析器。Treetop的文档很少,我以前从未做过这种事情。我的解析器表现得好像有一个无限循环,但没有堆栈跟踪;事实证明很难追踪到。有人可以指出入门级解析/AST指南的方向吗?我真的需要一些列出规则、常见用法等的东西来使用像Treetop这样的工具。我的语法分析器在GitHub上,以防有人希望帮助我改进它。class{initialize=lambda(name){receiver.name=name}greet=lambda{IO.puts("He
我有多个ActiveRecord子类Item的实例数组,我需要根据最早的事件循环打印。在这种情况下,我需要打印付款和维护日期,如下所示:ItemAmaintenancerequiredin5daysItemBpaymentrequiredin6daysItemApaymentrequiredin7daysItemBmaintenancerequiredin8days我目前有两个查询,用于查找maintenance和payment项目(非排他性查询),并输出如下内容:paymentrequiredin...maintenancerequiredin...有什么方法可以改善上述(丑陋的)代
使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta
我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在
如何使用RSpec::Core::RakeTask初始化RSpecRake任务?require'rspec/core/rake_task'RSpec::Core::RakeTask.newdo|t|#whatdoIputinhere?endInitialize函数记录在http://rubydoc.info/github/rspec/rspec-core/RSpec/Core/RakeTask#initialize-instance_method没有很好的记录;它只是说:-(RakeTask)initialize(*args,&task_block)AnewinstanceofRake
关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?
我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("
我正在阅读SandiMetz的POODR,并且遇到了一个我不太了解的编码原则。这是代码:classBicycleattr_reader:size,:chain,:tire_sizedefinitialize(args={})@size=args[:size]||1@chain=args[:chain]||2@tire_size=args[:tire_size]||3post_initialize(args)endendclassMountainBike此代码将为其各自的属性输出1,2,3,4,5。我不明白的是查找方法。当一辆山地自行车被实例化时,因为它没有自己的initialize方法