引子
Web程序開發對開發工具提出了極大挑戰,面對用戶的眾多需求,許多公司推出一大堆開發平台:開發桌面應用程序和C/S程序的,開發中間件的,開發Web服務器的。這造成程序員面對一大堆工具無從下手。
Delphi和其他開發工具不同,因為它是一個開放系統,只要靈活使用一些控件,即可開發出各種類型的系統,不論N-TIE程序、多線程程序、分布計算程序(包括DCOM和CORBAR)、TCP程序、Web程序、ActiveX、中間件、推程序(Push),甚至你可以用它來寫匯編程序。
Delphi將ISAPI/NSAPI/CGI/WCGI等巧妙地封裝成一個類,用戶只要在編譯時選擇編譯結果,就可以得到不同的系統。
在Delphi4中Inprise公司進一步加強了對Web程序開發的支持,可以開發出更好更強的系統。以下是開發Web應用程序中的幾個常見問題,可以供大家參考。如果沒有特別申明,則表示程序運行於 Delphi 4下。
如何從Web Server Application返回一幅圖像?
Web Server Application不僅可以生成復雜的頁面文檔,也可以根據用戶請求返回不同的圖像。當然有比較簡單的方法,根據輸入參數不同,〈img src...〉標記也指向不同的URL地址。這裡我們不用這個辦法,而是利用DLL返回圖像。
當然要首先建立一個頁面容器(page producer),內容如下:
〈Html〉
〈body〉This is a test〈BR〉〈img src=″/scripts/mydll.dll/picture″〉〈/body〉
〈/Html〉
接下來我們設定對應於PathInfo的動作事件,返回圖像結果,源代碼如下:
(注意:單元聲明中要包含JPEG單元)
procedure TWebModule1.WebModule1WebActionItem1Action(Sender: TObject;Request: TWebRequest; Response: TWebResponse; var Handled: Boolean);
var
Jpg: TJpegImage;
S: TMemoryStream;
begin
Jpg := TJpegImage.Create;
try
Jpg.LoadFromFile(′test′);
S := TMemoryStream.Create;
try
Jpg.SaveToStream(S);
S.Position := 0;
Response.ContentType := ′image/jpeg′;
Response.ContentStream := S;
// 必須在流釋放前完成
Response. SendResponse;
finally
S.Free;
end;
finally
Jpg.Free;
end;
end;
實際上用這種方法和前面提到的簡單做法相比,具有更安全和更靈活的特點。在某些地方靈活使用,以此為基礎稍加修改可以產生一般開發工具難以實現的效果。
如何在ISAPI/NSAPI動態連接庫(DLL)中使用本地數據庫驅動程序(native Access driver)?
這是因為DAO 3.0 或者DAO 3.5是所謂thread-safe(線程安全程序),而一個Web服務器(例如IIS)會隨用戶請求產生多個線程,同時把ISAPI對應的DLL也列為線程。這時ISAPI就會通過BDE通知DAO,告訴它不符合線程安全規定。
解決的方法有很多,如果你一定要訪問Access 95/97庫,那麼可以通過ODBC訪問。ODBC不會經過DAO,而且也是一個線程安全程序。此外還有一些第三方的控件集,通過他們可以直接由BDE訪問Access 95/97,效率更高。
用戶訪問我Web服務器上的ISAPI DLL,可是報告:“Invalid filename"(無效的文件名),然而文件的確存在的。對了,我的數據庫在一台Novell 服務器上。這是為什麼呢?
你沒有設定對應於你IUSR_XXX賬戶的驅動器路徑映射(MAPING)。因為Novell不是采用FAT,所以要手工添加路徑映射。當然,可以做成開機登錄腳本。請牢記,如果你運行IIS作為Web服務器,而又涉及Novell,無論作為文件服務器或數據庫服務器,都要定義好路徑映射。
“Invalid configuration parameter for alias {alias_name}"(無效的別名配置),當我設置一個ODBC DSN,並通過它訪問ISAPI/NSAPI服務器時就出現這樣一個錯誤。
你如果要為訪問的用戶(IIS用戶)建立一個ODBC別名,那麼要注意創建一個SYSTEM DSN(系統DNS),而不要創建“用戶DNS”,雖然“用戶DNS”是缺省設定。
如何取得客戶機(訪問機器)的名稱和IP地址?
實現這個功能用TCP控件來做非常容易。從Internet頁面上選取一個TCP控件,然後直接就可以得到你所需要的:
Memo1.Lines.Add(TCP1.LocalHostName);
Memo1.Lines.Add(TCP1.LocalIp);
當然,如果你不希望這樣做,還有比較復雜的辦法:
uses Winsock;
procedure TForm1.FormCreate(Sender: TObject);
var
wVersionRequested : Word;
wsaData : TWSAData;
begin
{創建 WinSock}
wVersionRequested := MAKEWord(1, 1);
WSAStartup(wVersionRequested, wsaData);
end;
procedure TForm1.Button1Click(Sender: TObject);
var
p : PHostEnt;
s : array[0..128] of char;
p2 : pchar;
begin
{得到計算機名稱}
GetHostName(@s,128);
p:=GetHostByName(@s);
Memo1.Lines.Add(p^.h_Name);
{得到機器IP地址}
p2 := iNet_ntoa(PInAddr(p^.h_addr_list^)^);
Memo1.Lines.Add(p2);
end;
procedure TForm1.FormDestroy(Sender: TObject);
begin
{釋放 WinSock}
WSACleanup;
end;
這是一個調用了WINSOCK的獨立單元,你可以把它直接嵌入到你的程序中去。
在Delphi 3中為何不能創建真正的多線程DLL?
雖然Delphi3中的ISAPI DLL向導已經為創建多線程DLL生成了大量代碼,可是還是有一個嚴重的缺陷:沒有申明本應用程序是一個多線程的程序。所以需要你添加一句話:
IsMultiThread := TRUE;
把這句話放在DPR程序begin-end塊的開始處,使之成為第一句。
如何得知現在是否和Internet連接?
最簡單的辦法是用一個TCP元件得到自己當前的IP,通過判斷IP得知是否連入Internet。例如:
if TCP1.LocalIp = '0.0.0.0' then
ShowMessage('目前沒有連入Internet!');
需要注意的是:因為Internet和Intranet沒本質區別,所以一般不能判定是和Internet連接還是僅僅連入Intranet。當然。你也可以再加一個PING元件,去PING一個比較穩定、速度比較快的站點,如果連通則表明已經接入Internet。不過這種辦法通用性不好。
如何打印一個Web頁面?
可以選用Html控件的AutoPrint方法。例如:
uses Printers;
procedure TForm1.Button1Click(Sender: TObject);
var
OldCur: TCursor;
begin
OldCur := Screen.Cursor;
with Printer do
begin
BeginDoc;
Html1.AutoPrint(handle);
Title := Html1.URL;
EndDoc;
end;
Screen.Cursor := OldCur;
end;
此外還可以利用其PrintPage方法。不過我推薦你采用AutoPrint,因為這樣控制更靈活,可以過濾一些你不希望打印的內容。