eclipse jni so 调用_Linux下eclipse中通过jni调用.so实例

系统环境:64位Ubuntu14.04、eclipse 5.3版本,jdk1.8版本,vim编译器

1、本实例是java代码通过jni调用c/c++语言生成的.so库,本库提供两个方法,当调用读目录方法,并输入一个目录的路径时,会打印出输入目录下的所有文件,当调用,读文件方法,并输入一个目录文件的路径会读取本文件的前五个字节,并创建一个和输入文件名相同后缀添加一个.new的文件,然后将读取的五个字节写入到本文件中。

2、在eclipse中创建一个类,流程如下File->New->Java Project->在Projectname框中写入项目名称->点击finish。右击右侧栏的项目中的src选择new->Package->在name中输入demo(这是java中的包名)->Finish,然后右击src下的demo->new->class->在name框中输入MainClass(这个是类名)。

3、在类中写如下代码

Java代码

package demo;

importjava.io.BufferedReader;

importjava.io.IOException;

importjava.io.InputStreamReader;

public classMainClass {

static

{

//在linux中可使用这个函数然后直接写so的绝对路径加载so文件

System.load("/home/native/read_dir_file/libTestReadFile.so");

}

//在MainClass类中定义了一个String结构体

public String message = null;

public native void callCppFunction();

public native void read_dir();

public native void read_file();

public static void main(String [] args) throwsIOException

{

BufferedReader reader = newBufferedReader(newInputStreamReader(System.in));

String str = reader.readLine();

MainClass obj = new MainClass();

obj.message = str;

//obj.callCppFunction();

//obj.read_dir();//读取目录的方法

obj. read_file();//读取文件的方法

System.out.println("Javaoutput:" + obj.message);

}

}

4、进入java所在的目录src/demo/目录下执行javac MainClass.java(这个是刚刚写的java文件),生成一个MainClass.class文件 进入bin的上一层目录然后进入bin目录下,执行javah demo.MaicClass这时候在当前目录下生成一个MainClass.h文件,把这文件拷贝到写c的目录下。

头文件内容如下

Java代码,c的头文件相同

MainClass.h

/* DO NOT EDIT THIS FILE - it is machine generated */

#include

/* Header for classdemo_MainClass */

#ifndef_Included_demo_MainClass

#define_Included_demo_MainClass

#ifdef __cplusplus

extern "C"{

#endif

/*

* Class:demo_MainClass

* Method:callCppFunction

* Signature: ()V

*/

JNIEXPORT voidJNICALL Java_demo_MainClass_callCppFunction

(JNIEnv *, jobject);

/*

* Class:demo_MainClass

* Method:read_dir

* Signature: ()V

*/

JNIEXPORT void JNICALLJava_demo_MainClass_read_1dir

(JNIEnv *, jobject);//注意这里它会自动添加一个数字1

/*

* Class:demo_MainClass

* Method:read_file

* Signature: ()V

*/

JNIEXPORT void JNICALL Java_demo_MainClass_read_1file

(JNIEnv *, jobject);

#ifdef __cplusplus

}

#endif

#endif

5、在和.h同一目录下,创建一个MainClass.c文件文件内容如下

MainClass.c内容如下:

cpp代码

#include"demo_MainClass.h"

#include

#include

#include

#include

#include

#include

//读取目录函数

int cont = 0;

intdir_file_num(char * dirname)

{

printf("%sdir:\n", dirname);

DIR *dirp = opendir(dirname);//打开目录

if(dirp == NULL)

{

perror("open dir err");

return -1;

}

struct dirent * dentp = NULL;

//读取目录中所用文件

while((dentp = readdir(dirp)))

{

if(dentp->d_type == DT_REG)

{

printf("%s\n",dentp->d_name);

cont++;

}

if(dentp->d_type == DT_DIR)

{

if((strcmp(".",dentp->d_name)) ==0 || (strcmp("..", dentp->d_name)) == 0)

{

continue;

}

char new_dir_name[256] = {0};

sprintf(new_dir_name,"%s/%s",dirname, dentp->d_name);//directoryfull path

dir_file_num(new_dir_name);

}

}

return cont;

}

//读取文件前5个字节

intread_file_info(char *filename)

{

if(NULL == filename)

{

return -1;

}

printf("%s", filename);

FILE *fd;

size_t i = 0;

char buf[24] = {0};

charnew_file_name[256]={0};

sprintf(new_file_name, "%s.new",filename);

//读取文件前五个字节

if(!(fd = fopen(filename, "r")))

{

return -2;

}

i = fread(buf, 1, 5, fd);

if( i < 5)

{

printf("文件读取失败!\n");

return -3;

}

fclose(fd);

fd = fopen(new_file_name, "w");

if(fd == NULL)

{

printf("文件打开失败");

return -4;

}

i = fwrite(buf, 1, 5, fd);

if(i == 0)

{

printf("文件写入失败\n");

return -5;

}

fclose(fd);

return 0;

}

JNIEXPORT void

JNICALLJava_demo_MainClass_callCppFunction(JNIEnv *env, jobject obj)

