| 
	首先,用perl实现一个Hello World的例子。
 
	复制代码 代码示例:#!/usr/bin/env perluse strict;
 use warnings;
 use CGI::Fast;
 while(my $q = new CGI::Fast)
 {
 print $q->header("text/plain");
 print "Hello World";
 }
 和CGI的区别仅在于多了一个循环来接受请求,CGI::Fast对象和CGI接口是一样的,而且该脚本也可以当做CGI脚本使用。
 
	搭建nginx + FastCGI 环境 
	相关阅读:CentOS5安装Nginx1.4+PHP5.5 FastCGI的步骤详解
 使用nginx的fastcgi cache提高网站的php访问速度
 Nginx中的FastCGI模块(FastCGI)详解
 Nginx学习笔记之有关FastCGI的知识
 教你在Gentoo中配置Nginx+MySQL+PHP(FastCGI)环境
 
	Perl使用CGI::Fast包来提供FastCGI服务,该包提供两种方式来启动FastCGI进程,一个是直接使用该包提供的服务将当前进程变为FastCGI进程,另外一个是使用第三方工具spawn-fcgi来启动。
 nginx配置方式例子:
 
 
	复制代码 代码示例:location / {fastcgi_pass   127.0.0.1:8184;
 fastcgi_param  SCRIPT_FILENAME  /scripts$fastcgi_script_name;
 includefastcgi_params;
 }
 配置好nginx后,使用spawn-fcgi来启动前面的Hello World:
 
 
	复制代码 代码示例:$ spawn-fcgi -n -a 127.0.0.1 -p 8184 -f ./main.pl调试支持
 在前面的命令行里使用了参数-n,让spawn-fcgi不要fork出多个进程,并阻塞,允许用户Ctrl+C来关闭,产品服务器可以去掉这个参数来充分利用服务器的多核来提供更高的并发数。我之前写了一个bash脚本,允许在文件改动的情况下重启服务,方便调试perl程序,代码如下:
 
 
	复制代码 代码示例:#!/bin/bash#PID文件和需要启动的脚本
 PID_FILE=service.pid
 MAIN=main.pl
 #关闭之前启动的进程
 term() {
 test -e $PID_FILE || return
 pid=`cat $PID_FILE`
 kill -s -0 $pid  || return
 echo "Terminating $MAIN $pid"
 rm -f $PID_FILE
 kill $pid
 wait $pid
 }
 #当前脚本退出的时候也关闭启动了的FastCGI进程
 trap "term;exit" SIGINT SIGTERM
 while true
 do
 #首次启动或者文件改动后都需要关闭之前的进程
 term
 #以no fork方式启动脚本来调试,并将PID写入到文件
 spawn-fcgi -n -a 127.0.0.1 -p 8184 -f ./$MAIN  &
 pid=$!
 echo $pid > $PID_FILE
 echo "My Perl service started, pid = $pid"
 #监控文件变化
 files=`find . -name '*.pl' -o -name '*.pm' -o -name '*.html'`
 md5=`md5sum $files|md5sum`
 #wait for file change
 while [[ `md5sum $files|md5sum` = "$md5" ]]
 do
 sleep 1
 done
 echo "File changes detected, restarting service"
 done
 该脚本已在Mac OSX和Linux下测试通过
 路由系统
 做Web开发离不开路由实现,来对不同请求来做出特定的响应。
 路由请求依赖HTTP Method和URI两部分,因此主要就是需要这两者来做分派。
 在CGI中可以通过环境变量REQUEST_METHOD和REQUEST_URI来获取请求方法和URI。
 因此一个简单的路由系统实际上可以分解为一个二级的map,注册路由实际上就是往这个map里放入规则对应的处理函数,而分派请求则是从这个map里根据规则获取对应的处理函数。
 一个简单的例子:
 
 代码示例:
 
 my %routers = ();
 sub not_found
 {
 print "Status: 404\n";
 print "Content-Type: text/html\n\n";
 print<<EOF
 <html>
 <body>
 <h1>404 Not found</h1>
 Cannot find $ENV{REQUEST_PATH}.
 </body>
 </html>
 EOF
 }
 
	sub add_rule{
 my ($method, $path, $callback) = @_;
 my $handlers = $routers{$method};
 $handlers = $routers{$method} = {} if not $handlers;
 $handlers->{$path} = $callback;
 }
 sub dispatch
 {
 my $q = shift;
 my $method = $ENV{REQUEST_METHOD};
 my $uri = $ENV{REQUEST_URI};
 $uri =~ s/\?.*$//;
 my $handler = ($routers{$method} || {})->{$uri} || not_found;
 eval
 {
 &$handler($q);
 };
 print STDERR "Failed to handle $method $uri: $@\n" if $@;
 }
 
	使用这个路由系统的例子:
 
	代码示例:
 sub index
 {
 my ($q) = @_;
 print $q->header('text/plain');
 print "Hello World!";
 }
 router::add_rule('GET', '/', \&index);
 模板系统
 perl提供了大量的模板系统的实现,我个人最喜欢的是Template Toolkit,文档也非常丰富,网站是 http://www.template-toolkit.org/ 。
 将前面的index修改为使用模板的例子:
 
 
	代码示例:
 use Template;
 my $tt = new Template({INCLUDE_PATH => 'templates', INTERPOLATE => 1});
 sub index
 {
 my ($q) = @_;
 my $output = '';
 print $q->header('text/html');
 
	    $tt->process('index.html', {world => 'World'}, $output) || die $tt->error();print $output;
 }
 
	其中templates/index.html文件内容:
 
	(责任编辑:IT)代码示例:
 
 <html>
 <head><title>Demo</title></head>
 <body>
 Hello ${world}
 </body>
 </html>
 
 |