[英]Java vs Delphi Base64 Encoding
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)引发Exception {if(args.length!= 8){usage(); 返回; }
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) {}
}
}
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);
}
}
}
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代码不会产生您声称的输出。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.