使用Delphi调用Http GET url的最简单方法是什么?

时间:2008-11-19 10:52:35

标签: delphi http network-programming

我想在我的应用程序中调用一个Web服务,我可以在导入WSDL时使用它,或者只使用带有URL和参数的“HTTP GET”,所以我更喜欢后者,因为它很简单。

我知道我可以使用indy idhttp.get来完成这项工作,但这很简单,我不想在我的应用程序中添加复杂的indy代码。

更新:对不起,如果我不清楚,我的意思是“不要添加复杂的indy代码”,我不想为这个简单的任务添加indy组件,并且更喜欢更轻方式。

8 个答案:

答案 0 :(得分:33)

使用Indy调用RESTful Web服务非常简单。

将IdHTTP添加到您的uses子句中。请记住,IdHTTP在您的网址上需要“HTTP://”前缀。

function GetURLAsString(const aURL: string): string;
var
  lHTTP: TIdHTTP;
begin
  lHTTP := TIdHTTP.Create;
  try
    Result := lHTTP.Get(aURL);
  finally
    lHTTP.Free;
  end;
end;

答案 1 :(得分:27)

您可以像这样使用WinINet API

uses WinInet;

function GetUrlContent(const Url: string): string;
var
  NetHandle: HINTERNET;
  UrlHandle: HINTERNET;
  Buffer: array[0..1024] of Char;
  BytesRead: dWord;
begin
  Result := '';
  NetHandle := InternetOpen('Delphi 5.x', INTERNET_OPEN_TYPE_PRECONFIG, nil, nil, 0);

  if Assigned(NetHandle) then 
  begin
    UrlHandle := InternetOpenUrl(NetHandle, PChar(Url), nil, 0, INTERNET_FLAG_RELOAD, 0);

    if Assigned(UrlHandle) then
      { UrlHandle valid? Proceed with download }
    begin
      FillChar(Buffer, SizeOf(Buffer), 0);
      repeat
        Result := Result + Buffer;
        FillChar(Buffer, SizeOf(Buffer), 0);
        InternetReadFile(UrlHandle, @Buffer, SizeOf(Buffer), BytesRead);
      until BytesRead = 0;
      InternetCloseHandle(UrlHandle);
    end
    else
      { UrlHandle is not valid. Raise an exception. }
      raise Exception.CreateFmt('Cannot open URL %s', [Url]);

    InternetCloseHandle(NetHandle);
  end
  else
    { NetHandle is not valid. Raise an exception }
    raise Exception.Create('Unable to initialize Wininet');
end;

来源:http://www.scalabium.com/faq/dct0080.htm

WinINet API使用InternetExplorer使用的相同内容,因此您还可以免费获得InternetExplorer设置的任何连接和代理设置。

答案 2 :(得分:15)

实际上,接受的答案中的代码对我不起作用。所以我稍微修改了它,所以它实际上返回String并在执行后优雅地关闭所有内容。示例将检索到的数据作为UTF8String返回,因此它适用于ASCII和UTF8页面。

uses WinInet;

function GetUrlContent(const Url: string): UTF8String;
var
  NetHandle: HINTERNET;
  UrlHandle: HINTERNET;
  Buffer: array[0..1023] of byte;
  BytesRead: dWord;
  StrBuffer: UTF8String;
begin
  Result := '';
  NetHandle := InternetOpen('Delphi 2009', INTERNET_OPEN_TYPE_PRECONFIG, nil, nil, 0);
  if Assigned(NetHandle) then
    try
      UrlHandle := InternetOpenUrl(NetHandle, PChar(Url), nil, 0, INTERNET_FLAG_RELOAD, 0);
      if Assigned(UrlHandle) then
        try
          repeat
            InternetReadFile(UrlHandle, @Buffer, SizeOf(Buffer), BytesRead);
            SetString(StrBuffer, PAnsiChar(@Buffer[0]), BytesRead);
            Result := Result + StrBuffer;
          until BytesRead = 0;
        finally
          InternetCloseHandle(UrlHandle);
        end
      else
        raise Exception.CreateFmt('Cannot open URL %s', [Url]);
    finally
      InternetCloseHandle(NetHandle);
    end
  else
    raise Exception.Create('Unable to initialize Wininet');
