2016-10-29 30 views
0

我正試圖用C++重寫這個JavaScript base64編碼例程(請注意,它是非標準的base64,並且在解碼字符串的開頭有一個。在C++中按位移動Base64編碼功能

這裏是JS腳本的例子 - https://jsfiddle.net/km53844e/1/

這裏是C++腳本的例子 - http://www.tutorialspoint.com/compile_cpp11_online.php?PID=0Bw_CjBb95KQMZ0d5UHc0enhRbXM

我已張貼下面的JavaScript的base64類。

在JS腳本中,它正確地將CcnK轉換爲CMlaKA。然而,在C++腳本中,它錯誤地將其轉換爲CMlaKr(不知道爲什麼,它可能與JS腳本中的附加NaN語句有關?

或者是否可能與null終止的字符串在C++腳本嗎?我發現,如果我刪除此,我得到CMlaKs這仍然是不正確的。

我試着加入以下應對isnans,但它不工作。

if (isnan(char_array_4[1])) { 
      char_array_4[2] = char_array_4[1] = 64; 
     } else if (isnan(char_array_4[2])) { 
      char_array_4[3] = 64; 
     } 

C++代碼:

std::string base64_encode(unsigned char const* bytes_to_encode, unsigned int in_len) { 
    std::string ret; 
    int i = 0; 
    int j = 0; 
    unsigned char char_array_3[3]; 
    unsigned char char_array_4[4]; 

    while (in_len--) { 
    char_array_3[i++] = *(bytes_to_encode++); 
    if (i == 3) { 
     char_array_4[0] = char_array_3[0] & 0x3f; 
     char_array_4[1] = ((char_array_3[0] & 0x0f) << 2) + ((char_array_3[1] & 0xc0) >> 6); 
     char_array_4[2] = ((char_array_3[1] & 0x03) << 4) + ((char_array_3[2] & 0xf0) >> 4); 
     char_array_4[3] = (char_array_3[2] & 0xfc) >> 2; 

     if (isnan(char_array_4[1])) { 
      char_array_4[2] = char_array_4[1] = 64; 
     } else if (isnan(char_array_4[2])) { 
      char_array_4[3] = 64; 
     } 

     for(i = 0; (i < 4) ; i++) 
     ret += base64_chars[char_array_4[i]]; 
     i = 0; 
    } 
    } 

    if (i) 
    { 
    for(j = i; j < 3; j++) 
    char_array_3[j] = '\0'; 

    char_array_4[0] = char_array_3[0] & 0x3f; 
    char_array_4[1] = ((char_array_3[0] & 0x0f) << 2) + ((char_array_3[1] & 0xc0) >> 6); 
    char_array_4[2] = ((char_array_3[1] & 0x03) << 4) + ((char_array_3[2] & 0xf0) >> 4); 
    char_array_4[3] = (char_array_3[2] & 0xfc) >> 2; 

     if (isnan(char_array_4[1])) { 
      char_array_4[2] = char_array_4[1] = 64; 
     } else if (isnan(char_array_4[2])) { 
      char_array_4[3] = 64; 
     } 

    for (j = 0; (j < i + 1); j++) 
     ret += base64_chars[char_array_4[j]]; 

    while((i++ < 3)) 
     ret += '='; 

    } 

    return ret; 

} 

JS代碼:

var Base64 = { 
     _keyStr: ".ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz=", 

    encode : function (input) { 
     var output = [], 
      chr1, chr2, chr3, enc1, enc2, enc3, enc4, 
      i = 0; 
     while (i < input.length) { 
      chr1 = input[i++]; 
      chr2 = input[i++]; 
      chr3 = input[i++]; 

      enc1 = chr1 & 0x3f; 
      enc2 = (chr1 >> 6) | ((chr2 & 0xf) << 2); 
      enc3 = (chr2 >> 4) | ((chr3 & 0x3) << 4); 
      enc4 = chr3 >> 2; 

      if (isNaN(chr2)) { 
       enc3 = enc4 = 64; 
      } else if (isNaN(chr3)) { 
       enc4 = 64; 
      } 

      output.push([this._keyStr.charAt(enc1), 
         this._keyStr.charAt(enc2), 
         this._keyStr.charAt(enc3), 
         this._keyStr.charAt(enc4)].join('')); 
     } 

     return output.join(''); 
    }, 

    decodeAsArray: function (b) { 
     var d = this.decode(b), 
      a = [], 
      c; 
       //alert("decoded base64:" + d); 
     for (c = 0; c < d.length; c++) { 
      a[c] = d.charCodeAt(c) 
     } 
       //alert("returning a"); 
     return a 
    }, 

    decode: function(input) { 
     var output = ""; 
     var chr1, chr2, chr3 = ""; 
     var enc1, enc2, enc3, enc4 = ""; 
     var i = 0; 

     do { 
      enc1 = this._keyStr.indexOf(input.charAt(i++)) ; 
      enc2 = this._keyStr.indexOf(input.charAt(i++)) ; 
      enc3 = this._keyStr.indexOf(input.charAt(i++)) ; 
      enc4 = this._keyStr.indexOf(input.charAt(i++)) ; 

      chr1 = (enc1 | ((enc2 & 3) << 6)); 
      chr2 = (enc2 >> 2) | ((enc3 & 0x0F) << 4); 
      chr3 = (enc3 >> 4) | (enc4 << 2); 

      output = output + String.fromCharCode(chr1); 
      if (enc3 != 64) { 
       output = output + String.fromCharCode(chr2); 
         } 
      if (enc4 != 64) { 
       output = output + String.fromCharCode(chr3); 
      } 
      chr1 = chr2 = chr3 = ""; 
      enc1 = enc2 = enc3 = enc4 = ""; 
     } while (i < input.length); 

     return (output); 
    } 

}; 
+0

它是如何base64,如果它有65個值?是不是base65? –

+0

那麼base65然後.. –

+0

'isNaN(chr2)'檢查意味着相應的位置在輸入序列之外。當輸入序列的長度不是3的倍數時會發生這種情況。您的C++代碼不能正確處理該情況。 – Leon

回答

1

所以,在你的C++代碼)期待相比,JavaScript的isnan(應爲:

if (isnan(char_array_3[1])) { // char_array_3[1] = chr2 
    char_array_4[2] = char_array_4[1] = 64; // char_array_4[2] = enc3 & char_array_4[1] = enc2 
} else if (isnan(char_array_3[2])) { // char_array_3[2] = chr3 
    char_array_4[3] = 64; // char_array_4[3] = enc2 
} 

但主要的問題是, isnan()函數僅用於C++中的浮點值 ,與JavaScript中的 的含義不同。

代替使用該isnan()函數,取代的base64_encode()以下部分組成:

while (in_len--) { 
    char_array_3[i++] = *(bytes_to_encode++); 
    if (i == 3) { 
    ... 
    if (isnan(char_array_4[1])) { 
     char_array_4[2] = char_array_4[1] = 64; 
    } else if (isnan(char_array_4[2])) { 
     char_array_4[3] = 64; 
    } 
    ... 
    for(i = 0; (i < 4) ; i++) 
     ret += base64_chars[char_array_4[i]]; 
    i = 0; 
    } 

通過以下操作之一:

1-防止意外的值時小於3個字節左在 輸入緩衝區中強制它們爲0x00。也在每個下一個循環之前(for(j=0;j<3;j++) char_array_3[j]=0x00;)。

2-當2個字節已被從輸入緩衝器if (i == 2)加載,輸出緩衝器的最後 項目被設置爲64。

-3-當僅1字節已經被從輸入緩衝器if (i == 1)加載 輸出緩衝器的最後2個項目被設置爲64。

for(j=0;j<3;j++) char_array_3[j]=0x00; // initialize input array 
    while (in_len--) { 
    char_array_3[i++] = *(bytes_to_encode++); 
    if ((i == 3) || (in_len == 0)) { // encode when 3 bytes or end of buffer 
    ... 
    if (i == 1) { // instead of (isnan(char_array_4[1])) 
     // both char_array_3[1] and char_array_3[2] are not defined 
     char_array_4[3] = char_array_4[2] = 64; 
    } else if (i == 2) { // instead of (isnan(char_array_4[2])) 
     // char_array_3[2] is not defined 
     char_array_4[3] = 64; 
    } 
    ... 
    for(i = 0; (i < 4) ; i++) 
     ret += base64_chars[char_array_4[i]]; 
    i = 0; 
    for(j=0;j<3;j++) char_array_3[j]=0x00; // initialize input array 
    } 

base64_encode()功能在C++中比較的JavaScript最後的錯誤是在用於中間項的輸出緩衝計算。 相反那些以下分配的:

char_array_4[0] = char_array_3[0] & 0x3f; 
    char_array_4[1] = ((char_array_3[0] & 0x0f) << 2) + ((char_array_3[1] & 0xc0) >> 6); // NOK 
    char_array_4[2] = ((char_array_3[1] & 0x03) << 4) + ((char_array_3[2] & 0xf0) >> 4); // NOK 
    char_array_4[3] = (char_array_3[2] & 0xfc) >> 2; 

使用以下物質:

1-當計算char_array_4[1](=在JS ENC2),條目 在JS char_array_3[0](= CHR1 )和char_array_3[1](= chr2 in JS)相反。

2-當計算char_array_4[2](=在JS ENC3),條目 char_array_3[1](= CHR2在JS)和char_array_3[2](= CHR3在JS)是相反的。

// JS => enc1 = chr1 & 0x3f; 
    char_array_4[0] = (char_array_3[0] & 0x3f); // OK 
    // JS => enc2 = (chr1 >> 6) | ((chr2 & 0xf) << 2); 
    char_array_4[1] = ((char_array_3[0] & 0xc0) >> 6) + ((char_array_3[1] & 0x0f) << 2); // OK 
    // JS => enc3 = (chr2 >> 4) | ((chr3 & 0x3) << 4); 
    char_array_4[2] = ((char_array_3[1] & 0xf0) >> 4) + ((char_array_3[2] & 0x03) << 4); // OK 
    // JS => enc4 = chr3 >> 2; 
    char_array_4[3] = (char_array_3[2] & 0xfc) >> 2; 

那些修正後,將條件塊if (i)是無用的,應被除去。

+0

嗨J. Piquard。我已經按照你的建議嘗試過了,正如你可以在C++在線編譯器中看到的那樣,它不起作用。它仍然給我和以前一樣的結果。 –

+0

@Ke。在檢查添加的源代碼後,我已經更新了我的答案。 –

+0

是的,這是正確的答案,問題發生在char_array_4的內部(不在isnan語句中)。雖然,我不太清楚爲什麼javascript版本需要isnan。這是因爲它基本上是base65嗎?因爲腳本中有65個字符(包括開頭的)。 –