Java与Delphi Base64编码

时间:2015-02-17 13:13:27

标签: java delphi encoding base64

Delphi和Java生成完全不同的值。

我有MD5列表,我需要使用Base64然后URLEncode对它们进行编码。

这是相同哈希值的delphi和java输出。

如何使用Delphi与Java输出相同?

MD5输入值

BB8C890E3E6372F2720709262BD42BF4

Java Base64 编码输出值:

vIRYPbGf3toJDQehgv3SOamnTNk=

Delphi Base64 编码输出值:

QkI4Qzg5MEUzRTYzNzJGMjcyMDcwOTI2MkJENDJCRjQ=

delphi代码行是:

IdEncoderMIME1.EncodeString('BB8C890E3E6372F2720709262BD42BF4');

Java Source是:

String md5 = new String(Base64.encodeBase64(decodedHex));

完整的Java代码(我想用Delphi获得相同的结果):     public static void main(String [] args)抛出异常     {     if(args.length!= 8)     {     用法();     返回;     }

WebClient webClient = null;

try
{
byte[] decodedHex = Hex.decodeHex(args[0].toCharArray());

String sha1 = new String(Base64.encodeBase64(decodedHex));

decodedHex = Hex.decodeHex(args[1].toCharArray());

String md5 = new String(Base64.encodeBase64(decodedHex));

String rep = args[2];
String comment = args[3];
String Server = args[4];
String ServerPort = args[5];
String username = args[6];
String password = args[7];

String params = "[{\"sha1\":\"" + sha1 + "\",\"md5\":\"" + md5 + "\",\"rep\":\"" + rep + "\",\"comment\":\"" + comment + "\"}]";

// Note the usage of URLEncoder to ensure special characters can be used within the parameters.
String url = "https://" + Server + ":" + ServerPort + "/setRep?fileReps=" + URLEncoder.encode(params, "UTF-8");

System.out.println("\nWeb API params with Base64 values but are not yet URL encoded:");
System.out.println("\n" + params + "\n");
System.out.println("\nWeb API URL call with URL encoded parameters:");
System.out.println("\n" + url + "\n");

webClient = new WebClient(BrowserVersion.FIREFOX_24);

webClient.getOptions().setUseInsecureSSL(true);

DefaultCredentialsProvider userCredentials = (DefaultCredentialsProvider)webClient.getCredentialsProvider();
userCredentials.addCredentials(username, password);
webClient.setCredentialsProvider(userCredentials);

Page setFileRepResponsePage = webClient.getPage(url);

String pageResponse = setFileRepResponsePage.getWebResponse().getContentAsString();

System.out.println("HTTP Status code: " + setFileRepResponsePage.getWebResponse().getStatusCode());
System.out.println(pageResponse);

}
catch(Exception ex)
{
ex.printStackTrace();
}
finally
{
try
{
webClient.closeAllWindows();
}
catch(Exception ex) {}
}
}

2 个答案:

答案 0 :(得分:2)

package main;

import java.io.UnsupportedEncodingException;
import java.util.Base64;

public class test {
    public static void main(String[] args) {
        try {
            // outputs QkI4Qzg5MEUzRTYzNzJGMjcyMDcwOTI2MkJENDJCRjQ=
            System.out.println(Base64.getEncoder()
                .encodeToString("BB8C890E3E6372F2720709262BD42BF4".getBytes("US-ASCII")));
        } catch (UnsupportedEncodingException e) {
            throw new RuntimeException(e);
        }
    }
}

答案 1 :(得分:2)

IdEncoderMIME1.EncodeString('BB8C890E3E6372F2720709262BD42BF4');

这需要文本输入,编码为ASCII,然后编码为base64。

String md5 = new String(Base64.encodeBase64(decodedHex));

这将获取decodedHex中的二进制数据并编码为base64。

因此,你的两段代码正在做不同的事情。为了让你取得任何进展,你真的需要弄清楚你想做什么。在我看来,采用二进制哈希,编码为十六进制,编码为ASCII然后编码为base64是没有意义的。

所以我会说德尔福代码就是问题所在。你从二进制哈希和base64编码开始。我无法为您提供代码,因为您没有给我们足够的代码来处理。例如,我很确定你不是以'BB8C890E3E6372F2720709262BD42BF4'作为字符串文字开头的。

但实际上你的主要任务不是编写代码,而是要牢牢掌握你应该执行的编码。

FWIW,文本和二进制文件之间的这种混淆是Delphi标签中反复出现的主题之一。我猜这一切都源于Delphi程序员习惯使用string的习惯,好像它是一个字节数组。这种文本和二进制之间区别的模糊导致像TIdEncoderMIME.EncodeString这样的方法,它们鼓励你错误地思考base64编码对文本输入的操作。


根据您的最新编辑,您希望将十六进制字符串解码为二进制,并对base64进行编码。在Delphi中这样做是这样的:

uses
  System.Classes,
  System.NetEncoding;

function HexStringToBase64(const HexStr: string): string;
var
  md5bytes: TBytes;
begin
  SetLength(md5Bytes, Length(HexStr) div 2);
  HexToBin(PChar(HexStr), Pointer(md5Bytes), Length(md5Bytes));
  Result := TNetEncoding.Base64.EncodeBytesToString(md5Bytes);
end;

现在,除非您使用XE7或更高版本,否则您将不会拥有TNetEncoding。如果是这样,请选择另一个base64库。如果您更喜欢使用Indy编码器,那么您可以这样写:

function HexStringToBase64(const HexStr: string): string;
var
  md5bytes: TIdBytes;
begin
  SetLength(md5Bytes, Length(HexStr) div 2);
  HexToBin(PChar(HexStr), Pointer(md5Bytes), Length(md5Bytes));
  Result := TIdEncoderMIME.EncodeBytes(md5Bytes);
end;

而FWIW,您的Java代码不会产生您声称的输出。