2015-06-27 136 views
0

我是学习C++ Builder的新手。三天前,我安装了Embarcadero®。 C++Builder®2010.这种语言对我来说非常有趣。将Delphi代码的Indy 10写入C++ Builder的Indy 10

在Delphi中,我通常使用Indy 9和10的TIdMappedPortTCP编写一个简单的代理服务器。我通常使用它的OnExecute和OnOutboundData事件修改通过代理的数据。

因为我是C++ Builder中的新成员,所以我不知道如何将我的Delphi代码转换为完全正确的C++ Builder代码。

我试过和尝试过很多办法,包括读几本书,其中之一是的Borland C++ Builder中 - 完整的参考,通过赫伯特·希,以及增加知识。不幸的是,在这本书中并没有讨论与我的状况有关的所有非常重要的事情。此外,我找到谷歌的参考,但我没有找到。

所以,我冒昧地寻求你的帮助。我真的需要它。 请帮忙!非常感谢你。

以下是我想要写入C++ Builder的Indy 10的Delphi代码。

...... 

procedure TForm.IdMappedPortTCP1Execute(AContext: TIdContext); 
var 
Mydata, NetData: string; 
begin 
    if (Pos('HTTP',netstring(AContext)) <> 0) or (Pos('GET',netstring(AContext)) <> 0) then begin 
    NetData := netstring(AContext); 
    TIdMappedPortContext(AContext).OutboundClient.IOHandler.Write(AddHeader(netstring(AContext),'Connection: Keep-Alive')); 
    Sleep(1000); 
    Mydata := 'GET http://website.com/ HTTP/1.1'+#13#10+'Host: website.com'#13#10; 
    NetData := Mydata + Netdata; 
    TIdMappedPortContext(AContext).NetData := netbyte(Netdata); 
    TIdMappedPortContext(AContext).OutboundClient.IOHandler.Write(netbyte(Mydata + NetData)); 
    end; 
end; 

...... 
+2

也许这不是学习新语言的方式。 –

+1

恕我直言,这是正确的方式之一,先生大卫!... – RzV

+0

我会从语言基础开始 –

回答

3

一个字面翻译C++ Builder中应该是这样的:

...... 

String __fastcall AddHeader(String S, String Header) 
{ 
    S = StringReplace(S, "\r\n\r\n", "\r\n" + Header + "\r\n\r\n", TReplaceFlags() << rfReplaceAll); 
    return S; 
} 

void __fastcall TForm::IdMappedPortTCP1Execute(TIdContext *AContext) 
{ 
    String Mydata, NetData; 

    if ((netstring(AContext).Pos("HTTP") != 0) || (netstring(AContext).Pos("GET") != 0)) 
    { 
     NetData = netstring(AContext); 
     TIdMappedPortContext(AContext)->OutboundClient->IOHandler->Write(AddHeader(netstring(AContext), "Connection: Keep-Alive")); 
     Sleep(1000); 
     Mydata = "GET http://website.com/ HTTP/1.1\r\nHost: website.com\r\n"; 
     NetData = Mydata + Netdata; 
     static_cast<TIdMappedPortContext*>(AContext)->NetData = netbyte(Netdata); 
     static_cast<TIdMappedPortContext*>(AContext)->OutboundClient->IOHandler->Write(netbyte(Mydata + NetData)); 
    } 
} 

...... 

这里有一个稍微精简版:

...... 

String __fastcall AddHeader(String S, String Header) 
{ 
    return StringReplace(S, "\r\n\r\n", "\r\n" + Header + "\r\n\r\n", TReplaceFlags() << rfReplaceAll); 
} 

void __fastcall TForm::IdMappedPortTCP1Execute(TIdContext *AContext) 
{ 
    String NetData = netstring(AContext); 
    if ((NetData.Pos("HTTP") != 0) || (NetData.Pos("GET") != 0)) 
    { 
     Sleep(1000); 
     String Mydata = "GET http://website.com/ HTTP/1.1\r\nHost: website.com\r\n" + AddHeader(NetData, "Connection: Keep-Alive"); 
     static_cast<TIdMappedPortContext*>(AContext)->NetData = netbyte(Mydata); 
    } 
} 

