实现一个在JNI中调用Java对象的工具类,从此一行代码就搞定!

简介: 我们知道在jni中执行一个java函数需要调用几行代码才行,如jclass objClass = (*env).GetObjectClass(obj);jmethodID methodID = (*env).GetMethodID(objClass, methodName, methodSig);jobject result = (*env).CallObjectMethod(obj, methodID, ...);这样使用起来很不方便,尤其当需要大量的调用java函数就会产生大量的上述代码,由此我产生了一个开发封装这些操作的工具类,以便大量简化我们的开发。作者:Benn

前言


我们知道在jni中执行一个java函数需要调用几行代码才行,如


jclass objClass = (*env).GetObjectClass(obj);
jmethodID methodID = (*env).GetMethodID(objClass, methodName, methodSig);
jobject result = (*env).CallObjectMethod(obj, methodID, ...);
复制代码


这样使用起来很不方便,尤其当需要大量的调用java函数就会产生大量的上述代码,由此我产生了一个开发封装这些操作的工具类,以便大量简化我们的开发。


简单封装


其实可以看到整个过程基本是固定不变的:先获取Class,然后获取method,然后在执行call。所以可以简单的先封装成一系列工具函数,如:


jobject callObjMethod(JNIEnv *env, jobject obj, const char *methodName, const char *methodSig, ...){
    va_list args;
    jclass objClass = (*env).GetObjectClass(obj);
    jmethodID methodID = (*env).GetMethodID(objClass, methodName, methodSig);
    va_start(args,methodSig);
    jobject result = (*env).CallObjectMethodV(obj, methodID, args);
    va_end(args);
    return result;
}
jint callIntMethod(JNIEnv *env, jobject obj, const char *methodName, const char *methodSig, ...){
    va_list args;
    jclass objClass = (*env).GetObjectClass(obj);
    jmethodID methodID = (*env).GetMethodID(objClass, methodName, methodSig);
    va_start(args,methodSig);
    jint result = (*env).CallIntMethodV(obj, methodID, args);
    va_end(args);
    return result;
}
jboolean callBooleanMethod(JNIEnv *env, jobject obj, const char *methodName, const char *methodSig, ...){
    va_list args;
    jclass objClass = (*env).GetObjectClass(obj);
    jmethodID methodID = (*env).GetMethodID(objClass, methodName, methodSig);
    va_start(args,methodSig);
    jboolean result = (*env).CallBooleanMethodV(obj, methodID, args);
    va_end(args);
    return result;
}
复制代码


这样当我们要通过jni执行某个java函数的时候,就一行代码就可以搞定了,比如String.length()


jint len = callIntMethod(env, str, "length", "()I")
复制代码


这样就可以大大减少了代码量,而且代码也更易读了。


优化


通过上面可以看到这些函数大部分代码都非常类似,只有一行代码和返回值有区别,所以我考虑使用函数模版来进行优化,如下:


template <typename T>
T callMethod(JNIEnv *env, jobject obj, const char *methodName, const char *methodSig, ...){
    va_list args;
    jclass objClass = (*env).GetObjectClass(obj);
    jmethodID methodID = (*env).GetMethodID(objClass, methodName, methodSig);
    va_start(args,methodSig);
    T result;
    if(typeid(T) == typeid(jobject)){
         result = (*env).CallObjectMethodV(obj, methodID, args);
    }
    if(typeid(T) == typeid(jdouble)){
        result = (*env).CallDoubleMethodV(obj, methodID, args);
    }
    ...
    va_end(args);
    return *result;
}
复制代码


这样只要调用callMethod<return type>即可,愿望很美好,但是上面代码实际上是无法通过编译。

因为模版函数实际上是在编译时,根据调用的类型,拷贝生成多个具体类型的函数以便使用。

所以如果有这样的调用callMethod<jobject>(...),在编译时就会拷贝成一个如下的函数:


