用Delphi编写CGI程序(四)

类别:Delphi 点击:0 评论:0 推荐:
用Delphi编写CGI程序(四)
从 这 一 讲 开 始,我 们 将 进 入 CGI 程 序 设 计 的 学 习 过 程。通 过 前 面 几 讲 的 学 习,您 已 经 掌 握 了 CGI 程 序 设 计 的 基 础 知 识。现 在,您 可 以 坐 下 来 编 写 CGI 程 序 了!

 

三、CGI 程 序 设 计
  1、服 务 器 端 附 件 (SSI) 及 网 关
  2、网 关:通 过 WEB 连 接 其 他 协 议

在 编 写 CGI 程 序 的 过 程 中,最 好 遵 循 以 下 几 个 应 用 程 序 设 计 的 要 点:
  (1)提 出 问 题 -- 您 要 解 决 的 问 题
  (2)设 计 阶 段 -- 构 想 出 CGI 程 序 的 基 本 框 架 和 功 能
  (3)编 码 阶 段 -- 用 行 动 实 现 思 想
  (4)程 序 移 植 -- 编 写 可 移 植 的 代 码
  (5)精 益 求 精 -- 使 程 序 更 上 一 层 楼

1、服 务 器 端 附 件 (SSI) 及 网 关

   在 本 节 将 介 绍 服 务 器 端 附 件 SSI(Server Side Include) 和 网 关。严 格 的 说,SSI 并 不 是 CGI 程 序 设 计 的 内 容,但 SSI 可 以 完 成 一 些 简 单 CGI 程 序 所 能 完 成 的 工 作,有 时 SSI 甚 至 是 最 好 的 选 择。因 此,在 这 里 作 简 单 介 绍。

   SSI 定 义 了 一 组 嵌 入 HTML 文 本 的 命 令,在 HTML 文 本 送 往 HTTP 客 户 端 前,WEB 服 务 器 对 这 些 SSI 命 令 先 作 预 处 理,将 处 理 后 的 HTML 文 本 输 出 到 HTTP 客 户 端 的 浏 览 器。

   SSI 的 命 令 格 式 是:

   < !--# 命 令 参 数 =" 值 " -->

   SSI 命 令 同 Java 或 JavaScript 不 同,它 是 在 服 务 器 端 处 理 的,而 不 是 在 客 户 端 处 理 的,这 一 点 同 CGI 程 序 是 相 似 的。当 然,在 SSI 的 功 能 范 围 内,其 优 点 同 CGI 程 序 相 同,可 以 提 高 服 务 器 的 资 源 利 用 率,而 且 在 客 户 端 用 任 何 WEB 浏 览 器 都 可 以 浏 览 含 有 SSI 的 HTML 文 本。

   下 面 是 常 用 的 六 个 SSI 命 令:
  (1)Include命令
  (2)echo 命 令
  (3)exec 命 令
  (4)config 命 令
  (5)fsize 命 令
  (6)flastmod 命 令

   include 命 令

   唯 一 支 持 的 参 数 是 file,这 是 在 当 前 HTML 文 本 中 插 入 file 参 数 指 定 的 文 件 的 内 容。如 果 您 了 解 C 语 言,可 以 看 出 它 和 C 语 言 中 的“#include”命 令 的 功 能 是 一 样 的。例 如,有 两 个 HTML 文 本:main.html 和 header.html,在 main.html 中 使 用 include 命 令:  main.html:
  < html>
  < tilte> Test Include SSI Command < /title>
  < !--#include file="header.html" -->
  < body>
  The above header comes from header.html!
  < /body>
  < /html>
  header.html:
  < H1> This is a title in header.html! < /H1>
  ( 不 过,好 象 OmniHTTPD 不 支 持 include 命 令,:< !)


  echo 命 令

  唯 一 支 持 的 参 数 是 var,它 用 于 显 示 服 务 器 提 供 的 变 量,例 如:
  DOCUMENT_NAME:当 前 文 件 名
  DOCUMENT_URL:SSI 文 本 的 相 对 路 径
  DATE_LOCAL:当 地 日 期
  DATE_GMT:GMT(Creenwich 标 准 时 间 ) 日 期
  LAST_MODIFIED:包 含 此 SSI 命 令 的 文 件 的 最 近 修 改 日 期
  HTTP_USER_AGENT:浏 览 器 名。

  例 如:main.shtml   < html>
  This document was last updated on < !--#echo var="LAST_MODIFIED"-->
  < /html>


   当 您 用 浏 览 器 打 开 main.shtml 时 可 以 看 到 最 后 修 改 时 间。( 要 注 意 的 是,必 须 将 main.shtml 存 储 在 OmniHTTPD 的 HtDocs 目 录 下,在 浏 览 器 中 用 地 址 http://localhost/main.shtml 访 问。)

   exec 命 令
   两 个 参 数 是 cgi 和 cmd。前 者 调 用 一 个 可 执 行 文 件,如 cgi="/cgi-bin/finger.cgi";后 者 调 用 一 个 系 统 命 令,如 cmd="ls"。遗 憾 的 是,OmniHTTPD 不 支 持 此 SSI 命 令 ( 也 许 现 在 的 最 新 版 支 持 )。

   config 命 令
   此 命 令 设 置 服 务 器 处 理 文 件 和 显 示 日 期 的 方 法。它 有 两 个 参 数:
  (1)timefmt,决 定 显 示 日 期 的 格 式。在 UNIX 中 用 man strftime 查 询 可 用 的 值。
  (2)sizefmt,决 定 显 示 文 件 长 度 的 格 式。值 为 bytes 或 addrev。这 个 命 令 在 OmniHTTPD 中 也 不 支 持。

   fsize 命 令  
   此 命 令 显 示 给 定 文 件 的 大 小。参 数 是 file,指 定 文 件 的 路 径 及 文 件 名。

   flastmod 命 令
   此 命 令 显 示 指 定 文 件 的 最 近 修 改 日 期。参 数 是 file,指 定 文 件 的 路 径 及 文 件 名。

2、网 关:通 过 WEB 连 接 其 他 协 议
   HTTP 协 议 无 法 访 问 Internet 的 所 有 资 源,要 访 问 HTTP 协 议 以 外 的 资 源 时 ( 例 如 POP3 和 SMTP 收 发 电 子 邮 件 ),就 需 要 网 关。CGI 程 序 是 实 现 网 关 的 好 方 法。

   在 许 多 UNIX 的 HTTP 服 务 器 中 提 供 一 些 常 用 的 网 关,例 如 finger、wais、archie 等。但 在 OmniHTTPD 中,没 有 提 供 这 些 网 关。但 我 们 可 以 通 过 编 写 CGI 程 序 向 OmniHTTPD 中 增 加 网 关 功 能。

表 单 及 其 处 理

   HTML 表 单 是 WEB 文 档 的 一 部 分 , 用 于 将 用 户 填 写 的 信 息 提 交 给 服 务 器 。 通 常 , 这 些 信 息 传 递 给 CGI 程 序 , CGI 程 序 依 据 输 入 信 息 进 行 一 系 列 操 作 或 数 据 加 工 , 再 生 成 表 示 处 理 结 果 的 HTML 文 档 发 回 给 客 户 浏 览 器 。

   由 此 可 见 , CGI 程 序 的 关 键 部 分 是 得 到 输 入 数 据 和 生 成 HTML 文 档 , 而 操 作 和 数 据 处 理 部 分 同 大 多 数 应 用 程 序 相 同 。 在 这 一 讲 , 我 将 介 绍 如 何 在 Perl 和 Delphi 中 得 到 数 据 及 输 出 HTML 文 档 。

   首 先 , 我 们 建 立 一 个 名 为 greeting.html 的 HTML 文 档 :

   greeting.html 文 件 ( 存 储 在 OmniHTTPD 的 HtDocs 目 录 下 )
   < html>
   < head>
   < title> This is a greeting page! < /title>
   < h1> Greeting < /h1>
   < body>
   < hr>
   < form action="/cgi-bin/greeting.pl" method=POST>
   < p> Your First Name: < input type=text name="firstname" size=60 maxlength=80> < /p>
   < p> Your Last Name: < input type=text name="lastname" size=60 maxlength=80> < /p>
   < hr>
   < p> < input type=submit value="All OK!"> < input type=reset value="Clear All"> < /p>
   < /form>
   < /body>
   < /html>


   下 面 是 Perl 的 CGI 程 序 greeting.pl :
   greeting.pl 文 件 ( 存 储 在 OmniHTTPD 的 cgi-bin 目 录 下 ) # Greeting You!
require "cgi-lib.pl";

# ===================================
# get input values

&ReadParse(*input);
$mFirstName = $input{'firstname'};
$mLastName = $input{'lastname'};

# ===================================
# do some operations here

$mFullName = "$mFirstName $mLastName";

# ===================================
# create HTML document to output

print &PrintHeader;
print "< html>< head>< title> Greeting You! < /title>< /head>\n";
print "< body> Hello, < i>$mFullName< /i> !\n";
print "< hr> by Greeting.pl < /body>< /html>";

# ===================================
# All done!


   通 过 浏 览 http://localhost/greeting.html 测 试 CGI 程 序 。

   在 上 面 的 Perl 程 序 中 , 作 为 CGI 程 序 , 必 须 用 require "cgi-lib.pl" 来 引 用 cgi-lib.pl 文 件 , 此 文 件 有 许 多 用 于 CGI 编 程 的 函 数 和 过 程 。 require 相 当 于 C 中 的 #include , 但 要 注 意 的 是 , require 语 句 后 必 须 有 分 号 。

   ReadParse 过 程 读 取 HTML 表 单 提 交 的 数 据 , 参 数 是 一 个 数 组 指 针 。 要 注 意 的 是 , 在 Perl 中 , 函 数 和 过 程 的 调 用 要 在 前 面 加 上 & 符 号 。

   mFirstName 、 mLastName 和 mFullName 是 变 量 。 在 Perl 中 , 变 量 名 前 要 有 $ 符 号 。

   PrintHeader 函 数 实 际 上 返 回 值 是 "Content-type: text/html\n\n" , 它 告 诉 浏 览 器 下 面 的 数 据 是 HTML 文 档 。

   生 成 HTML 文 档 部 分 很 简 单 , 就 是 用 print 语 句 将 HTML 文 档 的 内 容 输 出 就 可 以 了 。 怎 么 样 , 会 用 Perl 编 写 CGI 程 序 了 吧 ?

   下 面 , 我 们 来 编 写 一 段 Delphi 程 序 来 完 成 相 同 的 功 能 :

   先 关 闭 Delphi 中 的 所 有 项 目 , 选 择 菜 单 File/New , 在 对 话 框 中 选 择 Web Server Application 类 型 , 使 用 CGI Stand-alone excutable 选 项 , 则 出 现 一 个 新 项 目 , 其 主 窗 口 名 为 WebModule1 。

   在 WebModule1 的 Actions 属 性 上 双 击 鼠 标 , 出 现 Actions 属 性 编 辑 窗 口 。 在 窗 口 中 新 建 一 Action , 名 为 WebActionItem1 , 将 其 Default 属 性 设 为 True ; 并 在 它 的 Events 中 双 击 OnAction 事 件 , 添 入 下 面 的 代 码 :
procedure TWebModule1.WebModule1WebActionItem1Action(Sender: TObject;Request: TWebRequest; Response: TWebResponse; var Handled: Boolean);
var
  mFirstName, mLastName, mFullName: string;
  HtmlDoc: string;
begin
// Get Input Values:
  mFirstName := Request.ContentFields.Values['firstname'];
  mLastName := Request.ContentFields.Values['lastname'];
// Do some operations here
  mFullName := mFirstName + ' ' + mLastName;

// Create HTML document to output
  HtmlDoc := '< html>< head>< title> Greeting You! < /title>< /head>';
  HtmlDoc := HtmlDoc + '< body> Hello, < i>' + mFullName + '< /i> !';
  HtmlDoc := HtmlDoc + '< hr> by Greeting.cgi < /body>< /html>';

  Response.Content := HtmlDoc;
end;


   将 此 工 程 的 单 元 存 为 cgimain.pas , 将 工 程 存 为 greeting.dpr 。 编 译 ( 用 Ctrl+F9) 后 , 将 greeting.exe 复 制 到 OmniHTTPD 的 cgi-bin 目 录 下 , 并 改 名 为 greeting.cgi 。 同 时 , 将 前 面 我 们 写 的 greeting.html 做 如 下 修 改 :

   将 < form action="/cgi-bin/greeting.pl" method=POST> 改 成
    < form action="/cgi-bin/greeting.cgi" method=POST>

   这 样 , 通 过 浏 览 http://localhost/greeting.html 就 可 以 测 试 用 Delphi 编 写 的 CGI 程 序 了 。

   从 这 个 程 序 可 以 看 出 , 在 Delphi 中 CGI 程 序 一 得 到 一 个 请 求 就 发 生 WebActionItem 的 OnAction 事 件 。 在 这 个 事 件 中 , 数 据 输 入 和 HTML 文 档 生 成 是 这 样 做 的 :

   通 过 Request.ContentFields.Values[HTML 表 单 元 素 名 ] 得 到 表 单 元 素 的 值 。

   通 过 对 Response.Content 赋 值 生 成 HTML 文 档 。

   下 面 是 Delphi 程 序 的 三 个 文 件 内 容 : -----------------------------------------------------------
greeting.dpr :

program greeting;

{$APPTYPE CONSOLE}

uses
 HTTPApp,
 CGIApp,
 cgimain in 'cgimain.pas' {WebModule1: TWebModule};

{$E cgi}

{$R *.RES}

begin
 Application.Initialize;
 Application.CreateForm(TWebModule1, WebModule1);
 Application.Run;
end.
-----------------------------------------------------------

cgimain.pas :

unit cgimain;

interface

uses Windows, Messages, SysUtils, Classes, HTTPApp;

type
 TWebModule1 = class(TWebModule)
  procedure WebModule1WebActionItem1Action(Sender: TObject;Request: TWebRequest; Response: TWebResponse; var Handled: Boolean);
 private
  { Private declarations }
 public
  { Public declarations }
 end;

var
 WebModule1: TWebModule1;

implementation

{$R *.DFM}

procedure TWebModule1.WebModule1WebActionItem1Action(Sender: TObject; Request: TWebRequest; Response: TWebResponse; var Handled: Boolean);
var
 mFirstName, mLastName, mFullName: string;
 HtmlDoc: string;
begin
 // Get Input Values:
 mFirstName := Request.ContentFields.Values['firstname'];
 mLastName := Request.ContentFields.Values['lastname'];

 // Do some operations here
 mFullName := mFirstName + ' ' + mLastName;

 // Create HTML document to output
 HtmlDoc := '< html>< head>< title> Greeting You! < /title>< /head>';
 HtmlDoc := HtmlDoc + '< body> Hello, < i>' + mFullName + '< /i> !';
 HtmlDoc := HtmlDoc + '< hr> by Greeting.cgi < /body>< /html>';

 Response.Content := HtmlDoc;
end;

end.

-----------------------------------------------------------
cgimain.dfm :
object WebModule1: TWebModule1
 OldCreateOrder = False
 Actions = <
  item
   Default = True
   Name = 'WebActionItem1'
   OnAction = WebModule1WebActionItem1Action
  end>
 Left = 192
 Top = 107
 Height = 150
 Width = 215
end


本文地址:http://com.8s8s.com/it/it5987.htm