...... 

但无论哪种方式,这肯定是不是在Indy中实现可行HTTP代理的可靠方法。实际上,Indy 10为此目的引入了一个特定的TIdHTTPProxyServer组件。你应该认真考虑使用它,而不是TIdMappedPortTCP。例如,上面可以在TIdHTTPProxyServer做过这样的:

class TIdHTTPProxyServerContextAccess : public TIdHTTPProxyServerContext 
{ 
public: 
    void SetCommand(String Value) { FCommand = Value; } 
    void SetDocument(String Value) { FDocument = Value; } 
    void SetTarget(String Value) { FTarget = Value; } 
}; 

void __fastcall TForm1.IdHTTPProxyServer1HTTPBeforeCommand(TIdHTTPProxyServerContext *AContext) 
{ 
    static_cast<TIdHTTPProxyServerContextAccess*>(AContext)->SetCommand("GET"); 
    static_cast<TIdHTTPProxyServerContextAccess*>(AContext)->SetTarget ("http://website.com/"); 
    static_cast<TIdHTTPProxyServerContextAccess*>(AContext)->SetDocument("/"); 

    AContext->Headers->Values["Host"] = "website.com"; 
    AContext->Headers->Values["Connection"] = "Keep-Alive"; 

    /* 
    the original code was not changing the Host/Port where the 
    HTTP request was being sent to. But if you needed to, 
    you can do it like this... 

    static_cast<TIdTCPClient*>(AContext->OutboundClient)->Host = "website.com"; 
    static_cast<TIdTCPClient*>(AContext->OutboundClient)->Port = 80; 
    */ 
} 

更新:你链接的netstring()netbyte()功能有语法错误,并有不必要的开销(有没有必要MIME只涉及到将字符串转换为字节数组,反之亦然,Indy具有专门用于该目的的功能)。以下是修正版本:

String __fastcall netstring(TIdMappedPortContext* AContext) 
{ 
    return BytesToStringRaw(AContext->NetData); 
} 

TIdBytes __fastcall netbyte(String S) 
{ 
    return ToBytes(S, IndyTextEncoding_8Bit()); 
} 

所以,你可以实际上只是完全消除的功能:

void __fastcall TForm::IdMappedPortTCP1Execute(TIdContext *AContext) 
{ 
    TIdMappedPortContext *ctx = static_cast<TIdMappedPortContext*>(AContext) 
    String NetData = BytesToStringRaw(ctx->NetData); 
    if ((NetData.Pos("HTTP") != 0) || (NetData.Pos("GET") != 0)) 
    { 
     Sleep(1000); 
     String Mydata = "GET http://website.com/ HTTP/1.1\r\nHost: website.com\r\n" + AddHeader(NetData, "Connection: Keep-Alive"); 
     ctx->NetData = ToBytes(Mydata); 
    } 
} 
+0

非常感谢。你的回答对我很有帮助。顺便说一句,我仍然有错误** [BCC32错误] Unit1.cpp(116):E2303类型名称期望**相关的调用netstring + netbyte函数,你可以看到函数[here](http:// pastebin。 COM/Qq64J7iD)。和我的UNIT.CPP&UNIT.H代码,你可以查看[here](http://pastebin.com/QiFMfnyE)。你能帮我解决这个问题吗? – RzV

+0

我更新了我的答案。 –

+0

谢谢,雷米先生。但是,我仍然有两个错误:[[BCC32错误] Unit1.cpp(111):E2268调用未定义函数'BytesToStringRaw''和'[BCC32错误] Unit1.cpp(118):E2268调用未定义函数'IndyTextEncoding_8Bit' '。 Fyi,我把'#pragma link“IdGlobal”'和'#pragma link“IdGlobalProtocols”'放到我的** unit1.cpp **中,还有#include“IdGlobal.hpp”和'#include“IdGlobalProtocols .hpp“'到** unit1.h **中,但这些错误仍然会发生。那么,我应该使用哪个Indy 10组件? – RzV