鍍金池/ 問答/Java  C++  網(wǎng)絡(luò)安全/ JNI調(diào)用FindClass獲取package的時候失敗. 不帶包名則調(diào)用成功為

JNI調(diào)用FindClass獲取package的時候失敗. 不帶包名則調(diào)用成功為什么?

c++進(jìn)程啟動jvm虛擬機(jī) 然后獲取java相關(guān)類方法(使用FindClass), 如果只有一個類名是成功的,但是如果有包的話就會調(diào)用失敗,相關(guān)代碼如下:

//MyTest.java
package com;
import  java.lang.management.ManagementFactory;

public class MyTest {
    //private static int magic_counter = 777;

    public static boolean start(String path) {
        System.out.print("MyTest start: " + path);
        return true;
    }
}
// main.cpp
#include <iostream>
#include <jni.h>
#include <memory.h>

int main()
{
    char opt1[] = "-Djava.compiler=NONE";
    char opt2[] = "-Djava.class.path=./MyTest.jar";
    //char opt2[] = "-Djava.class.path=.";
    char opt3[] = "-verbose:NONE";

    JavaVMOption options[3];
    options[0].optionString = opt1; options[0].extraInfo = NULL;
    options[1].optionString = opt2; options[1].extraInfo = NULL;
    options[2].optionString = opt3; options[2].extraInfo = NULL;

    JavaVMInitArgs jargv;
    jargv.version = JNI_VERSION_1_6;
    jargv.nOptions = 3;
    jargv.options = options;
    jargv.ignoreUnrecognized = JNI_TRUE;

    JavaVM* jvm = NULL;
    JNIEnv* jenv = NULL;
    jint res = JNI_CreateJavaVM( &jvm, (void**)&jenv, &jargv  );
    if ( 0 != res  )
        return 1;

    jclass jc = jenv->FindClass("/com/MyTest");
    //jclass jc = jenv->FindClass("MyTest");
    //jclass jc = jenv->FindClass("java/lang/String");
    if ( NULL == jc  ) {
        if( jenv->ExceptionOccurred()  )
            jenv->ExceptionDescribe();
        else {
            std::cout << "jc null" << std::endl;
        }
        return 1;
    }

    jmethodID jmid = jenv->GetStaticMethodID( jc, "start", "(Ljava/lang/String;)Z");
    if ( NULL == jmid) {
        std::cout << "jmid null" << std::endl;
        return 1;
    }
    else {
        std::cout << "jmid ok" << std::endl;
    }

    jstring jniStr = jenv->NewStringUTF("jni to java");
    jboolean res1 = jenv->CallBooleanMethod( jc, jmid, jniStr);
    if(res1) {
        std::cout << "CallBooleanMethod ok" << std::endl;
    }

    std::cout << "\n";
    std::cout << "Hello, World!" << std::endl;
        return 0;
}

相關(guān)命令如下:

javac MyTest.java
jar cvf MyTest.jar MyTest.class
g++ main.cpp -I
~/jdk/include/ -I~/jdk/include/linux -L~s/jdk/jre/lib/amd64/server -ljvm

這是運(yùn)行a.out會出現(xiàn)報(bào)錯提示:類沒有找到:
Exception in thread "main" java.lang.NoClassDefFoundError: /com/MyTest
Caused by: java.lang.ClassNotFoundException: .com.MyTest

如果java中把包名com去掉,c++中直接找類MyTest,那么FindClass就是成功的.

環(huán)境: ubuntu 16.04 jdk 1.8

回答
編輯回答
氕氘氚

已解決.
java對于帶報(bào)名的類的查找是需要路徑的,看修改就知道了.
mkdir com
mv MyTest.jar com/
tar cvf MyTest.jar com

2017年5月29日 15:00