簡體   English   中英

如何在JavaScript中使用字節數組將字符串轉換為base64編碼?

[英]How to convert a string to base64 encoding using byte array in JavaScript?

我有以下.NET代碼,首先將字符串轉換為字節數組,即可將字符串轉換為Base64編碼。 我在Stack Overflow上嘗試了不同的答案,以將字節數組轉換為字符串,然后將btoa()函數用於JavaScript中的base64編碼。 但是,我沒有得到下面共享的確切編碼值。

對於字符串值,

BBFDC43D-4890-4558-BB89-50D802014A97

我需要Base64編碼,

PcT9u5BIWEW7iVDYAgFKlw ==

.NET代碼:

String str = "BBFDC43D-4890-4558-BB89-50D802014A97"
Guid guid = new Guid(str);
Console.WriteLine(guid);    // bbfdc43d-4890-4558-bb89-50d802014a97
Byte[] bytes = guid.ToByteArray();
Console.WriteLine(bytes);   // System.Byte[]
String s = Convert.ToBase64String(bytes, Base64FormattingOptions.InsertLineBreaks);
Console.WriteLine(s);   // PcT9u5BIWEW7iVDYAgFKlw==

目前,我嘗試使用以下代碼,但未產生預期的結果:

function strToUtf8Bytes(str) {
  const utf8 = [];
  for (let ii = 0; ii < str.length; ii++) {
    let charCode = str.charCodeAt(ii);
    if (charCode < 0x80) utf8.push(charCode);
    else if (charCode < 0x800) {
      utf8.push(0xc0 | (charCode >> 6), 0x80 | (charCode & 0x3f));
    } else if (charCode < 0xd800 || charCode >= 0xe000) {
      utf8.push(0xe0 | (charCode >> 12), 0x80 | ((charCode >> 6) & 0x3f), 0x80 | (charCode & 0x3f));
    } else {
      ii++;
      // Surrogate pair:
      // UTF-16 encodes 0x10000-0x10FFFF by subtracting 0x10000 and
      // splitting the 20 bits of 0x0-0xFFFFF into two halves
      charCode = 0x10000 + (((charCode & 0x3ff) << 10) | (str.charCodeAt(ii) & 0x3ff));
      utf8.push(
        0xf0 | (charCode >> 18),
        0x80 | ((charCode >> 12) & 0x3f),
        0x80 | ((charCode >> 6) & 0x3f),
        0x80 | (charCode & 0x3f),
      );
    }
  }
  return utf8;
}

const str = "BBFDC43D-4890-4558-BB89-50D802014A97";
const strByteArr = strToUtf8Bytes(str);
const strBase64 = btoa(strByteArr);
// NjYsNjYsNzAsNjgsNjcsNTIsNTEsNjgsNDUsNTIsNTYsNTcsNDgsNDUsNTIsNTMsNTMsNTYsNDUsNjYsNjYsNTYsNTcsNDUsNTMsNDgsNjgsNTYsNDgsNTAsNDgsNDksNTIsNjUsNTcsNTU=

您的代碼的問題是Guid的表示形式。 在C#代碼中,您正在將“ BBFDC43D-4890-4558-BB89-50D802014A97”轉換為UUID,這是一個128位數字。 在JavaScript代碼中,您正在做其他事情。 您遍歷字符串並計算字符串的字節數組。 他們根本不平等。 現在您必須選擇

  1. 在JS中實現適當的guid轉換(這可能會有所幫助: https : //gist.github.com/daboxu/4f1dd0a254326ac2361f8e78f89e97ae
  2. 在C#中,以與JS中相同的方式計算字節數組

您的問題是由以下原因引起的:

  • btoa()使用ASCII編碼
  • guid.ToByteArray(); 不使用ASCII編碼

如果您像這樣修改C#代碼:

String str = "BBFDC43D-4890-4558-BB89-50D802014A97";
//Guid guid = new Guid(str);
//Console.WriteLine(guid);
// bbfdc43d-4890-4558-bb89-50d802014a97
//Byte[] bytes = guid.ToByteArray();
byte[] bytes = System.Text.Encoding.ASCII.GetBytes(str);
//Console.WriteLine(bytes);   // System.Byte[]
String s = Convert.ToBase64String(bytes, Base64FormattingOptions.InsertLineBreaks);
Console.WriteLine(s);

您將獲得以下輸出:

QkJGREM0M0QtNDg5MC00NTU4LUJCODktNTBEODAyMDE0QTk3

該字符串與從btoa()函數返回的字符串相同:

var rawString = "BBFDC43D-4890-4558-BB89-50D802014A97";
var b64encoded = btoa(rawString);
console.log(b64encoded);

輸出:

QkJGREM0M0QtNDg5MC00NTU4LUJCODktNTBEODAyMDE0QTk3

更新-由於您無法修改C#代碼

您應該通過結合Piotr的答案此SO答案來修改Javascript代碼

function guidToBytes(guid) {
    var bytes = [];
    guid.split('-').map((number, index) => {
        var bytesInChar = index < 3 ? number.match(/.{1,2}/g).reverse() : number.match(/.{1,2}/g);
        bytesInChar.map((byte) => { bytes.push(parseInt(byte, 16)); })
    });
    return bytes;
}

function arrayBufferToBase64(buffer) {
    var binary = '';
    var bytes = new Uint8Array(buffer);
    var len = bytes.byteLength;
    for (var i = 0; i < len; i++) {
        binary += String.fromCharCode(bytes[i]);
    }
    return btoa(binary);
}

var str = "BBFDC43D-4890-4558-BB89-50D802014A97";
var guidBytes = guidToBytes(str);
var b64encoded = arrayBufferToBase64(guidBytes);
console.log(b64encoded);

輸出:

PcT9u5BIWEW7iVDYAgFKlw==

您的字符串是一個十六進制值,可用於創建GUID。 然后,使用以下命令將GUID轉換為字節數組:

Byte[] bytes = guid.ToByteArray();

GUID是一個16字節的值,可以表示為十六進制值。 當您將此GUID轉換為字節數組時,將獲得該值的16個字節, 而不是十六進制值的字節表示形式。

在提供的JavaScript函數中,您正在執行其他操作:您正在將字符串直接轉換為字節數組。

在C#中,您可以使用Encoding進行等效操作:

String str = "BBFDC43D-4890-4558-BB89-50D802014A97";
Byte[] bytes = Encoding.UTF8.GetBytes(str);

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM