关于JNI返回十六进制字符串到JAVA层的说明

由于java层使用的是unicode编码方式即UTF-16,而c/c++本地代码使用的是UTF-8编码方式,所以在JNI层返回16进制字符串的问题比较难搞,不多说,直接粘码:

JAVA部分代码:
public class MainActivity extends Activity {
    static {                                        
        System.loadLibrary("serialjni");                    
    }
    private byte[] bytes = new byte[12];

    private void callback() {
        int i;
        for(i=0; i<12; i++){
            System.out.println(bytes[i]&0xff);//见注释一
            if((bytes[i]&0xff) == 0xff){
                System.out.println("0xff\n");
                }
            }
        System.out.println(bytes);
    } 
 

JNI部分   
    bytearray = (*env)->NewByteArray(env,BUFSIZE+1);
    jclass cls1 = (*env)->GetObjectClass(env, obj);
    jclass cls2 = (*env)->GetObjectClass(env, obj);
    fid = (*env)->GetFieldID(env, cls1, "bytes", "[B");

    if (fid == NULL)
    {
            return; /* failed to find the field */
        }
        jmethodID mid = (*env)->GetMethodID(env, cls2, "callback", "()V");
        if (mid == NULL)
        {
            return; /* method not found */
        }


          (*env)->SetByteArrayRegion(env, bytearray, 0, BUFSIZE+1, buff1);
          (*env)->SetObjectField(env, obj, fid, bytearray);
          (*env)->CallVoidMethod(env, obj, mid);

                                             

注释一

在计算机中,正数是直接用原码表示的,如单字节5,在计算机中就表示为:0000 0101。负数用补码表示,如单字节-5,在计算机中表示为1111 1011。在C/C++中字节5编码为(UTF-8)0000 0101   -5编码为1111 1011,而在JAVA中5编码为(UTF-16)0000 0000 0000 0101, 而-5编码为1111 1111 1111 1011;

在C/C++中如果遇上这样的十六进制代码1000 0001,在通过JNI返回java的过程中VM就会把它当成负数,转化为1111 1111 1000 0001,这样就变成了-129;

前面说了JAVA采用的是UTF-16的编码方式,我们想要的是和C/C++本地代码一样的char类型十六进制buff串,必须在JAVA层得到的字符&0xff后才能得到和C/C++ 同样的16进制代码; 

期君粘转!

   
      
      


你可能感兴趣的:(关于JNI返回十六进制字符串到JAVA层的说明)