0

我在尝试String用 Java 对 a 进行编码时遇到了麻烦。

我在 C# 中有以下代码和string Bpz2Gjg01d7VfGfD8ZP1UA==,当我执行 C# 代码时,我得到:

QnB6MkdqZzAxZDdWZkdmRDhaUDFVQT090

public static void Main(string[] args)
{
    string strWord = "Bpz2Gjg01d7VfGfD8ZP1UA==";
    byte[] encbuff = Encoding.UTF8.GetBytes(strWord);
    string strWordEncoded = HttpServerUtility.UrlTokenEncode(encbuff);
    Console.WriteLine(strWordEncoded);
}

我试图在 Java 中复制以前的代码,在第一次尝试中我使用了javax.xml.bind.DatatypeConverter Class

public static void main(String[] args) {
    String strWord = "Bpz2Gjg01d7VfGfD8ZP1UA==";
    byte[] encbuff = strWord.getBytes(StandardCharsets.UTF_8);
    String strWordEncoded = DatatypeConverter.printBase64Binary(encbuff);
    System.out.println(strWordEncoded);
}

但我得到以下信息String(与 C# 相比缺少最后一个零string):

QnB6MkdqZzAxZDdWZkdmRDhaUDFVQT09

在我的第二次尝试中,我使用了 BouncyCastleBase64编码器:

public static void main(String[] args) {
   String strWord = "Bpz2Gjg01d7VfGfD8ZP1UA==";
   byte[] encbuff = strWord.getBytes(StandardCharsets.UTF_8);
   String strWordEncoded = new String(Base64.encode(encbuff));
   System.out.println(strWordEncoded);
}

但是我得到了完全相同的前一个String(仍然缺少最后一个零):

QnB6MkdqZzAxZDdWZkdmRDhaUDFVQT09

有谁知道可能会发生什么?

4

2 回答 2

1

我看过 .NET 框架代码。UrlTokenEncode实际上从 base64 字符串的末尾删除任何额外的=填充符号,并将它们替换为填充符号的数量,所以要么0, 1, 要么2. 这就是导致0字符串末尾多余的原因。所以请注意:该HttpServerUtility.UrlTokenEncode方法不是普通的 Base64 编码器。它实际上Convert.ToBase64String在内部用于常规编码,并在顶部添加了更多内容(请参阅我对问题的评论)。如果您需要创建这个确切的字符串,则需要在常规 base64 编码的基础上在 Java 中实现相同的更改。

于 2019-03-29T18:13:55.773 回答
0

我根据给我的评论找到了一个解决方案,基本上我在Microsoft的Reference Source中查看了该方法的源代码。

然后我将 C# 代码翻译成 Java 代码,它看起来像这样:

public static String UrlTokenEncode(byte[] input) {
     try {
         if (input == null) {
         return null;
         }

         if (input.length < 1) {
             return null;
         }

         String base64Str = null;
         int endPos = 0;
         char[] base64Chars = null;

         base64Str = Base64.toBase64String(input);
         if (base64Str == null) {
             return null;
         }

         for (endPos = base64Str.length(); endPos > 0; endPos--) {
             if (base64Str.charAt(endPos - 1) != '=') {
                 break;
             }
         }

         base64Chars = new char[endPos + 1];
         base64Chars[endPos] = (char) ((int) '0' + base64Str.length() - endPos);
         for (int iter = 0; iter < endPos; iter++) {
             char c = base64Str.charAt(iter);
              switch (c) {
                 case '+':
                     base64Chars[iter] = '-';
                     break;
                 case '/':
                     base64Chars[iter] = '_';
                     break;
                 case '=':
                     base64Chars[iter] = c;
                 break;
                 default:
                     base64Chars[iter] = c;
                 break;
             }
          }
         return new String(base64Chars);
     } catch (Exception e) {
         return null;
     }
}

最后我测试了这个方法,得到了想要的输出:

public static void main(String[] args) {
   String strWord = "Bpz2Gjg01d7VfGfD8ZP1UA==";
   byte[] encbuff = strWord.getBytes(StandardCharsets.UTF_8);
   String strWordEncoded = UrlTokenEncode(encbuff);
}

M2NIclh4eEwxRGp2MEsyeFc0SHVDZz090

于 2019-03-29T23:34:37.533 回答