關於JNI返回十六進制字符串到JAVA層的說明

因爲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

相關文章
相關標籤/搜索