jobject callMethod(JNIEnv *env, jobject obj, const char *methodName, const char *methodSig, ...){
    va_list args;
    jclass objClass = (*env).GetObjectClass(obj);
    jmethodID methodID = (*env).GetMethodID(objClass, methodName, methodSig);
    va_start(args,methodSig);
    jobject result;
    if(typeid(jobject) == typeid(jobject)){
         result = (*env).CallObjectMethodV(obj, methodID, args);
    }
    if(typeid(jobject) == typeid(jdouble)){
        result = (*env).CallDoubleMethodV(obj, methodID, args);
    }
    ...
    va_end(args);
    return *result;
}
复制代码


注意这行代码:


if(typeid(jobject) == typeid(jdouble)){
      result = (*env).CallDoubleMethodV(obj, methodID, args);
}
复制代码


虽然实际上是无法执行的代码,但是编译时还是会进行检查,由于将jdouble类型的赋值给jobject类型的result,所以编译不通过,类型无法转换。而且这里用强转static_cast等方法都不行。

我考虑两种方法来解决这个问题,一种是保证编译不报错,因为运行时不会执行的代码,只要通过编译就可以。另外一种是不同的类型编译不同的代码。


void指针


在c++中void指针可以被赋值任何类型指针,且void指针强转为任何类型指针在编译时不会报错。代码如下:


template <typename T>
T callMethod(JNIEnv *env, jobject obj, const char *methodName, const char *methodSig, ...){
    va_list args;
    jclass objClass = (*env).GetObjectClass(obj);
    jmethodID methodID = (*env).GetMethodID(objClass, methodName, methodSig);
    va_start(args,methodSig);
    T* result = new T();
    if(typeid(T) == typeid(jobject)){
        jobject objec = (*env).CallObjectMethodV(obj, methodID, args);
        void *p = &objec;
        result = (T*)p;
    }
    if(typeid(T) == typeid(jdouble)){
        jdouble doub = (*env).CallDoubleMethodV(obj, methodID, args);
        void *p = &doub;
        result = (T*)p;
    }
    va_end(args);
    return *result;
}
复制代码


当然利用void指针很不安全,虽然可以通过编译,但是执行时如果类型不同会直接造成crash。所以并不建议这种方式。


模版函数特例化


将差异代码部分封装到另一个模版函数中,并且对每种类型进行特例化,这样还可以去掉if-else判断,代码如下:


template <typename K>
K call2Result(JNIEnv *env, jobject obj, jmethodID methodID, va_list args){
    return *(new K());
}
template <>
jobject call2Result(JNIEnv *env, jobject obj, jmethodID methodID, va_list args){
    return (*env).CallObjectMethodV(obj, methodID, args);
}
template <>
jdouble call2Result(JNIEnv *env, jobject obj, jmethodID methodID, va_list args){
    return (*env).CallDoubleMethodV(obj, methodID, args);
}
...
template <typename T>
T callMethod(JNIEnv *env, jobject obj, const char *methodName, const char *methodSig, ...){
    va_list args;
    jclass objClass = (*env).GetObjectClass(obj);
    jmethodID methodID = (*env).GetMethodID(objClass, methodName, methodSig);
    va_start(args,methodSig);
    T result = call2Result<T>(env, obj, methodID, args);
    va_end(args);
    return result;
}
复制代码


这样在编译时,如果返回值是jobject类型的,当编译到call2Result时,就会直接调用jobject call2Result(...)这个函数,就不再涉及类型转换的问题。

这样去掉了if判断,但是由于没有通用的函数,所以所有使用的类型都需要特例化,如果某个类型未特例化,代码执行可能就会有问题。而在jni中,与java对应的类型其实就那么十几种,所以我们只要全部实现一遍call2Result即可。


undefined reference to

使用模版函数出现这个问题,是因为没有将模版函数的实现写在头文件中,只将模版函数的声明在头文件中,而在源文件中实现的。

所以我们应该将模版函数的实现也写进头文件中,而模版函数特例化则可以在源文件中实现,但是注意要include头文件。


返回值是void类型


因为void的特殊性,所以如果当成泛型来处理会有很多问题,这里把返回值是void类型的单独实现一个函数即可。

总结


