- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我从其他几份报告中读到,人们通常在 4-80 ns 上花费一个普通的、基本的 JNI 调用:
For trivial native methods, last year I found calls to average 40 ns on my Windows desktop, and 11 ns on my Mac desktop ..
来自 Possible increase of performace using JNI?
However JNI calls often take around 30 ns ..
当我在我的 JNI 代码中调用简单方法时(简单的意思是不超过一个时间 int 返回类型 int 的参数),我得到的往返调用时间(用 System.nanoTIme 测量)在 50,000 到80,000 纳秒。
如果我对 VM 进行“预热”并在计时之前运行调用数百次,我仍然会得到大约 2000-4000 ns(低于 800-1000)。 (如上所述,我听说其他人报告 < 100 ns.. 并且在频繁调用时加起来确实比这高 10-20 倍。)
这是正常速度吗?是什么导致我的 native 代码调用速度如此之慢?
更新:
JNIEXPORT jint JNICALL Java_com_snap2d_gl_RenderControl_correctGammaNative
(JNIEnv *env, jobject obj, jint pixel) {
return X2D_correctGamma(pixel, 1.0f);
}
其中 X2D_correctGamma(int,float) 是一种校正像素 Gamma 值的方法(自发布以来我已经实现了 native 代码)。
Java 基准测试:
for(int i = 0; i < 100; i++) {
long t1 = System.nanoTime();
correctGammaNative(0xFFF);
long t2 = System.nanoTime();
System.out.println(t2 - t1);
}
这是“热身”代码。大多数 printlns 在初始调用后读取 800-1000ns。
不幸的是,我可能不得不放弃它,因为它应该用于渲染,每秒调用它数千次会使帧速率降至 1 FPS。
系统信息:
在 JDK1.6.0_32(64 位)、JDK1.7.0_04(64 位)和 JRE1.7.0_10(32 位)上表现相似
Windows 7 64 位
16GB 内存
i7-3770 四核 CPU @ 3.4-3.9ghz
GNU GCC MinGW 编译器(32 位和 64 位)
最佳答案
Is this normal speed?
没有。如果您真的每次 JNI 调用获得 50,000-80,000 ns,则说明发生了一些奇怪的事情。
What could be causing my native code to be called so much slower?
不知道。它几乎可以是任何东西。但是,如果您向我们展示 native 代码和 Java 代码,我们将能够更好地进行解释。
我的钱会花在这上面,这根本不是 JNI 调用的问题。相反,我希望它是您进行基准测试的方式的产物。您可以做(或不做)的很多事情都会导致 Java 基准测试产生虚假结果。我们需要查看您的基准测试代码。
好的,您的更新表明您之前报告的时间(50,000-80,000 或 2000-4000)不正确或不相关。考虑到以下情况,800-1000ns 的计时听起来是合理的。
我认为您的基准测试存在三个缺陷。
您正在尝试测量几纳秒量级的时间间隔。但是您的测量没有考虑到调用 System.nanoTime()
需要很长时间。您需要做的是测量在每对 System.nanoTime()
调用之间进行几千或几百万次 JNI 调用所花费的时间,然后计算并打印平均值。
您的代码没有将进行 JNI 调用所花费的时间与执行调用主体所花费的时间分开。 (或者也许它确实如此……而您还没有向我们展示该代码。)。我怀疑 gamma 校正将比 JNI 调用开销花费更长的时间。
你的热身不够。您运行代码的时间是否足以让 JIT 编译启动是值得怀疑的。此外,您的基准代码仅限于单个方法调用这一事实意味着即使 JIT 编译器确实运行了,您也有可能d 从不调用该方法的 JIT 编译版本。将基准代码放入一个方法中,并重复调用该方法。
关于Java JNI 调用比预期慢(至少 2 毫秒/调用),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14269798/
我刚开始使用 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这样的更高级别,但仍然能够侵入进程内存。 有什么线索吗? []的 最佳答案 这
我是一名优秀的程序员,十分优秀!