end;

希望像我这样的人在寻找简单的代码如何在Delphi中检索页面内容。 干杯,阿尔迪斯:)

答案 3 :(得分:7)

如果可以下载到文件,您可以使用ExtActns单元中的TDownloadURL。比直接使用WinInet简单得多。

procedure TMainForm.DownloadFile(URL: string; Dest: string);
var
  dl: TDownloadURL;
begin
  dl := TDownloadURL.Create(self);
  try
    dl.URL := URL;
    dl.FileName := Dest;
    dl.ExecuteTarget(nil); //this downloads the file
  finally
    dl.Free;
  end;
end;

使用此功能时,也可以获得进度通知。只需将事件处理程序分配给TDownloadURL的OnDownloadProgress事件。

答案 4 :(得分:7)

在较新的Delphi版本中,最好使用THTTPClient单位的System.Net.HttpClient,因为它是标准版和跨平台版。简单的例子是

function GetURL(const AURL: string): string;
var
  HttpClient: THttpClient;
  HttpResponse: IHttpResponse;
begin
  HttpClient := THTTPClient.Create;
  try
    HttpResponse := HttpClient.Get(AURL);
    Result := HttpResponse.ContentAsString();
  finally
    HttpClient.Free;
  end;
end;

答案 5 :(得分:4)

使用Windows HTTP API也很容易。

procedure TForm1.Button1Click(Sender: TObject);
var http: variant;
begin
 http:=createoleobject('WinHttp.WinHttpRequest.5.1');
 http.open('GET', 'http://lazarus.freepascal.org', false);
 http.send;
 showmessage(http.responsetext);
end;

在上面的代码中,我暗示COM已经为主VCL线程初始化了。据报道,对于简单的应用程序或LCL应用程序可能并非总是如此。同样,对于异步(多线程)工作肯定不会这样。

以下是运行实际代码的代码段。注意 - 功能是奖金。它不需要工作。因此,当我发出请求时,我不关心他们的结果,结果会被忽略并转储。

procedure TfmHaspList.YieldBlinkHTTP(const LED: boolean; const Key_Hardware_ID: cardinal);
var URL: WideString;
begin
  URL := 'http://127.0.0.1:1947/action.html?blink' +
    IfThen( LED, 'on', 'off') + '=' + IntToStr(Key_Hardware_ID);

  TThread.CreateAnonymousThread(
    procedure
    var Request: OleVariant;
    begin
      // COM library initialization for the current thread
      CoInitialize(nil);
      try
        // create the WinHttpRequest object instance
        Request := CreateOleObject('WinHttp.WinHttpRequest.5.1');
        // open HTTP connection with GET method in synchronous mode
        Request.Open('GET', URL, False);
        // set the User-Agent header value
//        Request.SetRequestHeader('User-Agent', 'Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.0');
        // sends the HTTP request to the server, the Send method does not return
        // until WinHTTP completely receives the response (synchronous mode)
        Request.Send;
//        // store the response into the field for synchronization
//        FResponseText := Request.ResponseText;
//        // execute the SynchronizeResult method within the main thread context
//        Synchronize(SynchronizeResult);
      finally
        // release the WinHttpRequest object instance
        Request := Unassigned;
        // uninitialize COM library with all resources
        CoUninitialize;
      end;
    end
  ).Start;
end;

答案 6 :(得分:2)

使用HTTPSEND单元中的Synapse TCP/IP功能(HTTPGetText, HTTPGetBinary)。它将为您执行HTTP提取,并且除了Winsock之外不需要任何外部DLL。最新的SVN版本在Delphi 2009中运行得非常好。它使用阻塞函数调用,因此没有事件要编程。

更新:单位非常轻,不是基于组件。 SVN的最新版本在Delphi XE4中运行得非常好。

答案 7 :(得分:0)

如果您的应用程序仅限Windows,我建议您使用WinSock。它很简单,允许执行任何HTTP请求,可以同步和异步工作(在专用线程中使用带有回调的非阻塞WSASend / WSARecv或旧的send / recv)。