系統(tǒng)環(huán)境:64位Ubuntu14.04径荔、eclipse 5.3版本,jdk1.8版本,vim編譯器
1笤成、本實例是java代碼通過jni調(diào)用c/c++語言生成的.so庫,本庫提供兩個方法眷茁,當(dāng)調(diào)用讀目錄方法炕泳,并輸入一個目錄的路徑時,會打印出輸入目錄下的所有文件上祈,當(dāng)調(diào)用培遵,讀文件方法,并輸入一個目錄文件的路徑會讀取本文件的前五個字節(jié)登刺,并創(chuàng)建一個和輸入文件名相同后綴添加一個.new的文件籽腕,然后將讀取的五個字節(jié)寫入到本文件中。
2纸俭、在eclipse中創(chuàng)建一個類皇耗,流程如下File->New->Java Project->在Projectname框中寫入項目名稱->點擊finish。右擊右側(cè)欄的項目中的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中可使用這個函數(shù)然后直接寫so的絕對路徑加載so文件
System.load("/home/native/read_dir_file/libTestReadFile.so");
}
//在MainClass類中定義了一個String結(jié)構(gòu)體
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/目錄下執(zhí)行javac MainClass.java(這個是剛剛寫的java文件)呜袁,生成一個MainClass.class文件 進入bin的上一層目錄然后進入bin目錄下,執(zhí)行javah demo.MaicClass這時候在當(dāng)前目錄下生成一個MainClass.h文件简珠,把這文件拷貝到寫c的目錄下阶界。
頭文件內(nèi)容如下
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 void JNICALL Java_demo_MainClass_callCppFunction
(JNIEnv *, jobject);
/*
* Class:demo_MainClass
* Method:read_dir
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_demo_MainClass_read_1dir
(JNIEnv *, jobject);//注意這里它會自動添加一個數(shù)字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同一目錄下荐操,創(chuàng)建一個MainClass.c文件文件內(nèi)容如下
MainClass.c內(nèi)容如下:
cpp代碼
#include"demo_MainClass.h"
#include <stdio.h>
#include <stdlib.h>
#include <dirent.h>
#include <unistd.h>
#include <sys/types/h>
#include <string.h>
//讀取目錄函數(shù)
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個字節(jié)
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字符串轉(zhuǎn)換為c風(fēng)格的字符串
c_str = (*env)->GetStringUTFChars(env,j_str, NULL);
if(NULL == c_str)
{
return;
}
int i = 0;
//調(diào)用顯示目錄下文件的函數(shù)
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字符串轉(zhuǎn)換為c風(fēng)格的字符串
c_str = (*env)->GetStringUTFChars(env,j_str, NULL);
if(NULL == c_str)
{
return;
}
int i = 0;
//調(diào)用顯示目錄下文件的函數(shù)
i = dir_file_num((char*)c_str);
if(i < 0)
{
printf("dir_file_numerr!\n");
return;
}
}
//讀取文件前5個字節(jié)
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字符串轉(zhuǎn)換為c風(fēng)格的字符串
c_str = (*env)->GetStringUTFChars(env,j_str, NULL);
if(NULL == c_str)
{
return;
}
int i = 0;
//調(diào)用讀取文件函數(shù)
i = dir_file_num((char *)c_str);
if(i < 0)
{
printf("dir_file_num err!\n");
return;
}
}
6、代碼寫完之后開始制作動態(tài)庫珍策,首先執(zhí)行
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所在路徑攘宙,然后執(zhí)行
gcc -shareddemo_MainClass.o -o libTestReadFile.so
生成一個libTestReadFile.so文件屯耸,這里的demo_MainClass.o就是上一步生成的.o文件拐迁。
7、這里就已經(jīng)完成了在Ubuntu下中的eclipse調(diào)用.so文件了疗绣,這時候就可以在eclipse中執(zhí)行了线召,然后輸入一個目錄路徑就可以了。