- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我刚刚向我正在开发的 Android 游戏添加了一些计算量大的代码。有问题的代码是碰撞检测例程的集合,这些例程经常被调用(游戏循环的每次迭代)并且正在进行大量计算。我觉得我的碰撞检测实现相当完善,而且速度相当快,我可以用 Java 实现。
我一直在使用 Traceview分析代码,这段新的碰撞检测代码毫不奇怪地使我的游戏逻辑的持续时间增加了一倍。这显然是一个问题,因为对于某些设备,这种性能下降可能会使我的游戏从可玩状态变为不可玩状态。
我一直在考虑优化此代码的不同方法,我想知道通过将代码移至 C++ 并使用 JNI 访问它是否会显着节省性能?
上述问题是我最关心的问题,也是我提出问题的原因。我确定以下两个原因是使用 JNI 的其他积极结果。但是,这并不足以说服我将我的代码移植到 C++。
这将使代码更清晰。由于大多数碰撞检测都是某种矢量数学,因此能够使用重载运算符比在 Java 中使用一些更冗长的矢量类要干净得多。
内存管理会更简单。你说简单点?好吧,这是一款不欢迎运行垃圾收集器的游戏,因为如果 GC 必须不断中断以进行清理,它最终可能会破坏游戏的性能。在 C 中,我不必担心垃圾收集器,因此我可以避免在 Java 中使用临时静态变量做的所有丑陋事情,而只依赖于 C++ 的良好旧堆栈内存
尽管这个问题可能很冗长,但我想我已经涵盖了所有要点。有了这些信息,是否值得将我的代码从 Java 移植到 C++ 并使用 JNI 访问它(出于提高性能的原因)?另外,有没有办法衡量或估计潜在的性能提升?
编辑:
所以我做到了。结果?从 TraceView 的角度来看,我的碰撞检测程序的速度提高了 6 倍。
虽然到达那里并不容易。除了必须进行 JNI 舞蹈之外,我还必须进行一些我没有预料到的优化。主要是使用直接分配的浮点缓冲区将数据从 Java 传递到 native 。我最初的尝试只是使用一个 float 组来保存有问题的数据,因为从 Java 到 C++ 的转换更自然,但那真的很慢。直接缓冲区完全回避了 java 和 native 之间数组复制的性能问题,给我留下了 6 倍的提升。
另外,我没有使用自己的向量类,而是使用了 Eigen 数学库。我不确定这对性能有多大影响,但至少,它节省了我开发自己的(效率较低的)向量类的时间。
另一个教训是,过多的日志记录不利于性能(不明显的 jic)。
最佳答案
不是真正直接回答您的问题,但以下链接可能对您有用:
第二个链接是这样写的:
Native code isn't necessarily more efficient than Java. For one thing, there's a cost associated with the Java-native transition, and the JIT can't optimize across these boundaries. If you're allocating native resources (memory on the native heap, file descriptors, or whatever), it can be significantly more difficult to arrange timely collection of these resources. You also need to compile your code for each architecture you wish to run on (rather than rely on it having a JIT). You may even have to compile multiple versions for what you consider the same architecture: native code compiled for the ARM processor in the G1 can't take full advantage of the ARM in the Nexus One, and code compiled for the ARM in the Nexus One won't run on the ARM in the G1.
Native code is primarily useful when you have an existing native codebase that you want to port to Android, not for "speeding up" parts of a Java app.
关于android - 使用JNI,还是不使用JNI(Android性能),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9031567/
我刚开始使用 JNI,但遇到以下问题。 我有一个包含简单类的 C++ 库。我从 Java Android 项目中调用了三个 JNI 方法,分别实例化所述类、调用实例化类的方法并销毁它。我保留了对该对象
背景 我有一个 android 项目,它使用 JNI(使用 NDK)以 Java 和 C/C++ 进行编码。 我在java端创建了一个Jni java包装器,它将自己完成所有Jni操作,而除了这个包装
我想传递一个java对象的地址,JNI方法应该填充所传递对象的字段。1. java代码传递给定的对象引用。 JNI 方法应该能够缓存这个对象。这是一次性通话。2. 然后,java 对象使用不同的参数重
我本质上是在尝试遵循描述的 SO 解决方案 here , 但我遇到了问题。 这基本上就是我在 JNI 库中所做的事情: handle = dlopen("/data/data/lib/my.packa
我试图实现某种异常处理 一开始我打电话 jni::ExceptionDescribe() 之后我使用了的实现 How to obtain a description of a Java excepti
是否可以从不同 jni 库的另一个方法调用 1 个库的 jni 方法?例如:我有 2 个库 lib_1.so 和 lib_2.so。 我想从 lib_2.so 调用 lib_1.so 的方法 get_
我想在native方法中使用动态注册,所以我需要设置JNI_onLoad功能。我只是写了一个函数来获取两个数字的总和。但是,它无法正确构建。我该如何更正错误? 这是我的 *.cpp 文件,我将此文件命
我已经为 C 头文件制作了一个 make 文件,它工作正常,但是说 JNICALL 和 JNIEnv 存在语法错误,但我已经弄明白了这是因为头文件中的类型。 Image of the failure
我需要实现一个本地方法,比方说“public native void someFunc();”。我有两个库,libabc.so 和 libdef.so。 Java 使用 System.loadLibr
背景 我正在 eclipse 中为 android 开发一个应用程序,现在我遇到了一个问题,我需要你的帮助。所以我必须从 JAVA 应用程序调用用 C 编写的函数。但是在我编写代码的过程中,我有一些问
我正在使用 Android 上的 Java native 接口(interface)将当前 Activity 传递给 native 方法。但我没有使用类似 JNI 的函数名称来执行此操作。我正在手动注
我正在使用 JNI,我想知道是否可以通过 delegate 进行通信。 例如: Kotlin typealias MessageReceived = (msg: String) -> Unit ext
看来我对 JNI 的运气并不好。我正在等待我买的书到货,但现在是试错法。 我正在使用 JNI 来实现 Lua 求值器。 evaluatorNew() 只是创建一个新的 Evaluator() 对象,创
我有下面的代码,我想调用在同一个源文件中实现的函数,在本例中使用 C 语言: JNIEXPORT jstring JNICALL MyClass_get_1Uname__C (JNIEnv *env,
硬件手机和平板电脑内存太少,但 HAXM 工作正常? 我的基本问题是我的应用程序(很可能是我的 Java 应用程序加载的 JNI 动态库)太大。如果未使用硬件电话和模板调用 JNI 指令,Java 应
我正面临崩溃 JNI WARNING : 0x44f81e80 is not a valid JNI reference, in Ldalvik/system/NativeStart;. run()v
我移植了很多数学知识。我正在使用 over to c++ from java 并看到这样做有很大的性能提升,但我无法弄清楚要使用什么 jni 函数来摆脱我不再需要的变量。例如,我知道当您的 jni 方
我正在使用 JNI 调用一个静态 java 方法,该方法又创建一个 Swing JFrame 并显示它。代码相当简单,Java 代码独立运行(即 java StartAWT 做它应该做的事),而当使用
我正在尝试创建一个新线程,因此我将 VM 从我的方法初始化(从 Java 调用)传递到我的新线程。在线程中,我调用 AttachCurrentThread 并获取 JNIEnv* env。 稍后,我尝
我想知道是否有可能从java调用C++方法。 我非常希望能够从 java 读取内存进程。 我懂c++,但我想使用像java这样的更高级别,但仍然能够侵入进程内存。 有什么线索吗? []的 最佳答案 这
我是一名优秀的程序员,十分优秀!