Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
nginx
GitHub Repository: nginx/nginx.org
Path: blob/main/xml/cn/docs/http/server_names.xml
1 views
1
<!--
2
Copyright (C) Igor Sysoev
3
Copyright (C) Nginx, Inc.
4
-->
5
6
<!DOCTYPE article SYSTEM "../../../../dtd/article.dtd">
7
8
<article name="虚拟主机名"
9
link="/cn/docs/http/server_names.html"
10
lang="cn"
11
rev="2"
12
translator="cfsego"
13
author="Igor Sysoev"
14
editor="Brian Mercer">
15
16
17
<section>
18
19
<para>
20
21
虚拟主机名使用<link doc="ngx_http_core_module.xml" id="server_name"/>指令定义,用于决定由某台<link doc="ngx_http_core_module.xml" id="server">虚拟主机</link>来处理请求。具体请参考《<link doc="request_processing.xml">nginx如何处理一个请求</link>》。虚拟主机名可以使用确切的名字,通配符,或者是正则表达式来定义:
22
23
<programlisting>
24
server {
25
listen 80;
26
server_name example.org www.example.org;
27
...
28
}
29
30
server {
31
listen 80;
32
server_name *.example.org;
33
...
34
}
35
36
server {
37
listen 80;
38
server_name mail.*;
39
...
40
}
41
42
server {
43
listen 80;
44
server_name ~^(?&lt;user&gt;.+)\.example\.net$;
45
...
46
}
47
</programlisting>
48
49
nginx以名字查找虚拟主机时,如果名字可以匹配多于一个主机名定义,比如同时匹配了通配符的名字和正则表达式的名字,那么nginx按照下面的优先级别进行查找,并选中第一个匹配的虚拟主机:
50
51
<list type="enum">
52
53
<listitem>
54
确切的名字;
55
</listitem>
56
57
<listitem>
58
最长的以星号起始的通配符名字:<literal>*.example.org</literal>
59
</listitem>
60
61
<listitem>
62
最长的以星号结束的通配符名字:<literal>mail.*</literal>
63
</listitem>
64
65
<listitem>
66
第一个匹配的正则表达式名字(按在配置文件中出现的顺序)。
67
</listitem>
68
69
</list>
70
</para>
71
72
</section>
73
74
75
<section id="wildcard_names"
76
name="通配符名字">
77
78
<para>
79
通配符名字只可以在名字的起始处或结尾处包含一个星号,并且星号与其他字符之间用点分隔。所以,“<literal>www.*.example.org</literal>”和“<literal>w*.example.org</literal>”都是非法的。不过,上面的两个名字可以使用正则表达式描述,即“<literal>~^www\..+\.example\.org$</literal>”和“<literal>~^w.*\.example\.org$</literal>”。星号可以匹配名字的多个节(各节都是以点号分隔的)。“<literal>*.example.org</literal>”不仅匹配<literal>www.example.org</literal>,也匹配<literal>www.sub.example.org</literal>
80
</para>
81
82
<para>
83
有一种形如“<literal>.example.org</literal>”的特殊通配符,它可以既匹配确切的名字“<literal>example.org</literal>”,又可以匹配一般的通配符名字“<literal>*.example.org</literal>”。
84
</para>
85
86
</section>
87
88
89
<section id="regex_names"
90
name="正则表达式名字">
91
92
<para>
93
nginx使用的正则表达式兼容PCRE。为了使用正则表达式,虚拟主机名必须以波浪线“~”起始:
94
95
<programlisting>
96
server_name ~^www\d+\.example\.net$;
97
</programlisting>
98
99
否则该名字会被认为是个确切的名字,如果表达式含星号,则会被认为是个通配符名字(而且很可能是一个非法的通配符名字)。不要忘记设置“<literal>^</literal>”和“<literal>$</literal>”锚点,语法上它们不是必须的,但是逻辑上是的。同时需要注意的是,域名中的点“.”需要用反斜线“\”转义。含有“<literal>{</literal>”和“<literal>}</literal>”的正则表达式需要被引用,如:
100
101
<programlisting>
102
server_name "~^(?&lt;name&gt;\w\d<b>{</b>1,3<b>}</b>+)\.example\.net$";
103
</programlisting>
104
105
否则nginx就不能启动,错误提示是:
106
107
<programlisting>
108
directive "server_name" is not terminated by ";" in ...
109
</programlisting>
110
111
命名的正则表达式捕获组在后面可以作为变量使用:
112
113
<programlisting>
114
server {
115
server_name ~^(www\.)?(<b>?&lt;domain&gt;</b>.+)$;
116
117
location / {
118
root /sites/<b>$domain</b>;
119
}
120
}
121
</programlisting>
122
123
PCRE使用下面语法支持命名捕获组:
124
125
<table note="yes">
126
127
<tr>
128
<td><literal>?&lt;<value>name</value>&gt;</literal></td>
129
<td>从PCRE-7.0开始支持,兼容Perl 5.10语法</td>
130
</tr>
131
132
<tr>
133
<td><literal>?'<value>name</value>'</literal></td>
134
<td>从PCRE-7.0开始支持,兼容Perl 5.10语法</td>
135
</tr>
136
137
<tr>
138
<td><literal>?P&lt;<value>name</value>&gt;</literal></td>
139
<td>从PCRE-4.0开始支持,兼容Python语法</td>
140
</tr>
141
142
</table>
143
144
如果nginx不能启动,并显示错误信息:
145
146
<programlisting>
147
pcre_compile() failed: unrecognized character after (?&lt; in ...
148
</programlisting>
149
150
说明PCRE版本太旧,应该尝试使用<literal>?P&lt;name&gt;</literal>。捕获组也可以以数字方式引用:
151
152
<programlisting>
153
server {
154
server_name ~^(www\.)?(.+)$;
155
156
location / {
157
root /sites/<b>$2</b>;
158
}
159
}
160
</programlisting>
161
162
不过,这种用法只限于简单的情况(比如上面的例子),因为数字引用很容易被覆盖。
163
</para>
164
165
166
</section>
167
168
169
<section id="miscellaneous_names"
170
name="其他类型的名字">
171
172
<para>
173
有一些主机名会被特别对待。
174
</para>
175
176
<para>
177
如果需要用一个非默认的<link doc="ngx_http_core_module.xml" id="server">虚拟主机</link>处理请求头中不含<header>Host</header>字段的请求,需要指定一个空名字:
178
179
<programlisting>
180
server {
181
listen 80;
182
server_name example.org www.example.org "";
183
...
184
}
185
</programlisting>
186
</para>
187
188
<para>
189
如果<link doc="ngx_http_core_module.xml" id="server"/>块中没有定义<link doc="ngx_http_core_module.xml" id="server_name"/>,nginx使用空名字作为虚拟主机名。
190
<note>
191
nginx 0.8.48版本以下(含)在同样的情况下会使用机器名作为虚拟主机名。
192
</note>
193
</para>
194
195
<para>
196
如果以“<literal>$hostname</literal>”(nginx 0.9.4及以上版本)定义虚拟主机名,机器名将被使用。
197
</para>
198
199
<para>
200
如果使用IP地址而不是主机名来请求服务器,那么请求头的<header>Host</header>字段包含的将是IP地址。可以将IP地址作为虚拟主机名来处理这种请求:
201
202
<programlisting>
203
server {
204
listen 80;
205
server_name nginx.org
206
www.nginx.org
207
""
208
<b>192.168.1.1</b>
209
;
210
...
211
}
212
</programlisting>
213
</para>
214
215
<para>
216
在匹配所有的服务器的例子中,可以见到一个奇怪的名字“<literal>_</literal>”:
217
218
<programlisting>
219
server {
220
listen 80 default_server;
221
server_name _;
222
return 444;
223
}
224
</programlisting>
225
226
这没什么特别的,它只不过是成千上万的与真实的名字绝无冲突的非法域名中的一个而已。当然,也可以使用“<literal>--</literal>”和“<literal>!@#</literal>”等等。
227
</para>
228
229
<para>
230
nginx直到0.6.25版本还支持一个特殊的名字“<literal>*</literal>”,这个名字一直被错误地理解成是一个匹配所有的名字。但它从来没有像匹配所有的名字,或者通配符那样工作过,而是用来支持一种功能,此功能现在已经改由<link doc="ngx_http_core_module.xml" id="server_name_in_redirect"/>指令提供支持了。所以,现在这个特殊的名字“<literal>*</literal>”已经过时了,应该使用<link doc="ngx_http_core_module.xml" id="server_name_in_redirect"/>指令取代它。需要注意的是,使用<link doc="ngx_http_core_module.xml" id="server_name"/>指令无法描述匹配所有的名字或者默认服务器。这是<link doc="ngx_http_core_module.xml" id="listen"/>指令的属性,而不是<link doc="ngx_http_core_module.xml" id="server_name"/>指令的属性。具体请参考《<link doc="request_processing.xml">nginx如何处理一个请求</link>》。可以定义两个服务器都监听*:80和*:8080端口,然后指定一个作为端口*:8080的默认服务器,另一个作为端口*:80的默认服务器:
231
232
<programlisting>
233
server {
234
listen 80;
235
listen 8080 default_server;
236
server_name example.net;
237
...
238
}
239
240
server {
241
listen 80 default_server;
242
listen 8080;
243
server_name example.org;
244
...
245
}
246
</programlisting>
247
</para>
248
249
250
</section>
251
252
253
<section id="optimization"
254
name="优化">
255
256
<para>
257
确切名字和通配符名字存储在哈希表中。哈希表和监听端口关联。哈希表的尺寸在配置阶段进行了优化,可以以最小的CPU缓存命中失败来找到名字。设置哈希表的细节参见<link doc="../hash.xml">这篇文档</link>
258
</para>
259
260
<para>
261
nginx首先搜索确切名字的哈希表,如果没有找到,搜索以星号起始的通配符名字的哈希表,如果还是没有找到,继续搜索以星号结束的通配符名字的哈希表。
262
</para>
263
264
<para>
265
因为名字是按照域名的节来搜索的,所以搜索通配符名字的哈希表比搜索确切名字的哈希表慢。注意特殊的通配符名字“<literal>.example.org</literal>”存储在通配符名字的哈希表中,而不在确切名字的哈希表中。
266
</para>
267
268
<para>
269
正则表达式是一个一个串行的测试,所以是最慢的,而且不可扩展。
270
</para>
271
272
<para>
273
鉴于以上原因,请尽可能使用确切的名字。举个例子,如果使用<literal>example.org</literal><literal>www.example.org</literal>来访问服务器是最频繁的,那么将它们明确的定义出来就更为有效:
274
275
<programlisting>
276
server {
277
listen 80;
278
server_name example.org www.example.org *.example.org;
279
...
280
}
281
</programlisting>
282
283
下面这种方法相比更简单,但是效率也更低:
284
285
<programlisting>
286
server {
287
listen 80;
288
server_name .example.org;
289
...
290
}
291
</programlisting>
292
</para>
293
294
<para>
295
如果定义了大量名字,或者定义了非常长的名字,那可能需要在<i>http</i>配置块中使用<link doc="ngx_http_core_module.xml" id="server_names_hash_max_size"/><link doc="ngx_http_core_module.xml" id="server_names_hash_bucket_size"/>指令进行调整。<link doc="ngx_http_core_module.xml" id="server_names_hash_bucket_size"/>的默认值可能是32,或者是64,或者是其他值,取决于CPU的缓存行的长度。如果这个值是32,那么定义“<literal>too.long.server.name.example.org</literal>”作为虚拟主机名就会失败,而nginx显示下面错误信息:
296
<programlisting>
297
could not build the server_names_hash,
298
you should increase server_names_hash_bucket_size: 32
299
</programlisting>
300
301
出现了这种情况,那就需要将指令的值扩大一倍:
302
303
<programlisting>
304
http {
305
server_names_hash_bucket_size 64;
306
...
307
</programlisting>
308
309
如果定义了大量名字,得到了另外一个错误:
310
311
<programlisting>
312
could not build the server_names_hash,
313
you should increase either server_names_hash_max_size: 512
314
or server_names_hash_bucket_size: 32
315
</programlisting>
316
317
那么应该先尝试设置<link doc="ngx_http_core_module.xml" id="server_names_hash_max_size"/>的值差不多等于名字列表的名字总量。如果还不能解决问题,或者服务器启动非常缓慢,再尝试提高<link doc="ngx_http_core_module.xml" id="server_names_hash_bucket_size"/>的值。
318
</para>
319
320
<para>
321
如果只为一个监听端口配置了唯一的主机,那么nginx就完全不会测试虚拟主机名了(也不会为监听端口建立哈希表)。不过,有一个例外,如果定义的虚拟主机名是一个含有捕获组的正则表达式,这时nginx就不得不执行这个表达式以得到捕获组。
322
</para>
323
324
</section>
325
326
327
<section id="compatibility"
328
name="兼容性">
329
330
<para>
331
<list type="bullet">
332
333
<listitem>
334
从0.9.4版本开始,支持特殊的虚拟主机名“<literal>$hostname</literal>”。
335
</listitem>
336
337
<listitem>
338
从0.8.48版本开始,默认的虚拟主机名是空名字“”。
339
</listitem>
340
341
<listitem>
342
从0.8.25版本开始,支持虚拟主机名中使用命名的正则表达式捕获组。
343
</listitem>
344
345
<listitem>
346
从0.7.40版本开始,支持虚拟主机名中使用正则表达式的捕获组。
347
</listitem>
348
349
<listitem>
350
从0.7.12版本开始,支持空名字“”。
351
</listitem>
352
353
<listitem>
354
从0.6.25版本开始,通配符和正则表达式名字可以作为第一个虚拟主机名。
355
</listitem>
356
357
<listitem>
358
从0.6.7版本开始,支持正则表达式的虚拟主机名。
359
</listitem>
360
361
<listitem>
362
从0.6.0版本开始,支持形如<literal>example.*</literal>的通配符名字。
363
</listitem>
364
365
<listitem>
366
从0.3.18版本开始,支持形如<literal>.example.org</literal>的特殊通配符名字。
367
</listitem>
368
369
<listitem>
370
从0.1.13版本开始,支持形如<literal>*.example.org</literal>的通配符名字。
371
</listitem>
372
373
</list>
374
</para>
375
376
</section>
377
378
</article>
379
380