因爲java層使用的是unicode編碼方式即UTF-16,而c/c++本地代碼使用的是UTF-8編碼方式,因此在JNI層返回16進制字符串的問題比較難搞,很少說,直接粘碼:java
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);
}
c++
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);
編碼
註釋一:spa
在計算機中,正數是直接用原碼錶示的,如單字節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;code
在C/C++中若是趕上這樣的十六進制代碼1000 0001,在經過JNI返回java的過程當中VM就會把它當成負數,轉化爲1111 1111 1000 0001,這樣就變成了-129;
unicode
前面說了JAVA採用的是UTF-16的編碼方式,咱們想要的是和C/C++本地代碼同樣的char類型十六進制buff串,必須在JAVA層獲得的字符&0xff後才能獲得和C/C++ 一樣的16進制代碼; 字符串
期君粘轉!
it