由于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进制代码;
期君粘转!