首先,用perl实现一个Hello World的例子。
和CGI的区别仅在于多了一个循环来接受请求,CGI::Fast对象和CGI接口是一样的,而且该脚本也可以当做CGI脚本使用。
Perl使用CGI::Fast包来提供FastCGI服务,该包提供两种方式来启动FastCGI进程,一个是直接使用该包提供的服务将当前进程变为FastCGI进程,另外一个是使用第三方工具spawn-fcgi来启动。
nginx配置方式例子:
配置好nginx后,使用spawn-fcgi来启动前面的Hello World:
调试支持
在前面的命令行里使用了参数-n,让spawn-fcgi不要fork出多个进程,并阻塞,允许用户Ctrl+C来关闭,产品服务器可以去掉这个参数来充分利用服务器的多核来提供更高的并发数。我之前写了一个bash脚本,允许在文件改动的情况下重启服务,方便调试perl程序,代码如下:
该脚本已在Mac OSX和linux下测试通过
路由系统
做Web开发离不开路由实现,来对不同请求来做出特定的响应。
路由请求依赖HTTP Method和URI两部分,因此主要就是需要这两者来做分派。
在CGI中可以通过环境变量REQUEST_METHOD和REQUEST_URI来获取请求方法和URI。
因此一个简单的路由系统实际上可以分解为一个二级的map,注册路由实际上就是往这个map里放入规则对应的处理函数,而分派请求则是从这个map里根据规则获取对应的处理函数。
一个简单的例子:
my %routers = ();
sub not_found
{
print "Status: 404n";
print "Content-Type: text/htmlnn";
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 $@;
}
使用这个路由系统的例子:
模板系统
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文件内容: