• JNI由浅入深_10_JNI 综合开发


    1、使用ndk-build时如果找不到某个类,可以使用下面两种方法解决:

    1、1 进入src目录

    D:project3JNIAndroidsrc>set classpath=D:project3JNIAndroidsrc

    1、2 设置classpath
    classpath    .;%JAVA_HOME%libdt.jar;%JAVA_HOME% ools.jar;E:developLibandroidplatformsandroid-8android.jar

    2、基础类

    ublic class NativeModule {
    
        public native int testArg(int i, boolean b, char c, double d);
    
        public native byte[] testByte(byte[] b);
    
        public native String[] testString(String s, String[] sarr);
    
        public native int setInfo(MyInfo info);
    
        public native MyInfo getInfo();
    
        static {
    
           System.loadLibrary("NativeModule");
    
        }
    
    }
    

    //其中MyInfo类定义如下:
    
    public class Record {
    
        int id;
    
        String name;
    
        byte[] data;
    
    }
    
    public class MyInfo {
    
        public boolean b;
    
        public char c;
    
        public double d;
    
        public int i;
    
        public byte[] array;
    
        public String s;
    
        public Record rec;
    
    }

    c 结构体定义

    typedef struct{
    
        int id;
    
        char name[255];
    
        char data[255];
    
    }Record;
    
     
    
    typedef struct{
    
        BOOL b;
    
        char c;
    
        double d;
    
        int i;
    
        char arr[255];
    
        char sz[255];
    
        Record rec;
    
    }MyInfo;

    3、编写C库

     3.1) Java与C不同类型参数转换实例

    //不同类型参数处理
    
    JNIEXPORT jintJNICALL Java_com_jinhill_util_NativeModule_testArg
    
      (JNIEnv *env, jobject jo, jint ji, jbooleanjb,  jchar jc, jdouble jd){
             //获取jint型值
             int i = ji;
             //获取jboolean型值
             BOOL b = jb;
             //获取jdouble型值
             double d = jd;
             //获取jchar型值,Java的char两字节
             char ch[5] = {0};
             int size = 0;
             size = WideCharToMultiByte(CP_ACP,NULL, (LPCWSTR)&jc, -1, ch, 5, NULL, FALSE);
             if(size <= 0) {
                       return -1;
             }
             Trace("ji=%d,jb=%d,jc=%s,jd=%lf",i, b, ch, d);
             return 0;
    
    }

     3.2) Java byte与C char数组类型数组转换实例

    //btye数组处理,形参作为输入或输出,返回btye数组
    
    JNIEXPORTjbyteArray JNICALL Java_com_jinhill_util_NativeModule_testByte
    
      (JNIEnv *env, jobject jo, jbyteArray jbArr){
             char chTmp[] = "Hello JNI!";
             int nTmpLen = strlen(chTmp);
             //获取jbyteArray
             char *chArr = (char*)env->GetByteArrayElements(jbArr,0);
             //获取jbyteArray长度
             int nArrLen = env->GetArrayLength(jbArr);
             char *szStrBuf =(char*)malloc(nArrLen*2+10);
             memset(szStrBuf, 0, nArrLen*2+10);
             Bytes2String(chArr, nArrLen, szStrBuf,nArrLen*2+10);
             Trace("jbArr=%s", szStrBuf);
             //将jbArr作为输出形参
             memset(chArr, 0, nArrLen);
             memcpy(chArr, chTmp, nTmpLen);
             //返回jbyteArray
             jbyteArray jarrRV =env->NewByteArray(nTmpLen); 
        jbyte *jby =env->GetByteArrayElements(jarrRV, 0);
             memcpy(jby, chTmp, strlen(chTmp));
        env->SetByteArrayRegion(jarrRV, 0,nTmpLen, jby); 
             return jarrRV;
    }

     3.3)   Java String与C char数组类型转换实例

     
    //String 和String[]处理
    
    JNIEXPORTjobjectArray JNICALL Java_com_jinhill_util_NativeModule_testString
    
      (JNIEnv *env, jobject jo, jstring jstr,jobjectArray joarr){
             int i = 0;
    
             char chTmp[50] = {0};
    
             //获取jstring值
    
             const char* pszStr = (char*)env->GetStringUTFChars(jstr, 0);
    
             Trace("jstr=%s", pszStr);
    
             //获取jobjectArray值
    
             int nArrLen =env->GetArrayLength(joarr);
    
             Trace("joarr len=%d",nArrLen);
    
             for(i=0; i<nArrLen; i++) {
                       jstring js =(jstring)env->GetObjectArrayElement(joarr, i);
                       const char* psz = (char*)env->GetStringUTFChars(js, 0);
                       Trace("joarr[%d]=%s",i, psz);
             }
    
             //将joarr作为输出形参
    
             jstring jstrTmp = NULL;
    
             for(i=0; i<nArrLen; i++) {
                       sprintf(chTmp, "No.%dHello JNI!", i);
                       jstrTmp =env->NewStringUTF(chTmp);
    
                       env->SetObjectArrayElement(joarr,i, jstrTmp);
                       env->DeleteLocalRef(jstrTmp);
             }
    
             //返回jobjectArray
    
             jclass jstrCls =env->FindClass("Ljava/lang/String;");
    
             jobjectArray jstrArray =env->NewObjectArray(2, jstrCls, NULL);
    
             for(i=0; i<2; i++) {
                       sprintf(chTmp, "No. %dReturn JNI!", i);
                       jstrTmp =env->NewStringUTF(chTmp);
                       env->SetObjectArrayElement(jstrArray,i, jstrTmp);
                       env->DeleteLocalRef(jstrTmp);
             }
    
             return jstrArray;
    
    }

     3.4)   Java 类与C结构体类型转换实例

    JNIEXPORT jint JNICALL Java_com_jinhill_util_NativeModule_setInfo
    
      (JNIEnv *env, jobject jo, jobject jobj){
    
        char chHexTmp[512] = {0};
    
        //将Java类转换成C结构体
    
        MyInfo mi;
    
        //获取Java中的实例类Record
    
        jclass jcRec = env->FindClass("com/jinhill/util/Record");
    
        //int id
    
        jfieldID jfid = env->GetFieldID(jcRec, "id", "I");
    
        //String name
    
        jfieldID jfname = env->GetFieldID(jcRec, "name", "Ljava/lang/String;");
    
        //byte[] data;
    
        jfieldID jfdata = env->GetFieldID(jcRec, "data", "[B");
    
        //获取Java中的实例类MyInfo
        jclass jcInfo = env->FindClass("com/jinhill/util/MyInfo");
    
        //获取类中每一个变量的定义
        //boolean b
        jfieldID jfb = env->GetFieldID(jcInfo, "b", "Z");
    
        //char c
        jfieldID jfc = env->GetFieldID(jcInfo, "c", "C");
    
        //double d
        jfieldID jfd = env->GetFieldID(jcInfo, "d", "D");
    
        //int i
        jfieldID jfi = env->GetFieldID(jcInfo, "i", "I");
    
        //byte[] array
        jfieldID jfa = env->GetFieldID(jcInfo, "array", "[B");
    
        //String s
        jfieldID jfs = env->GetFieldID(jcInfo, "s", "Ljava/lang/String;");
    
        //Record rec;
        jfieldID jfrec = env->GetFieldID(jcInfo, "rec", "Lcom/jinhill/util/Record;");
    
     
    
        //获取实例的变量b的值
        mi.b = env->GetBooleanField(jobj, jfb);
    
        //获取实例的变量c的值
        jchar jc = env->GetCharField(jobj, jfc);
    
        char ch[5] = {0};
    
        int size = 0;
    
        size = WideCharToMultiByte(CP_ACP, NULL, (LPCWSTR)&jc, -1, ch, 5, NULL, FALSE);
    
        mi.c = ch[0];
    
        //获取实例的变量d的值
        mi.d = env->GetDoubleField(jobj, jfd);
    
        //获取实例的变量i的值
        mi.i = env->GetIntField(jobj, jfi);
    
        //获取实例的变量array的值
        jbyteArray ja = (jbyteArray)env->GetObjectField(jobj, jfa);
    
        int  nArrLen = env->GetArrayLength(ja);
    
        char *chArr = (char*)env->GetByteArrayElements(ja, 0);
    
        memcpy(mi.arr, chArr, nArrLen);
    
        //获取实例的变量s的值
        jstring jstr = (jstring)env->GetObjectField(jobj, jfs);
    
        const char* pszStr = (char*)env->GetStringUTFChars(jstr, 0);
    
        strcpy(mi.sz, pszStr);
    
     
    
        //获取Record对象
    
        jobject joRec = env->GetObjectField(jobj, jfrec);
    
        //获取Record对象id值
        mi.rec.id = env->GetIntField(joRec, jfid);
    
        Trace("mi.rec.id=%d",mi.rec.id);
    
        //获取Record对象name值
        jstring jstrn = (jstring)env->GetObjectField(joRec, jfname);
    
        pszStr = (char*)env->GetStringUTFChars(jstrn, 0);
    
        strcpy(mi.rec.name, pszStr);
    
        //获取Record对象data值
        jbyteArray jbd = (jbyteArray)env->GetObjectField(joRec, jfdata);
        nArrLen = env->GetArrayLength(jbd);
        chArr = (char*)env->GetByteArrayElements(jbd, 0);
        memcpy(mi.rec.data, chArr, nArrLen);
    
        //日志输出
    
        Bytes2String(mi.arr, nArrLen, chHexTmp, sizeof(chHexTmp)); 
       Trace("mi.arr=%s, mi.b=%d, mi.c=%c, mi.d=%lf, mi.i=%d, 
     mi.sz=%s
     mi.rec.id=%d, mi.rec.name=%s", chHexTmp, mi.b, mi.c, mi.d, mi.i, mi.sz, mi.rec.id, mi.rec.name);
         return 0;
    
    }

     3.5)   C结构体类型与Java 类转换实例 

     
    JNIEXPORT jobject JNICALL Java_com_jinhill_util_NativeModule_getInfo
    
      (JNIEnv *env, jobject jo){
       wchar_t wStr[255] = {0};
        char chTmp[] = "Hello JNI";
        int nTmpLen = strlen(chTmp);
        //将C结构体转换成Java类
        MyInfo mi;
        memcpy(mi.arr, chTmp, strlen(chTmp));
        mi.b = TRUE;
        mi.c = 'B';
        mi.d = 2000.9;
        mi.i = 8;
    
        strcpy(mi.sz, "Hello World!");
        mi.rec.id = 2011;
        memcpy(mi.rec.data, "x01x02x03x04x05x06", 6);
        strcpy(mi.rec.name, "My JNI");
    
        //获取Java中的实例类Record
        jclass jcRec = env->FindClass("com/jinhill/util/Record");
    
        //int id
        jfieldID jfid = env->GetFieldID(jcRec, "id", "I");
    
        //String name
        jfieldID jfname = env->GetFieldID(jcRec, "name", "Ljava/lang/String;");
    
        //byte[] data;
        jfieldID jfdata = env->GetFieldID(jcRec, "data", "[B");
     
    
        //获取Java中的实例类
        jclass jcInfo = env->FindClass("com/jinhill/util/MyInfo");
    
        //获取类中每一个变量的定义
        //boolean b
        jfieldID jfb = env->GetFieldID(jcInfo, "b", "Z");
    
        //char c
        jfieldID jfc = env->GetFieldID(jcInfo, "c", "C");
    
        //double d
        jfieldID jfd = env->GetFieldID(jcInfo, "d", "D");
    
        //int i
        jfieldID jfi = env->GetFieldID(jcInfo, "i", "I");
    
        //byte[] array
        jfieldID jfa = env->GetFieldID(jcInfo, "array", "[B");
    
        //String s
        jfieldID jfs = env->GetFieldID(jcInfo, "s", "Ljava/lang/String;");
    
        //Record rec;
        jfieldID jfrec = env->GetFieldID(jcInfo, "rec", "Lcom/jinhill/util/Record;");
    
     
    
        //创建新的对象
        jobject joRec = env->AllocObject(jcRec);
    
     
    
        env->SetIntField(joRec, jfid, mi.rec.id);
    
     
    
        jstring jstrn = env->NewStringUTF(mi.rec.name);
    
        env->SetObjectField(joRec, jfname, jstrn);
    
     
    
        jbyteArray jbarr = env->NewByteArray(6); 
    
        jbyte *jb = env->GetByteArrayElements(jbarr, 0);
    
        memcpy(jb, mi.rec.data, 6);
    
        env->SetByteArrayRegion(jbarr, 0, 6, jb);
    
        env->SetObjectField(joRec, jfdata, jbarr);
    
        //创建新的对象
        jobject joInfo = env->AllocObject(jcInfo);
    
         //给类成员赋值
        env->SetBooleanField(joInfo, jfb, mi.b);
    
    
    //  MultiByteToWideChar (CP_ACP, 0, mi.c, -1, wStr, 255);
    //  env->SetCharField(joInfo, jfc, (jchar)wStr);
    
        env->SetCharField(joInfo, jfc, (jchar)mi.c);
    
        env->SetDoubleField(joInfo, jfd, mi.d);
    
        env->SetIntField(joInfo, jfi, mi.i);
    
        jbyteArray jarr = env->NewByteArray(nTmpLen); 
    
        jbyte *jby = env->GetByteArrayElements(jarr, 0);
        memcpy(jby, mi.arr, nTmpLen);
    
        env->SetByteArrayRegion(jarr, 0, nTmpLen, jby);
        env->SetObjectField(joInfo, jfa, jarr);
        jstring jstrTmp = env->NewStringUTF(chTmp);
        env->SetObjectField(joInfo, jfs, jstrTmp);
    
        env->SetObjectField(joInfo, jfrec, joRec);
        return joInfo;
    
    }
    

     3.6)   Java String与 C char数组转换时的中文问题

    //将jstring类型转换成windows类型
    char* jstringToWindows( JNIEnv *env, jstring jstr )
    {
    <span style="white-space:pre">	</span>int length = (env)->GetStringLength(jstr );
    <span style="white-space:pre">	</span>const jchar* jcstr = (env)->GetStringChars(jstr, 0 );
    <span style="white-space:pre">	</span>char* rtn = (char*)malloc( length*2+1 );
    <span style="white-space:pre">	</span>int size = 0;
    <span style="white-space:pre">	</span>size = WideCharToMultiByte( CP_ACP, 0, (LPCWSTR)jcstr, length,rtn,(length*2+1), NULL, NULL );
    <span style="white-space:pre">	</span>if( size <= 0 )
    <span style="white-space:pre">		</span>return NULL;
    <span style="white-space:pre">	</span>(env)->ReleaseStringChars(jstr, jcstr );
    <span style="white-space:pre">	</span>rtn[size] = 0;
    <span style="white-space:pre">	</span>return rtn;
    }
    
    //将windows类型转换成jstring类型
    jstring WindowsTojstring( JNIEnv* env, char* str )
    {
    <span style="white-space:pre">	</span>jstring rtn = 0;
    <span style="white-space:pre">	</span>int slen = strlen(str);
    <span style="white-space:pre">	</span>unsigned short * buffer = 0;
    <span style="white-space:pre">	</span>if( slen == 0 )
    <span style="white-space:pre">		</span>rtn = (env)->NewStringUTF(str );
    <span style="white-space:pre">	</span>else
    <span style="white-space:pre">	</span>{
    <span style="white-space:pre">		</span>int length = MultiByteToWideChar( CP_ACP, 0, (LPCSTR)str, slen, NULL, 0 );
    <span style="white-space:pre">		</span>buffer = (unsigned short *)malloc( length*2 + 1 );
    <span style="white-space:pre">		</span>if( MultiByteToWideChar( CP_ACP, 0, (LPCSTR)str, slen, (LPWSTR)buffer, length )>0 )
    <span style="white-space:pre">			</span>rtn = (env)->NewString( (jchar*)buffer, length );
    <span style="white-space:pre">	</span>}
    <span style="white-space:pre">	</span>if( buffer )
    <span style="white-space:pre">		</span>free( buffer );
    <span style="white-space:pre">	</span>return rtn;
    }


  • 相关阅读:
    OS + UNIX AIX performance
    web test LoadRunner fuction_list
    web test LoadRunner Linux
    web test LoadRunner SAP / java / Java Vuser / web_set_max_html_param_len
    网上购物系统(Task006)——数据访问层DAL
    Request 获取网址各片段
    网上购物系统(Task009)——FormView显示商品详细信息
    网上购物系统(Task007)——自定义DateList控件分页显示商品信息
    网上购物系统(Task010)——FormView编辑更新商品详细信息
    网上购物系统(Task011)——FormView插入删除商品详细信息
  • 原文地址:https://www.cnblogs.com/lbangel/p/4335853.html
Copyright © 2020-2023  润新知