©
This document uses PHP Chinese website manual Release
本文档试图解释一些在设置虚拟主机时经常问及的问题。这些示例向你展示了如何在一个服务器上通过基于域名的或是基于IP的虚拟主机来部署多个web站点。另一份关于如何在一个代理服务器后构建基于多个服务器的站点的说明文档也很快就会出来。
您的服务器有只一个IP地址,而在DNS中有很多域名(CNAMES)映射到这个机器。您而您想要在这个机器上运行www.example.com
和www.example.org
两个站点。
在您的Apache服务器配置中创建一个虚拟主机并不会自动在您的DNS中对主机名做相应更新。您必须自己在DNS中添加域名来指向您的IP地址。否则别人是无法看到您的web站点的。您可以在您的hosts
文件中添加这一条目来进行测试,但这种方法仅适用于那些有这些hosts
文件的机器来使用。
# 确保Apache在监听80端口
Listen 80
# 为虚拟主机在所有IP地址上监听
NameVirtualHost *:80
<VirtualHost *:80>
DocumentRoot /www/example1
ServerName www.example.com
# 你可以在这里添加其他指令
</VirtualHost>
<VirtualHost *:80>
DocumentRoot /www/example2
ServerName www.example.org
# 你可以在这里添加其他指令
</VirtualHost>
因为星号匹配所有IP地址,所以主服务器不接收任何请求。因为www.example.com
首先出现在配置文件中,所以它拥有最高优先级,可以认为是默认或主服务器。这意味着如果一个请求不能与某个ServerName
指令相匹配,它将会由第一个<VirtualHost>
段所伺服。
如果您愿意,您可以用确定的IP地址来取代"*
"。在这种情况下,VirtualHost
的参数必须与NameVirtualHost
的参数相符:
NameVirtualHost 172.20.30.40
<VirtualHost 172.20.30.40>
# 其他 ...
然而,当您的IP地址无法确定的时候,使用"*
"是很方便的,比如说,您的ISP给您配置的是动态IP地址,而您又使用了某种动态域名解析系统时。因为"*
"匹配任何IP地址,所以在这种情况下,不论IP地址如何变化,您都不需要另外进行配置。
上述配置就是您在绝大多数情况下使用基于域名的虚拟主机时将要用到的。事实上,仅在一种情况下这样的配置不会让您满意:您想为不同的IP地址或是端口提供不同的内容。
在这里讨论的任何技术都可以推广到使用任意数量的IP地址。
服务器有两个IP地址。一个(172.20.30.40
)用于主服务器server.domain.com
,另外一个(172.20.30.50
)用于构建两个或多个虚拟主机。
Listen 80
# "主"服务器运行于:172.20.30.40
ServerName server.domain.com
DocumentRoot /www/mainserver
# 这是另外一个IP地址
NameVirtualHost 172.20.30.50
<VirtualHost 172.20.30.50>
DocumentRoot /www/example1
ServerName www.example.com
# 你可以在这里添加其他指令 ...
</VirtualHost>
<VirtualHost 172.20.30.50>
DocumentRoot /www/example2
ServerName www.example.org
# 你可以在这里添加其他指令 ...
</VirtualHost>
任何不是针对172.20.30.50
的请求都将由主服务器来伺服。而提交给172.20.30.50
却没有主机名或没有"Host:
"头的请求,都将由www.example.com
伺服。
服务器有两个IP地址(192.168.1.1
和172.20.30.40
)。这个机器位于内部(局域网)网络和外部(广域网)之间。在外部,域名server.example.com
指向外部地址(172.20.30.40
),而在内部则指向内部地址(192.168.1.1
)。
服务器可以为来自内部和外部的请求提供同样的内容,您只需要一个<VirtualHost>
配置段就可以了。
NameVirtualHost 192.168.1.1
NameVirtualHost 172.20.30.40
<VirtualHost 192.168.1.1 172.20.30.40>
DocumentRoot /www/server1
ServerName server.example.com
ServerAlias server
</VirtualHost>
现在,从不同的网络提交的请求都会由同一个<VirtualHost>
段来伺服。
在内网中,您可以使用server
这个名字来代替server.example.com
这个全名。
跟上面一样,在上述的例子里,您可以用"*
"来代替具体的IP地址,这样就可以对所有的地址都返回相同的内容了。
如果您想让同一个IP的不同端口伺服多个域名。您可以借助在NameVirtualHost
指令中定义端口的方法来达到这个目的。如果您想使用不带"name:port
"的<VirtualHost name:port>
或是直接用Listen
指令,您的配置将无法生效。
Listen 80
Listen 8080
NameVirtualHost 172.20.30.40:80
NameVirtualHost 172.20.30.40:8080
<VirtualHost 172.20.30.40:80>
ServerName www.example.com
DocumentRoot /www/domain-80
</VirtualHost>
<VirtualHost 172.20.30.40:8080>
ServerName www.example.com
DocumentRoot /www/domain-8080
</VirtualHost>
<VirtualHost 172.20.30.40:80>
ServerName www.example.org
DocumentRoot /www/otherdomain-80
</VirtualHost>
<VirtualHost 172.20.30.40:8080>
ServerName www.example.org
DocumentRoot /www/otherdomain-8080
</VirtualHost>
一个有两个IP地址(172.20.30.40
和172.20.30.50
)分别对应域名www.example.com
和www.example.org
的配置如下:
Listen 80
<VirtualHost 172.20.30.40>
DocumentRoot /www/example1
ServerName www.example.com
</VirtualHost>
<VirtualHost 172.20.30.50>
DocumentRoot /www/example2
ServerName www.example.org
</VirtualHost>
如果存在主服务器,那么对没有出现在任一个<VirtualHost>
段中的请求(比如,对localhost
的请求)都会由主服务器来伺服。
如果您的服务器有两个IP地址(172.20.30.40
和172.20.30.50
)分别对应域名www.example.com
和www.example.org
。对每个域名,您都希望在80端口和8080端口发布您的网站。您可以这样配置:
Listen 172.20.30.40:80
Listen 172.20.30.40:8080
Listen 172.20.30.50:80
Listen 172.20.30.50:8080
<VirtualHost 172.20.30.40:80>
DocumentRoot /www/example1-80
ServerName www.example.com
</VirtualHost>
<VirtualHost 172.20.30.40:8080>
DocumentRoot /www/example1-8080
ServerName www.example.com
</VirtualHost>
<VirtualHost 172.20.30.50:80>
DocumentRoot /www/example2-80
ServerName www.example.org
</VirtualHost>
<VirtualHost 172.20.30.50:8080>
DocumentRoot /www/example2-8080
ServerName www.example.org
</VirtualHost>
您想在一些地址上配置基于域名的虚拟主机而在另外一些地址上配置基于IP的虚拟主机。
Listen 80
NameVirtualHost 172.20.30.40
<VirtualHost 172.20.30.40>
DocumentRoot /www/example1
ServerName www.example.com
</VirtualHost>
<VirtualHost 172.20.30.40>
DocumentRoot /www/example2
ServerName www.example.org
</VirtualHost>
<VirtualHost 172.20.30.40>
DocumentRoot /www/example3
ServerName www.example3.net
</VirtualHost>
# IP-based
<VirtualHost 172.20.30.50>
DocumentRoot /www/example4
ServerName www.example4.edu
</VirtualHost>
<VirtualHost 172.20.30.60>
DocumentRoot /www/example5
ServerName www.example5.gov
</VirtualHost>
<Virtual_host>
和mod_proxy
模块一起使用下面的例子允许一个前端机器代理一个运行在其他机器上的虚拟主机。在如下示例中,在192.168.111.2
机器上配置了一个同名的虚拟主机。这样,万一在同一台机器上代理了多个主机名,ProxyPreserveHost On
指令能确保指定的主机名顺利通过代理。
<VirtualHost *:*>
ProxyPreserveHost On
ProxyPass / http://192.168.111.2
ProxyPassReverse / http://192.168.111.2/
ServerName hostname.example.com
</VirtualHost>
_default_
"虚拟主机_default_
"虚拟主机这样配置可以捕获所有指向没指定的IP地址和端口的请求。比如:一个没被任何虚拟主机使用的地址/端口对。
<VirtualHost _default_:*>
DocumentRoot /www/default
</VirtualHost>
这样一个使用通配符端口的默认虚拟主机可以有效的防止请求被主服务器接收。
如果一个地址/端口对已经被一个基于域名的虚拟主机使用,那么"_default_
"虚拟主机决不会处理发向这个地址/端口对的请求。如果一个"Host:
"请求头中包含未知信息,或者干脆就没有,那么它会被第一个基于域名的虚拟主机(也就是在配置文件中最先出现的使用了那个地址/端口对的虚拟主机)处理。
您可以用AliasMatch
或RewriteRule
来重写任何请求,使它指向一个简单信息页面(或脚本)。
_default_
"虚拟主机与第一种一样,但我们想让服务器侦听很多端口而第二个"_default_
"虚拟主机单独侦听80端口。
<VirtualHost _default_:80>
DocumentRoot /www/default80
# ...
</VirtualHost>
<VirtualHost _default_:*>
DocumentRoot /www/default
# ...
</VirtualHost>
侦听80端口的"_default_
"虚拟主机(必须出现在所有使用通配符端口的虚拟主机之前)会捕获所有发向一个未指定的IP地址的请求。主服务器将不会用于伺服任何请求。
_default_
"虚拟主机如果我们只想在80端口上建立唯一的一个"_default_
"虚拟主机,我们应该这样配置:
<VirtualHost _default_:80>
DocumentRoot /www/default
...
</VirtualHost>
发向一个未指定地址的80端口的请求将会由这个虚拟主机伺服;而发向未设定地址的其他端口的请求则由主服务器伺服。
如果一个具有www.example.org
域名的虚拟主机(就是基于域名配置示例中的第二个)得到了自己的IP地址。为了避免一些域名服务器或代理服务器在移植期间仍对这个域名做老的解析,我们可以采用一种过渡方法:同时提供新旧两个IP地址的解析。
达到这个目的很简单。因为我们只要简单的把新地址(172.20.30.50
)加入VirtualHost
指令就行了。
Listen 80
ServerName www.example.com
DocumentRoot /www/example1
NameVirtualHost 172.20.30.40
<VirtualHost 172.20.30.40 172.20.30.50>
DocumentRoot /www/example2
ServerName www.example.org
# ...
</VirtualHost>
<VirtualHost 172.20.30.40>
DocumentRoot /www/example3
ServerName www.example.net
ServerAlias *.example.net
# ...
</VirtualHost>
现在这个虚拟主机就可以用新地址(表现为一个基于IP的虚拟主机)和旧地址(表现为一个基于域名的虚拟主机)同时进行访问了。
ServerPath
指令如果我们在同一个服务器上运行了两个基于域名的虚拟主机。为了匹配正确的虚拟主机,客户端必须发送正确的"Host:
"头。而旧的使用HTTP/1.0的客户端无法发送这样的头,这样Apache就无法辨别客户端想要连接哪个虚拟主机(会用主虚拟主机来伺服这个请求)。为了尽量提供向下兼容性,我们可以提供一个主虚拟主机来返回一个页面,在页面中加入指向基于域名的虚拟主机的URL前缀的链接。
NameVirtualHost 172.20.30.40
<VirtualHost 172.20.30.40>
# 主虚拟主机
DocumentRoot /www/subdomain
RewriteEngine On
RewriteRule ^/.* /www/subdomain/index.html
# ...
</VirtualHost>
<VirtualHost 172.20.30.40>
DocumentRoot /www/subdomain/sub1
ServerName www.sub1.domain.tld
ServerPath /sub1/
RewriteEngine On
RewriteRule ^(/sub1/.*) /www/subdomain$1
# ...
</VirtualHost>
<VirtualHost 172.20.30.40>
DocumentRoot /www/subdomain/sub2
ServerName www.sub2.domain.tld
ServerPath /sub2/
RewriteEngine On
RewriteRule ^(/sub2/.*) /www/subdomain$1
# ...
</VirtualHost>
由于ServerPath
指令的作用,发送到http://www.sub1.domain.tld/sub1/
的请求总会被sub1-vhost所伺服。
如果客户端发送了正确的"Host:
"头,发送到http://www.sub1.domain.tld/
的请求只会被sub1-vhost所伺服。如果没有发送"Host:
"头,客户端将会得到从主虚拟主机发送的信息页面。
请注意,这里还有一点小问题:如果客户端没有发送"Host:
"头,发送到http://www.sub2.domain.tld/sub1/
的请求还是会被sub1-vhost所伺服。
RewriteRule
指令用以确保正确发送了"Host:
"头的客户端可以任意使用这两种URL变量,比如说:使用或不使用URL前缀。