Android+JNI调用–文件操作

开发环境:Windows xp sp3 +MyEclipse 8.6+android2.3.3+jdk1.6+android-ndk-r6b

JNI概述:

JNI 是 Java Native Interface 的缩写,译为 Java 本地接口。它允许 Java 代码和其他语言编写的代码进行交互。在android 中提供 JNI 的方式,让 Java 程序可以调用 C/C++语言程序。 android 中很多 Java 类都具有 native 接口,这些接口由本地实现,然后注册到系统中。在android系统中实现JNI库需要连接.so共享库,如:lib<文件名>.so。

注意权限

Android NDK概述:
Android NDK是一个工具集,让你的Android应用程序里可以内嵌使用本地代码(C/C++)的组件。
Android应用程序运行在Dalvik虚拟机中。NDK可以让你使用C/C++这样的本地代码语言来实现你的应用程序中某些部分。这对某类程序是有帮助的,比如需要重用已有的C代码,或者为了提高运行速度。

NDK 提供:
1). 编译文件和工具集,用来将你的C/C++源文件编译成本地库。
2). 提供一种方式,将对应的本地库内嵌到应用程序包文件(.apk)中,最终发布到Android设备中。
3). 本地系统头文件和库,这些头文件和库从Android1.5开始往后都是被支持的。但使用本地活动(native activity)的程序只能运行在Android 2.3或更高的系统中。
4). 文档、示例、指南。
本例中JNI调用大概流程图如下:


1.编写Android JNI模块java调用类

Android虚拟机允许你的应用程序源代码通过JNI调用实现本地代码的方法,需要在应用程序中使用关键字native声明一个或多个方法表明该方法是通过本地调用实现的,如:

public native static int FileOpen(StringpFileName,int openMode);
public native static int FileLength(int fp);
public native static int FileSeek(int fp,int offset,int origin);
public native static CusBuffer FileRead(int fp,int nCount);
public native static int FileWrite(int fp,byte[] buf,int nCount);
public native static int FileClose(int fp);

除了声明native方法以外还必须为这些方法实现提供本地共享库,该共享库最终会被打包到.apk文件中,这些共享库需要更具标准的unix公约来命名lib<文件名>.so,如:libJNI_FileSys.so,其中JNI_FileSys使我们需要加载的库名。在应用程序中加载共享库的方法为:
static

{
System.loadLibrary("JNI_FileSys");
}

注:这里使用的文件名不需要lib前缀以及.so后缀名。

FileSys.java完整代码实现

package com.luoxudong.jni.reader;
import
com.luoxudong.jni.bean.CusBuffer;

/********************************************************************
* [Summary]

* TODO 文件操作类
* [Remarks]
* TODO 请在此处详细描述类的功能、调用方法、注意事项、以及与其它类的关系.
********************************************************************/
public class
FileSys {
static
{
System.loadLibrary("JNI_FileSys");
}

/**
*
* [Summary]
* MjFileOpen 打开文件
* @param strFileName 文件名
* @param openMode 打开类型
* @return 结果
*
*/
public int MjFileOpen(String strFileName,int openMode){
return FileOpen(strFileName, openMode);
}
/**
*
* [Summary]
* MjFileLength 计算文件长度
* @param fp 文件句柄
* @return 文件长度
*
*/
public int MjFileLength(int fp){
return FileLength(fp);
}

/**
*
* [Summary]
* MjFileSeek 文件seek操作
* @param fp 文件句柄
* @param offset 读取数据偏移量
* @param origin 开始位置指针
* @return
*
*/
public int MjFileSeek(int fp,int offset,int origin){
return FileSeek(fp, offset, origin);
}
/**
*
* [Summary]
* MjFileRead 读取文件数据
* @param fp 文件句柄
* @param nCount 读取字节数
* @return 实际读取字节数
*
*/
public CusBuffer MjFileRead(int fp,int nCount){
return FileRead(fp, nCount);
}
/**
*
* [Summary]
* MjFileWrite 写文件
* @param fp 文件句柄
* @param buf 写数据buffer
* @param nCount 需要写入的字节数
* @return 实际写入字节数
*
*/
public int MjFileWrite(int fp,byte[] buf,int nCount){
return FileWrite(fp, buf, nCount);
}
/**
*
* [Summary]
* MjFileClose 关闭文件
* @param fp 文件句柄
* @return 关闭文件状态
*
*/
public int MjFileClose(int fp){
return FileClose(fp);
}
//本地调用
public native static int FileOpen(String pFileName,int openMode);
public native static int FileLength(int fp);
public native static int FileSeek(int fp,int offset,int origin);
public native static CusBuffer FileRead(int fp,int nCount);
public native static int FileWrite(int fp,byte[] buf,int nCount);
public native static int FileClose(int fp);
}
2.实现本地方法调用接口
为了方便我们可以使用javah命令先生成对应C/C++语言中的.h然后再实现这些函数。FileSys.java编译成FileSys.class文件后,使用命令(当前目录为工程bin目录下)javah -jni com.luoxudong.jni.reader.FileSys,此时会在bin目录下生成一个.h文件,文件名格式如下:com_luoxudong_jni_reader_FileSys.h,为了方便本人把文件名改成JNI_FileSys.h。

JNI_FileSys.h代码:
/* DO NOT EDIT THISFILE - it is machine generated */
#include<jni.h>
/* Header for classcom_meijin_dict_reader_FileSys */

#ifndef_Included_com_meijin_dict_reader_FileSys
#define_Included_com_meijin_dict_reader_FileSys
#ifdef __cplusplus
extern"C" {
#endif
/*
* Class:com_meijin_dict_reader_FileSys
* Method:FileOpen
* Signature: (Ljava/lang/String;I)I
*/
JNIEXPORT jintJNICALL Java_com_meijin_dict_reader_FileSys_FileOpen
(JNIEnv *, jclass, jstring, jint);

/*
* Class:com_meijin_dict_reader_FileSys
* Method:FileLength
* Signature: (I)I
*/
JNIEXPORT jintJNICALL Java_com_meijin_dict_reader_FileSys_FileLength
(JNIEnv *, jclass, jint);

/*
* Class:com_meijin_dict_reader_FileSys
* Method:FileSeek
* Signature: (III)I
*/
JNIEXPORT jintJNICALL Java_com_meijin_dict_reader_FileSys_FileSeek
(JNIEnv *, jclass, jint, jint, jint);

/*
* Class:com_meijin_dict_reader_FileSys
* Method:FileRead
* Signature:(II)Lcom/meijin/dict/bean/CusBuffer;
*/
JNIEXPORT jobjectJNICALL Java_com_meijin_dict_reader_FileSys_FileRead
(JNIEnv *, jclass, jint, jint);

/*
* Class:com_meijin_dict_reader_FileSys
* Method:FileWrite
* Signature: (I[BI)I
*/
JNIEXPORT jintJNICALL Java_com_meijin_dict_reader_FileSys_FileWrite
(JNIEnv *, jclass, jint, jbyteArray, jint);

/*
* Class:com_meijin_dict_reader_FileSys
* Method:FileClose
* Signature: (I)I
*/
JNIEXPORT jintJNICALL Java_com_meijin_dict_reader_FileSys_FileClose
(JNIEnv *, jclass, jint);

#ifdef __cplusplus
}
#endif
#endif
其中JNIEXPORT和JNICALL两个宏是JNI的关键字,表示该函数需要被JNI调用,而jint,jstring,jbyteArray是以JNI为中介使JAVA中对应类型与本地类型对接的类型,jobject为需要返回的java对象,类型对应表如下:

Java类型

本地类型

字节(bit)

boolean

jboolean

8, unsigned

byte

jbyte

8

char

jchar

16, unsigned

short

jshort

16

int

jint

32

long

jlong

64

float

jfloat

32

double

jdouble

64

void

void

n/a

3.本地接口实现:
根据对应的.h文件实现其接口。
JNI_FileSys.c代码:
#include"JNI_FileSys.h"
#define LOG_TAG"JNI_FileSys"
#define LOGI(...)__android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__)
#define LOGE(...)__android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__)
#include<android/log.h>
#include"FileSys.h"

/*
* Class:com_luoxudong_jni_reader_FileSys
* Method:FileOpen
* Signature: ([BI)I
*/
JNIEXPORT jintJNICALL Java_com_luoxudong_jni_reader_FileSys_FileOpen
(JNIEnv *env, jclass jobj, jstring pFileName,jint openMode)
{
UINT8 *pbyFileName = (UINT8*)(*env)->GetStringUTFChars(env, pFileName, 0);
LOGI("file name:%s---opentype:%d", pbyFileName, openMode);
return FileOpen(pbyFileName,openMode);

}

/*
* Class:com_luoxudong_jni_reader_FileSys
* Method:FileLength
* Signature: (I)I
*/
JNIEXPORT jintJNICALL Java_com_luoxudong_jni_reader_FileSys_FileLength
(JNIEnv *env, jclass jobj, jint fd)
{
return FileLength(fd);
}

/*
* Class:com_luoxudong_jni_reader_FileSys
* Method:FileSeek
* Signature: (III)I
*/
JNIEXPORT jintJNICALL Java_com_luoxudong_jni_reader_FileSys_FileSeek
(JNIEnv *env, jclass jobj, jint fd, jintoffset, jint origin)
{

return FileSeek(fd, offset,origin);
}

/*
* Class:com_luoxudong_jni_reader_FileSys
* Method:FileRead
* Signature:(II)Lcom/luoxudong/jni/bean/CusBuffer;
*/
JNIEXPORT jobjectJNICALL Java_com_luoxudong_jni_reader_FileSys_FileRead
(JNIEnv *env, jclass jobj, jint fd, jintcount)
{
int nReadLen = 0;
UINT8 *pBuf = (UINT8*)malloc(count);
memset(pBuf, 0, count);

nReadLen = FileRead(fd, pBuf,count);

jbyte *pBy = (jbyte *)pBuf;
jbyteArray jarray =(*env)->NewByteArray(env, nReadLen);
(*env)->SetByteArrayRegion(env,jarray, 0, nReadLen, pBy);
jclass m_cls = (*env)->FindClass(env,"com/luoxudong/jni/bean/CusBuffer");
jmethodID m_mid =(*env)->GetMethodID(env, m_cls, "<init>", "()V");
jfieldID m_fid1 = (*env)->GetFieldID(env, m_cls,"buffer", "[B");
jfieldID m_fid2 = (*env)->GetFieldID(env, m_cls,"nBufferLen", "I");

jobject m_obj = (*env)->NewObject(env, m_cls,m_mid);
(*env)->SetObjectField(env,m_obj, m_fid1, jarray);
(*env)->SetIntField(env, m_obj,m_fid2, nReadLen);
return m_obj;
}

/*
* Class:com_luoxudong_jni_reader_FileSys
* Method:FileWrite
* Signature: (I[BI)I
*/
JNIEXPORT jintJNICALL Java_com_luoxudong_jni_reader_FileSys_FileWrite
(JNIEnv *env, jclass jobj, jint fd,jbyteArray buf, jint count)
{
jbyte *pjb = (jbyte*)(*env)->GetByteArrayElements(env, buf, 0);
jsize len =(*env)->GetArrayLength(env, buf);
UINT8 *byBuf = (UINT8 *)pjb;
pjb[len] = '\0';
return FileWrite(fd, byBuf,count);
}

/*
* Class:com_luoxudong_jni_reader_FileSys
* Method:FileClose
* Signature: (I)I
*/
JNIEXPORT jintJNICALL Java_com_luoxudong_jni_reader_FileSys_FileClose
(JNIEnv *env, jclass jobj, jint fd)
{
return FileClose(fd);
}
其中头部定义了些andriod中日志输出所需要的宏,以及其他关联的.h文件,在一些地方C跟C++的使用语法不太一样,如在C中调用UINT8 *pbyFileName = (UINT8*)(*env)->GetStringUTFChars(env, pFileName, 0),
而C++中的语法为env-> (UINT8 *)GetStringUTFChars(pFileName,0)。

4.生成共享库
把编写好的各种相关联代码放在一个文件夹中,编写android.mk文件,使用ndk生成libJNI_FileSys.so文件。


会在libs\armeabi目录下生成.so文件


5.应用程序调用
把生成好的libJNI_FileSys.so文件放入java工程的libs目录下,就可以使用了


6.当然在android下读写文件时还需要配置权限
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>

7.运行效果:


转载请指明出处:http://blog.csdn.net/rohsuton/article/details/6865628

