使用JNI从Java调用本机(C++)函数时发生访问冲突:这是我的代码问题,还是已知问题
Access violation calling a native (C++) function from Java using JNI: Is this an issue with my code, or a known issue?
我有一个在C++代码中定义的本机函数,它在Java中调用(在C++调用Java代码之后)。
(1) C++中本机函数的声明/定义:
// C++ declaration/definition of the native function:
JNIEXPORT void JNICALL setEncoderProgressStatus (JNIEnv * env, jobject theClass, jlong jEncoderDecoderDlg, jstring status)
{
// Do nothing. Obviously, the real function does something.
// But an access violation is reported even in the do-nothing case.
}
(2) 在C++中注册本机函数
// Registration of this function with JNI in the C++ code
JNINativeMethod commandLineEncoderMethods[] =
{
{"setEncoderProgressStatus","(JLjava/lang/String;)V", (void*)setEncoderProgressStatus}
};
// pEnv is a valid JNIEnv *
// jCommandLineEncoderClass has already been initialized
pEnv->RegisterNatives(jCommandLineEncoderClass, commandLineEncoderMethods, sizeof(commandLineEncoderMethods)/sizeof(JNINativeMethod))
(3) Java中本机函数的声明:
// The Java code, in turn, declares the native function as follows:
public static native void setEncoderProgressStatus(long EncoderDecoderDlg, String status);
(4) 在Java代码中调用本机函数:
// Finally, the Java code calls the native function as follows:
// The real code passes different args,
// but an access violation is reported even in the do-nothing case.
// Commenting out the following line of code makes the access violation disappear.
setEncoderProgressStatus(0, "");
请注意,此代码100%成功地工作,并且已经工作了很长时间。我刚刚注意到,在VisualStudio2010的调试器中运行时,会报告访问冲突。简单地注释掉对Java中本机函数的调用,就会导致访问冲突消失。
如果我知道在VS2010中以调试模式运行应用程序时,这是JNI的一个已知"侥幸",并且这并不代表我的代码有问题,那么我会感到满意。
但是,我想确认我的代码没有可能的问题。
*附录*
访问冲突不会导致程序执行停止。相反,访问违规的唯一指示是"输出"窗口中的一行。我不确定这是否意味着访问违规得到了处理;然而,我想这意味着它被处理了。然而,即使它被处理了,我也想知道它是否一定表明我的代码有问题。
这是输出窗口中的消息:
EncoderDecoder.exe中0x03fbb256处的首次机会异常:0xC0000005:访问违规读取位置0x003d0100
。。。
*第二次附录*
我发现,当C++应用程序保持与JVM的连接,并且我再次从C++应用程序执行相同的代码(导致调用相同的Java代码,进而导致Java代码调用相同的JNI函数)时,访问违规不会出现。
(在我的实际程序中,有许多不同的本机函数在许多不同的类中被调用,因此准确地隔离访问冲突何时被报告和何时停止被报告并不简单,但在不退出C++应用程序的情况下,使用相同加载的JVM,第三次运行相同的功能时,会出现没有第一次访问冲突。)n个错误(也没有明显的访问冲突错误)。
这让我相信,结合Ben Voigt下面的回答(以及相关评论),JVM代码确实导致first-chance exception (access violation)
作为其NORMAL处理的一部分发生,并且这不是我的代码的问题。
*附录3*
发件人http://blogs.msdn.com/b/davidklinems/archive/2005/07/12/438061.aspx:
首次机会异常消息通常并不意味着存在代码中的问题。
仍然有可能,碰巧是访问违规的首次机会异常更有可能代表真正的问题,在这种情况下,我的问题仍然悬而未决;但是,我认为证据表明,这可能只是JVM的工作方式,并不代表我的代码有问题。
这是未处理的访问冲突,还是在继续(让异常处理程序运行)的情况下继续调用?
JIT编译器有一大堆技巧来为函数生成本机代码,而不会让每次调用都要花费一个条件来查看函数是否已被JIT化。一种这样的技术是拦截第一个调用,因为它会生成一个访问冲突来触发JIT编译器。生成一次代码后,将来的调用将毫无例外地进行。
我不知道是否有任何Java JIT编译器真的这样做了,但如果不提供有关运行的JIT的具体信息,就无法得到确切的答案。它是Sun/Oracle HotSpot编译器吗?x86或x86_64或其他平台?您深入了解了因实现而异的实现细节。
- 写入位置0x0000000C时发生访问冲突
- 引发异常:读取访问冲突**dynamicArray**为0x1118235.发生
- 链表中写入访问冲突的未知原因
- C++中的openCV Mat访问冲突
- C++尝试深度复制唯一指针时出现内存访问冲突
- C++ 中动态二维数组的访问冲突
- 从嵌套循环中的 std::list 中删除将返回访问冲突
- 0xC0000005:访问冲突读取位置 0x00000000. 重载 == 运算符的问题
- 如何解决此C++访问冲突问题
- 计数排序输出问题:读取访问冲突
- 操作员问题 - 访问冲突
- 调用基构造函数时遇到问题 - 获取读取访问冲突
- (SDL 渲染问题)C++ 引发异常:读取访问冲突.这是无效的
- 使用JNI从Java调用本机(C++)函数时发生访问冲突:这是我的代码问题,还是已知问题
- 将内联GASM移植到x64 MASM访问冲突问题
- 访问冲突写入位置0x00000000.内存集函数问题
- 初始值设定项列表中的QString导致访问冲突.这里出了什么问题
- 写入位置0x00000000时发生访问冲突.指针问题
- 从c#调用c++ DLL和访问冲突问题,我的选择是什么
- c++跟踪日志问题,fstream访问冲突问题