{

jclass clazz;

jfieldID fid;

jstring j_str;

jstring j_newStr;

const char *c_str = NULL;

//获取MainClass类的Class引用

clazz = (*env)->GetObjectClass(env, obj);

if(NULL == clazz)

{

return;

}

//获取MainClass类实例变量message的属性ID

fid = (*env)->GetFieldID(env, clazz,"message", "Ljava/lang/String;");

if(fid < 0)

{

return;

}

//获取实例message的值

j_str = (jstring)(*env)->GetObjectField(env,obj, fid);

if(NULL == j_str)

{

return;

}

//将Unicode编码的java字符串转换为c风格的字符串

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

if(NULL == c_str)

{

return;

}

int i = 0;

//调用显示目录下文件的函数

i = dir_file_num((char *)c_str);

if(i < 0)

{

printf("dir_file_num err!\n");

return;

}

}

//打印目录下所用文件

JNIEXPORT void

JNICALLJava_demo_MainClass_read_1dir(JNIEnv *env, jobject obj)

{

jclass clazz;

jfieldID fid;

jstring j_str;

jstring j_newStr;

const char *c_str = NULL;

//获取MainClass类的Class引用

clazz = (*env)->GetObjectClass(env, obj);

if(NULL == clazz)

{

return;

}

//获取MainClass类实例变量message的属性ID

fid =(*env)->GetFieldID(env, clazz, "message","Ljava/lang/String;");

if(fid < 0)

{

return;

}

//获取实例message的值

j_str =(jstring)(*env)->GetObjectField(env, obj, fid);

if(NULL == j_str)

{

return;

}

//将Unicode编码的java字符串转换为c风格的字符串

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

if(NULL == c_str)

{

return;

}

int i = 0;

//调用显示目录下文件的函数

i = dir_file_num((char*)c_str);

if(i < 0)

{

printf("dir_file_numerr!\n");

return;

}

}

//读取文件前5个字节

JNIEXPORT void

JNICALLJava_demo_MainClass_read_1file(JNIEnv *env, jobject obj)

{

//定义变量

jclass clazz;

jfieldID fid;

jstring j_str;

jstring j_newStr;

const char *c_str = NULL;

//获取MainClass类的Class引用

clazz = (*env)->GetObjectClass(env, obj);

if(NULL == clazz)

{

return;

}

//获取MainClass类实例变量message的属性ID

fid = (*env)->GetFieldID(env, clazz,"message", "Ljava/lang/String;");

if(fid < 0)

{

return;

}

//获取实例message的值

j_str =(jstring)(*env)->GetObjectField(env, obj, fid);

if(NULL == j_str)

{

return;

}

//将Unicode编码的java字符串转换为c风格的字符串

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

if(NULL == c_str)

{

return;

}

int i = 0;

//调用读取文件函数

i = dir_file_num((char *)c_str);

if(i < 0)

{

printf("dir_file_num err!\n");

return;

}

}

6、代码写完之后开始制作动态库,首先执行

gcc -fPIC -D_REENTRANT-I/home/jdk1.8.0_77/include -I/home/jdk1.8.0_77/include/linux -cdemo_MainClass.c

生成一个.o文件,这里的/home/jdk1.8.0_77/include是我主机中jni.h所在路径,/home/jdk1.8.0_77/include/linux是jni.h所在路径,然后执行

gcc -shareddemo_MainClass.o -o libTestReadFile.so

生成一个libTestReadFile.so文件,这里的demo_MainClass.o就是上一步生成的.o文件。

7、这里就已经完成了在Ubuntu下中的eclipse调用.so文件了,这时候就可以在eclipse中执行了,然后输入一个目录路径就可以了。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以帮你解答这个问题。 首先,你需要在你的 Android 工程创建一个 `jni` 目录,并将 `jni_vlog_api.cpp` 和 `jni_vlog_api.hh` 文件放到该目录下。 接下来,你需要创建一个 `Android.bp` 文件,该文件描述了你的 `vlog.so` 库的构建规则。以下是一个简单的 `Android.bp` 文件示例: ``` cc_library { name: "vlog", srcs: ["jni/jni_vlog_api.cpp"], shared_libs: ["libstdc++"], include_dirs: ["jni"], target: { android: { cflags: ["-std=c++11"], shared_libs: ["liblog"], }, }, visibility: ["//visibility:public"], } ``` 上述 `Android.bp` 文件,我们定义了一个名为 `vlog` 的 `cc_library` 目标,它的源代码文件是 `jni/jni_vlog_api.cpp`。这个库依赖于 `libstdc++` 和 `liblog` 库。`include_dirs` 属性指定了编译器需要搜索头文件的目录。在这里,我们指定了 `jni` 目录。`target` 属性指定了我们要在 Android 平台上构建此库,并指定了一些特定于 Android 平台的构建选项。`visibility` 属性将库的可见性设置为公共。 最后,你需要运行 `mma` 命令(Make Module Android)来构建该库。在 Android Studio ,你可以通过选择 Build -> Make Module 'vlog',来构建该库。构建成功后,你可以在 `app/build/intermediates/cmake/debug/obj` 或 `app/build/intermediates/ndk/debug/lib` 目录找到生成的 `vlog.so` 库。 接下来,你可以在你的 Java 代码使用 `System.loadLibrary("vlog")` 来加载该库,并调用的函数。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值