附上完整源代码 android JNI调用-文件操作


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 要在Android应用程序中调用JNI操作指示灯,您需要完成以下步骤: 1.编写一个C或C++文件,其中包含您要执行的JNI操作。 2.将C或C++文件编译为共享库(.so文件)。 3.将共享库文件复制到您的Android项目的jniLibs目录中。 4.在Android项目中创建一个Java类,其中包含本地方法声明,并使用native关键字标记本地方法。 5.在Java类中加载共享库文件,并使用System.loadLibrary()方法加载库文件。 6.使用您在C或C++文件中编写的JNI代码实现本地Java方法。 7.最后,在您的Android应用程序中调用本地Java方法即可。 以下是一个简单的示例,说明如何在Android应用程序中使用JNI操作指示灯: C/C++文件: ```c #include <jni.h> #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <fcntl.h> JNIEXPORT void JNICALL Java_com_example_testjni_MainActivity_turnOnLed(JNIEnv *env, jobject obj) { int fd; fd = open("/sys/class/leds/led1/brightness", O_WRONLY); if (fd != -1) { write(fd, "1", 1); close(fd); } } JNIEXPORT void JNICALL Java_com_example_testjni_MainActivity_turnOffLed(JNIEnv *env, jobject obj) { int fd; fd = open("/sys/class/leds/led1/brightness", O_WRONLY); if (fd != -1) { write(fd, "0", 1); close(fd); } } ``` Java类: ```java package com.example.testjni; import android.support.v7.app.AppCompatActivity; import android.os.Bundle; public class MainActivity extends AppCompatActivity { static { System.loadLibrary("led"); } public native void turnOnLed(); public native void turnOffLed(); @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); turnOnLed(); // 延迟2秒 try { Thread.sleep(2000); } catch (InterruptedException e) { e.printStackTrace(); } turnOffLed(); } } ``` 在此示例中,我们使用JNI打开和关闭指示灯。在C/C++文件中,我们打开/sys/class/leds/led1/brightness文件,并将其设置为“1”或“0”,以打开或关闭LED灯。在Java类中,我们使用System.loadLibrary()方法加载共享库文件,并使用native关键字标记本地方法。然后,在onCreate()方法中,我们调用turnOnLed()方法打开LED灯,并在2秒后调用turnOffLed()方法关闭LED灯。 请注意,在使用JNI时,您需要小心处理内存分配和释放,以避免内存泄漏和其他问题。 ### 回答2: 在Android调用JNI操作指示灯需要经过以下几个步骤: 1. 首先,在Android上需要创建一个JNI接口来与底层硬件交互。使用JNI接口可以连接Java层和底层C/C++代码。可以在JNI接口中定义对应的函数来操作指示灯,比如打开和关闭指示灯。 2. 在Java层中,通过加载和调用JNI接口来实现对指示灯的控制。在Java代码中使用`System.loadLibrary("your_jni_library")`方法来加载JNI库。 3. 在C/C++层,实现JNI接口中定义的函数。可以使用底层硬件通信库,比如WiringPi、OpenCV等,在C/C++代码中操作底层硬件。 4. 在底层C/C++代码中,通过对指示灯所在的引脚进行驱动控制,来打开或关闭指示灯。具体的操作方式和硬件平台相关,可以通过GPIO库或相应的驱动程序来实现。 5. 在调用JNI的适当时机,可以在Android应用中的某个事件触发时调用这些JNI方法,来操作指示灯。比如在按钮点击事件发生时,调用JNI方法来打开或关闭指示灯。 需要注意的是,操作指示灯需要获取一定的硬件权限,可以通过在AndroidManifest.xml文件中声明相应的权限来获取。 通过以上步骤,我们就可以在Android上通过JNI操作指示灯了。当调用对应的JNI方法时,就可以控制指示灯的亮灭状态。 ### 回答3: 在Android调用JNI操作指示灯可以通过以下步骤实现: 首先,编写一个JNI的C/C++函数,用于控制指示灯的开关。可以使用特定的硬件库函数或者系统调用函数来控制指示灯的状态。这个函数需要传入一个参数,用于指定指示灯的状态,比如开启、关闭或者闪烁。 然后,在Android应用的Java代码中通过`System.loadLibrary()`方法加载之前编写好的JNI库,以便可以调用JNI函数。将该JNI函数封装到一个Java方法中,并为该方法添加`native`关键字,表示该方法是通过JNI调用的。例如: ```java public class LedController { static { System.loadLibrary("led_controller"); } public native static void controlLed(int state); // Other methods and code here... } ``` 接下来,在JNI的C/C++代码中实现`controlLed()`方法。例如,可以通过GPIO库或者Linux系统调用函数来控制指示灯的状态。根据传入的参数,调用相应的函数操作指示灯。示例代码如下: ```c #include <jni.h> // 其他可能需引用的头文件 JNIEXPORT void JNICALL Java_com_example_ledcontroller_LedController_controlLed(JNIEnv *env, jclass type, jint state) { // 执行对指示灯的控制操作 if (state == 0) { // 关闭指示灯 } else if (state == 1) { // 开启指示灯 } else if (state == 2) { // 闪烁指示灯 } } ``` 最后,在Android应用的Java代码中调用`controlLed()`方法来控制指示灯的状态。例如: ```java LedController.controlLed(1); // 开启指示灯 ``` 通过这样的步骤,就可以在Android应用中调用JNI函数来操作指示灯。当调用`controlLed()`方法时,会通过JNI加载C/C++代码并执行指定的指示灯操作

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值