THttpClient 是 Delphi/C++ Builder 新引入的一个 HTTP/HTTPS 协议客户端的封装,相当好用。我现在已经基本上不使用 libcurl 了,它已经能满足我绝大多数情况下的 HTTP 操作请求。
不过,最近遇到了一点小问题,我们先看 Post 函数的声明:
/// <summary>Post a raw file without multipart info</summary>
function Post(const AURL: string; const ASourceFile: string; const AResponseContent: TStream = nil; const AHeaders: TNetHeaders = nil): IHTTPResponse; overload;
/// <summary>Post TStrings values adding multipart info</summary>
function Post(const AURL: string; const ASource: TStrings; const AResponseContent: TStream = nil;
const AEncoding: TEncoding = nil; const AHeaders: TNetHeaders = nil): IHTTPResponse; overload;
/// <summary>Post a stream without multipart info</summary>
function Post(const AURL: string; const ASource: TStream; const AResponseContent: TStream = nil;
const AHeaders: TNetHeaders = nil): IHTTPResponse; overload;
/// <summary>Post a multipart form data object</summary>
function Post(const AURL: string; const ASource: TMultipartFormData; const AResponseContent: TStream = nil;
const AHeaders: TNetHeaders = nil): IHTTPResponse; overload;这是 THttpClient 的 Post 的几个重载,如果我要直接提交一个 JSON 或 XML 字符串到服务器端,上面的第 1 种重载要求先保存到文件再提交,第 2 种重载要求的是一个 TStrings 对象,看起来很有戏,第三个重载是投寄一个流进去,看起来也有戏,第四种重载明显就是我们需要的了。
好吧,然后我们进入第 2 种重载的源码,发现了什么?
LParams := '';
for I := 0 to ASource.Count - 1 do
begin
Pos := ASource[I].IndexOf('=');
if Pos > 0 then
LParams := LParams + ASource[I].Substring(0, Pos) + '=' + TURI.URLEncode(ASource[I].Substring(Pos + 1), True) + '&';
end;
LParams := LParams.Substring(0, LParams.Length - 1); // Remove last &
也就是说,它传递的是实际上是 Name -Value 值对,没配对的直接忽略掉了。显然也不符合我们的要求,可以被 Pass 掉了。剩下的唯一选择就是传递一个流进去。
好了,QJSON 和 QXML 都提供了保存到流里的支持,所以创建一个临时的内存流,保存下来后做为参数传递进去,用完释放就 OK 了。如果嫌不够直观,还可以用 QMacros 来用模板替换的方式来替换,效率一样杠杠的。下面是一个简单的例子(用 THttpClient 用轻码云发送模板短信):
var
AResponse: IHttpResponse;
AUrl: String;
AMacros: TQMacroManager;
AReply, AItem: TQJson;
ATimeStamp: String;
AStream: TStringStream;
const
AVerifyUrl
: String =
'#protocol#://api.qingmayun.com/#SoftVersion#/accounts/#accountSid#/SMS/templateSMS?'
+ 'sig=#sig#×tamp=#timestamp#';
APostContent: String = '{'#13#10 + '"templateSMS": {'#13#10 +
' "appId": "#appId#",'#13#10 + ' "templateId": "#templateId#",'#13#10
+ ' "to": "#to#",'#13#10 + ' "param": "#param#"'#13#10 +
' }'#13#10 + '}';
function CalcSign: String;
var
S: String;
ADigest: TQMD5Digest;
begin
S := AccountId + AuthToken + ATimeStamp;
ADigest := MD5Hash(S);
Result := BinToHex(@ADigest, SizeOf(ADigest), true);
end;
begin
Result := False;
AMacros := TQMacroManager.Create;
AStream := TStringStream.Create('', TEncoding.UTF8, False);
try
ATimeStamp := FormatDateTime('yyyymmddhhnnss', Now);
AMacros.Push('protocol', 'https');
AMacros.Push('SoftVersion', '20141029');
AMacros.Push('accountSid', AccountId);
AMacros.Push('appId', AppId);
AMacros.Push('templateId', TemplateId);
AMacros.Push('to', APhoneNum);
AMacros.Push('param', ACode + ',30');
AMacros.Push('sig', CalcSign);
AMacros.Push('timestamp', ATimeStamp);
AUrl := AMacros.Replace(AVerifyUrl, '#', '#');
FHttpClient.Accept := 'application/json';
FHttpClient.ContentType := 'application/json;charset=utf-8';
AStream.WriteString(AMacros.Replace(APostContent, '#', '#',
MRF_IN_DBL_QUOTER));
AStream.Position := 0;
AResponse := FHttpClient.Post(AUrl, AStream);
if Assigned(AResponse) then
begin
if AResponse.StatusCode = 200 then
begin
AReply := TQJson.Create;
try
if AReply.TryParse(AResponse.ContentAsString()) then
begin
FLastError := AReply.ValueByPath('result.respCode', '');
if FLastError = '00000' then
Result := true;
end;
finally
FreeAndNil(AReply);
end;
end
else
FLastError := IntToStr(AResponse.StatusCode);
end
else
FLastError := '-1';
finally
FreeAndNil(AMacros);
end;
