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