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) {}
}
}
答案 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代码不会产生您声称的输出。