fopen用java代码实现_三、Jni开发:C语言访问JAVA的属性和方法

这篇文章主要写了本地方法访问JAVA的属性、静态属性、方法、静态方法。

还有一些相关的小知识点在补充中有介绍。

Java代码

public class JniTest {

public static String static_key = "static key";

public String key = "key";

public static int static_num = 110;

public int num = 110;

public native static String getStringFromC();

public native String getString2FromC(int i);

public native void accessField();

public native void accessStaticField();

public native void accessMethod();

public native void accessStaticMethod();

public static void main(String[] args) {

System.out.println("\n\n---getStringFromC-----------------------");

String text1 = getStringFromC();

System.out.println(text1);

System.out.println("\n\n----getString2FromC----------------------");

JniTest j = new JniTest();

String text2 = j.getString2FromC(5);

System.out.println(text2);

System.out.println("\n\n---accessField------------------------");

j.accessField();

System.out.println("修改过后的变量为" + j.key);

System.out.println("\n\n----accessStaticField-----------------------");

j.accessStaticField();

System.out.println("修改过后的静态变量为" + static_num);

System.out.println("\n\n----accessMethod-----------------------");

j.accessMethod();

System.out.println("\n\n----accessStaticMethod-----------------------");

j.accessStaticMethod();

}

static{

System.loadLibrary("JniTest1");

}

//产生指定范围的随机数

public int genRandomInt(int max){

System.out.println("genRandomInt 执行了...");

return new Random().nextInt(max);

}

//产生UUID字符串

public static String getUUID(){

return UUID.randomUUID().toString();

}

}

C语言代码

#define _CRT_SECURE_NO_WARNINGS

#include "JniTest.h"

#include

#include

#include

JNIEXPORT jstring JNICALL Java_JniTest_getStringFromC

(JNIEnv * env, jclass jcla) {

return (*env)->NewStringUTF(env,"String From C");

}

JNIEXPORT jstring JNICALL Java_JniTest_getString2FromC

(JNIEnv *env, jobject jobj, jint num) {

//此处并没有使用num,读者可以自己实验

return (*env)->NewStringUTF(env, "String From C2");;

}

JNIEXPORT void JNICALL Java_JniTest_accessField

(JNIEnv *env, jobject jobj) {

jclass jcla = (*env)->GetObjectClass(env, jobj);

jfieldID fid = (*env)->GetFieldID(env, jcla, "key", "Ljava/lang/String;");

jstring jstr = (*env)->GetObjectField(env, jobj, fid);

//修改局部变量

char *c_str = (*env)->GetStringUTFChars(env, jstr, NULL);

char text[20] = "super";

strcat(text, c_str);

jstring new_str = (*env)->NewStringUTF(env, text);

(*env)->SetObjectField(env, jobj, fid, new_str);

}

JNIEXPORT void JNICALL Java_JniTest_accessStaticField

(JNIEnv *env, jobject jobj) {

jclass jcla = (*env)->GetObjectClass(env, jobj);

jfieldID fid = (*env)->GetStaticFieldID(env, jcla, "static_num", "I");

jint num = (*env)->GetStaticIntField(env, jcla, fid);

num = 119;

(*env)->SetStaticIntField(env, jcla, fid, num);

}

JNIEXPORT void JNICALL Java_JniTest_accessMethod

(JNIEnv *env, jobject jobj) {

jclass jcla = (*env)->GetObjectClass(env,jobj);

jmethodID mid = (*env)->GetMethodID(env, jcla, "genRandomInt", "(I)I");

jint random = (*env)->CallIntMethod(env, jobj, mid, 200);

printf("random num:%ld", random);

}

JNIEXPORT void JNICALL Java_JniTest_accessStaticMethod

(JNIEnv *env, jobject jobj) {

jclass jcla = (*env)->GetObjectClass(env, jobj);

jmethodID mid = (*env)->GetStaticMethodID(env, jcla, "getUUID", "()Ljava/lang/String;");

jstring uuid = (*env)->CallStaticObjectMethod(env, jcla, mid);

char *uuid_str = (*env)->GetStringUTFChars(env, uuid, JNI_FALSE);

//拼接

char filename[100];

sprintf(filename, "D://%s.txt", uuid_str);

FILE *fp = fopen(filename, "w");

fputs("i love jason", fp);

fclose(fp);

}

以上代码分别为:

JAVA方法调用无参本地方法。

JAVA方法调用有参的本地方法。

本地方法访问并修改JAVA中的属性

本地方法访问并修改JAVA中的静态属性

本地方法调用JAVA中的方法

本地方法调用JAVA中的静态方法

读者手撸一遍代码应该就能理解。

afbbdafabc8b

调用结果

afbbdafabc8b

生成的UUID文件

这个地方有个问题:就是random num:60这里为什么会比accessStaticMethod后打印。还需要研究一下。

补充

当你在JAVA中声明本地方法时,声明的是静态方法则本地方法属于类,传入的是jclass,当声明的是普通方法时,本地方法属于该类的实例化对象,传入的是jobject。

在获取JAVA属性的时候需要传入属性名称的字符串,还需要传入属性的签名,如下图中的 "Ljava/lang/String;" 就是签名。注意分号

jfieldID fid = (*env)->GetFieldID(env, jcla, "key", "Ljava/lang/String;");

afbbdafabc8b

签名

当本地方法访问JAVA的属性时,需要写签名,基本类型写的就是 “Z”,“B”这种,如果属性是一个对象,则需要写L+全类名;例如“Ljava/lang/String;”。

当本地方法访问的是JAVA的方法时,如访问上文中public int genRandomInt(int max);这个方法的签名就是"(I)I";

访问 public static String getUUID();这个方法时,签名是"()Ljava/lang/String;"

即先写括号,括号内写参数的前面,括号后面跟着写返回值的签名。

不会写的也可以偷懒直接生成。

Jni开发补充:怎么获取签名

这个函数这里的NULL;

char *c_str = (*env)->GetStringUTFChars(env, jstr, NULL);

在VS中按F12打开函数的原型。

const char* (JNICALL *GetStringUTFChars)(JNIEnv *env, jstring str, jboolean *isCopy);

注意看,这里的isCopy是一个布尔类型的指针。此处的iscopy是一个传出参数,不是传入参数,目的是让程序员知道该字符串是否复制了,至于是否复制是由Jni自己决定的。

在寻找某个属性或者方法的时候如果寻找不到,会抛出一个“异常"(错误)。

这个地方抛出的其实是一个Throwable,所以在JAVA中catch的时候要catch Throwable。关于异常这块这里只做简单的了解,后面会有单独的章节做介绍。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值