上面我们仅仅是实现了调用普通函数的工具,根据这个思路我们还可以实现调用静态函数、获取成员变量、赋值成员变量等,这样当我们在进行jni开发的时候,如果需要对java对象或类进行操作,只需要一行代码就可以了。



目录
相关文章
|
1天前
|
Java Spring
JAVA注解:传统与现代的完美结合,你的代码值得拥有!
【6月更文挑战第29天】Java注解,作为连接传统与现代的编程工具,简化企业级应用开发,提升代码可读性和维护性。通过自定义注解如`@Loggable`,可以将行为(如日志记录)与方法实现分离,减少模板代码。使用AOP(如Spring)处理注解,实现行为拦截,增强代码灵活性和可扩展性。拥抱Java注解,让代码更现代、更高效!
26 16
|
1天前
|
IDE Java 编译器
JAVA注解,你的代码需要的一次“心灵按摩”!
【6月更文挑战第29天】Java注解是提升代码可维护性的关键,它们是编译器和IDE理解代码意图的特殊标记,不同于仅作解释的注释。注解可用于编译时检查(如@Override、@NotNull)、自动生成代码(Lombok的@Getter、@Setter)、框架集成(Spring的@Autowired、MyBatis的@Mapper)。通过注解,代码变得更简洁、功能更强大,为项目带来效率提升。尝试使用注解,赋予代码新生命!
21 12
|
1天前
|
Java 编译器 开发者
JAVA注解,让代码“开口说话”的魔法术!
【6月更文挑战第29天】Java注解,一种元数据机制,让代码“开口”传达意图。它们增强可读性,提供编译器与框架处理代码的额外信息。例如,@Description注解描述方法功能,@Autowired在Spring中自动装配Bean,自定义注解如@MyCustomAnnotation允许定义独特行为。注解提升开发效率,是理解与使用Java的关键。
|
1天前
|
Java 编译器 数据库连接
JAVA注解:代码界的“隐形翅膀”?!
【6月更文挑战第29天】Java注解,编程的“隐形翅膀”,提供编译检查、框架集成及自定义元数据功能。如@Override确保方法重写正确,@Autowired在Spring中自动装配Bean。通过自定义注解,开发者能创造独特代码逻辑。例如,定义@SpecialProcessing注解标记需特殊处理的方法,增强代码可读性和可扩展性。利用注解,让代码飞翔在更广阔的世界。
11 1
|
1天前
|
IDE Java 数据库连接
JAVA注解:元数据,代码的“身份证”?!
【6月更文挑战第29天】Java注解,作为代码的“身份证”,提供元数据,用于编译时检查、自动生成代码和框架集成。例如,@Override确保方法重写正确,@Deprecated标记过时,@Autowired在Spring中实现依赖注入。Lombok的@Getter/@Setter简化getter/setter。注解提升代码质量和效率,是现代Java开发的关键实践。
7 0
|
1天前
|
IDE Java 数据库连接
JAVA注解:那些年,我们错过的代码元数据!
【6月更文挑战第29天】Java注解,非执行代码的元数据,常被误解为注释。其实,它们支持编译时检查(如@Override)、自动生成代码、依赖注入(如Spring)和ORM映射(如Hibernate)。通过自定义注解,如示例中的`@MyAnnotation`,可在运行时通过反射增强方法功能。别再错过注解的力量,让它们提升代码的灵活性和可维护性!
7 0
|
1天前
|
Java Spring
揭秘JAVA注解:你的代码竟藏着这么多小秘密?!
【6月更文挑战第29天】JAVA注解是元数据机制,用于为代码提供额外信息。它们简化代码,如`@Override`确保正确重写方法,提供配置(如Spring的`@Autowired`),并允许自定义注解。通过反射读取注解,与工具集成增强功能。注解开启更高效编程,是Java世界中的隐形助手。
java类模拟客户端调用servlet (httpClient)
最近做项目,用到JAVA普通类调用一个servlet应用,所以把部分代码也贴上来了。
123 0
java类模拟客户端调用servlet (httpClient)
java类模拟客户端调用servlet (httpClient)
174 0
|
22小时前
|
Java 数据处理 调度
Java多线程编程入门指南
Java多线程编程入门指南