-
Notifications
You must be signed in to change notification settings - Fork 105
/
Copy pathvariables.xml
1071 lines (945 loc) · 30.6 KB
/
variables.xml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<?xml version="1.0" encoding="utf-8"?>
<!-- $Revision$ -->
<!-- EN-Revision: 22583751fbfdaa3eaa41aeb6470d1343f5cb2c78 Maintainer: HonestQiao Status: ready -->
<!-- CREDITS: dallas, Gregory, verdana, mowangjuanzi, Luffy -->
<chapter xml:id="language.variables" xmlns="http://docbook.org/ns/docbook" xmlns:xlink="http://www.w3.org/1999/xlink">
<title>变量</title>
<sect1 xml:id="language.variables.basics">
<title>基础</title>
<simpara>
PHP 中的变量用一个美元符号后面跟变量名来表示。变量名是区分大小写的。
</simpara>
<para>
有效的变量名由字母(<literal>A-Z</literal>、<literal>a-z</literal> 或 128 到 255
之间的字节)或者下划线开头,后面跟上任意数量的字母,数字,或者下划线。
按照正常的正则表达式,它将被表述为:<code>^[a-zA-Z_\x80-\xff][a-zA-Z0-9_\x80-\xff]*$</code>。
</para>
<note>
<simpara>
PHP 不支持 Unicode 变量名,但是,某些字符编码(例如 UTF-8)会以这样的方式对字符进行编码,即多字节字符的所有字节都在允许的范围内,从而使其成为有效的变量名。
</simpara>
</note>
<note>
<simpara>
<literal>$this</literal> 是一个特殊的变量,它不能被赋值。PHP 7.1.0
之前,间接赋值(例如通过使用<link linkend="language.variables.variable">可变变量</link>)是可能的。
</simpara>
</note>
&tip.userlandnaming;
<example>
<title>有效变量名</title>
<programlisting role="php">
<![CDATA[
<?php
$var = 'Bob';
$Var = 'Joe';
echo "$var, $Var"; // 输出 "Bob, Joe"
$_4site = 'not yet'; // 合法变量名;以下划线开头
$i站点is = 'mansikka'; // 合法变量名;可以用中文
?>
]]>
</programlisting>
</example>
<example>
<title>无效变量名</title>
<programlisting role="php">
<![CDATA[
<?php
$4site = 'not yet'; // 非法变量名;以数字开头
]]>
</programlisting>
</example>
<simpara>
PHP 接受任意字节序列作为变量名。不遵循上述命名规则的变量名只能在运行时动态访问。有关如何访问变量的信息,请参阅<link
linkend="language.variables.variable">可变变量</link>。
</simpara>
<example>
<title>访问模糊的变量名</title>
<programlisting role="php">
<![CDATA[
<?php
${'invalid-name'} = 'bar';
$name = 'invalid-name';
echo ${'invalid-name'}, " ", $$name;
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
bar bar
]]>
</screen>
</example>
<para>
变量默认始终传值赋值。那也就是说,当表达式的值赋值给变量时,整个原始表达式的值被赋值到目标变量。这意味着,例如,当一个变量的值赋予另外一个变量时,改变其中一个变量的值,将不会影响到另外一个变量。有关这种类型的赋值操作,请参阅<link linkend="language.expressions">表达式</link>一章。
</para>
<para>
PHP 也提供了另外一种方式给变量赋值:<link linkend="language.references">引用赋值</link>。这意味着新的变量简单的引用(换言之,“成为其别名” 或者 “指向”)了原始变量。改动新的变量将影响到原始变量,反之亦然。
</para>
<para>
使用引用赋值,简单地将一个 &
符号加到将要赋值的变量前(源变量)。例如,下列代码片断将输出“My name is Bob”两次:
<informalexample>
<programlisting role="php">
<![CDATA[
<?php
$foo = 'Bob'; // 将 'Bob' 赋给 $foo
$bar = &$foo; // 通过 $bar 引用 $foo
$bar = "My name is $bar"; // 修改 $bar 变量
echo $bar;
echo $foo; // $foo 的值也被修改
?>
]]>
</programlisting>
</informalexample>
</para>
<para>
有一点重要事项必须指出,那就是只有变量才可以引用赋值。
<informalexample>
<programlisting role="php">
<![CDATA[
<?php
$foo = 25;
$bar = &$foo; // 合法的赋值
$bar = &(24 * 7); // 非法; 引用没有名字的表达式
function test()
{
return 25;
}
$bar = &test(); // 无效,因为 test() 没有通过引用返回变量。
?>
]]>
</programlisting>
</informalexample>
</para>
<para>
在 PHP 中声明变量不是必需的,但这是一种非常好的做法。访问未定义的变量将导致 <constant>E_WARNING</constant>(在
PHP 8.0.0 之前,为 <constant>E_NOTICE</constant>)。未定义的变量的默认值为 &null;。<function>isset</function>
语言结构可用于检测变量是否已初始化。
</para>
<para>
<example>
<title>未初始化变量的默认值</title>
<programlisting role="php">
<![CDATA[
<?php
// 未设置和未引用(不使用上下文)的变量。
var_dump($unset_var);
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
Warning: Undefined variable $unset_var in ...
NULL
]]>
</screen>
</example>
</para>
<simpara>
PHP 允许从未定义的变量自动生成数组(自动创建新数组)。
将元素追加到未定义的变量将创建一个新数组,并且不会生成警告。
</simpara>
<example>
<title>未定义变量自动生成数组</title>
<programlisting role="php">
<![CDATA[
<?php
$unset_array[] = 'value'; // 不会生成警告。
?>
]]>
</programlisting>
</example>
<warning>
<simpara>
当将一个文件包含在另一个使用相同变量名的文件中时,依赖未初始化变量的默认值是有问题的。
</simpara>
</warning>
<simpara>
可以使用 <function>unset</function> 语言结构来销毁变量。
</simpara>
<simpara>
有关变量相关函数的信息,请参阅<link linkend="ref.var">变量函数参考</link>。
</simpara>
</sect1>
<sect1 xml:id="language.variables.predefined">
<title>预定义变量</title>
<para>
PHP 提供了许多 <link linkend="reserved.variables">预定义变量</link>。PHP
提供了一套附加的预定数组,这些数组变量包含了来自 web
服务器(如果可用),运行环境,和用户输入的数据。这些数组在每个作用域内自动可用。因此通常被称为自动全局变量(autoglobals)或者超全局变量(superglobals)。(PHP
中没有用户自定义超全局变量的机制。)详情参阅<link linkend="language.variables.superglobals">超全局变量列表</link>。
</para>
<note>
<title>可变变量</title>
<para>
超级全局变量不能被用作函数或类方法中的<link linkend="language.variables.variable">可变变量</link>。
</para>
</note>
<para>
如果某些 <link linkend="ini.variables-order">variables_order</link>
中的变量没有设定,它们的对应的 PHP 预定义数组也是空的。
</para>
</sect1>
<sect1 xml:id="language.variables.scope">
<title>变量作用域</title>
<simpara>
变量的作用域是定义该变量的上下文。PHP 有函数作用域和全局作用域。在函数之外定义的任何变量都仅限于全局作用域。当包含文件时,该文件中的代码继承了包含语句所在行的变量作用域。
</simpara>
<example>
<title>全局变量作用域示例</title>
<programlisting role="php">
<![CDATA[
<?php
$a = 1;
include 'b.inc'; // 变量 $a 将在 b.inc 内可用
?>
]]>
</programlisting>
</example>
<simpara>
在命名函数或<link linkend="functions.anonymous">匿名</link>函数内创建的任何变量都仅限于函数主体的作用域。然而,<link
linkend="functions.arrow">箭头函数</link>会绑定父级作用域中的变量,使其在函数体内可用。如果在函数内部
include 文件,那么包含文件中的变量将如同在调用函数内部定义一样可用。
</simpara>
<example>
<title>局部变量作用域的示例</title>
<programlisting role="php">
<![CDATA[
<?php
$a = 1; // 全局作用域
function test()
{
echo $a; // $a 变量 $a 未定义,因为它引用了 $a 的局部版本
}
?>
]]>
</programlisting>
</example>
<simpara>
下面的示例会生成未定义变量 <constant>E_WARNING</constant>(PHP 8.0.0 之前是 <constant>E_NOTICE</constant>)诊断提示。这是因为
echo 语句引用了局部版本的变量 <varname>$a</varname>,而且在这个作用域内,它并没有被赋值。注意这与
C 语言有一点点不同,因为 C 中的全局变量会自动提供给函数,除非被局部定义特别覆盖。这可能引起一些问题,有些人可能不小心就改变了一个全局变量。PHP
中全局变量在函数中使用时必须声明为 global。
</simpara>
<sect2 xml:id="language.variables.scope.global">
<title><literal>global</literal> 关键字</title>
<simpara>
<literal>global</literal> 关键字用于将变量从全局作用域绑定到局部作用域。该关键字可以与变量列表或单个变量一起使用。将创建引用同名全局变量的局部变量。如果全局变量不存在,则将在全局作用域内创建该变量并赋值为 &null;。
</simpara>
<para>
<example>
<title>使用 <literal>global</literal></title>
<programlisting role="php">
<![CDATA[
<?php
$a = 1;
$b = 2;
function Sum()
{
global $a, $b;
$b = $a + $b;
}
Sum();
echo $b;
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
3
]]>
</screen>
</example>
</para>
<simpara>
在函数中声明了全局变量
<varname>$a</varname> 和 <varname>$b</varname>
之后,对任一变量的所有引用都会指向其全局版本。对于一个函数能够声明的全局变量的最大个数,PHP 没有限制。
</simpara>
<simpara>
在全局作用域内访问变量的第二个办法,是用特殊的 PHP 自定义
<varname>$GLOBALS</varname> 数组。前面的例子可以写成:
</simpara>
<para>
<example>
<title>使用 <varname>$GLOBALS</varname> 替代 global</title>
<programlisting role="php">
<![CDATA[
<?php
$a = 1;
$b = 2;
function Sum()
{
$GLOBALS['b'] = $GLOBALS['a'] + $GLOBALS['b'];
}
Sum();
echo $b;
?>
]]>
</programlisting>
</example>
</para>
<simpara>
<varname>$GLOBALS</varname>
是一个关联数组,每一个变量为一个元素,键名对应变量名,值对应变量的内容。<varname>$GLOBALS</varname>
之所以在全局作用域内存在,是因为 $GLOBALS 是一个<link
linkend="language.variables.superglobals">超全局变量</link>。以下范例显示了超全局变量的用处:
</simpara>
<para>
<example>
<title>演示超全局变量和作用域的例子</title>
<programlisting role="php">
<![CDATA[
<?php
function test_superglobal()
{
echo $_POST['name'];
}
?>
]]>
</programlisting>
</example>
</para>
<note>
<simpara>
在函数外部使用 <literal>global</literal> 关键字不是错误。如果文件是从函数内部 include 的,则可以使用它。
</simpara>
</note>
</sect2>
<sect2 xml:id="language.variables.scope.static">
<title>使用 <literal>static</literal> 变量</title>
<simpara>
变量作用域的另一个重要特性是 <emphasis>static</emphasis> 变量。静态变量仅在局部函数作用域中存在,但当程序执行离开此作用域时,其值并不丢失。看看下面的例子:
</simpara>
<para>
<example>
<title>演示需要静态变量的例子</title>
<programlisting role="php">
<![CDATA[
<?php
function Test()
{
$a = 0;
echo $a;
$a++;
}
?>
]]>
</programlisting>
</example>
</para>
<simpara>
本函数没什么用处,因为每次调用时都会将
<varname>$a</varname> 的值设为 <literal>0</literal> 并输出
<literal>0</literal>。将变量加一的 <varname>$a</varname>++
没有作用,因为一旦退出本函数则变量
<varname>$a</varname> 就不存在了。要写一个不会丢失本次计数值的计数函数,要将变量
<varname>$a</varname> 定义为静态的:
</simpara>
<para>
<example>
<title>使用静态变量的例子</title>
<programlisting role="php">
<![CDATA[
<?php
function test()
{
static $a = 0;
echo $a;
$a++;
}
?>
]]>
</programlisting>
</example>
</para>
<simpara>
现在,变量 <varname>$a</varname> 仅在第一次调用 test() 函数时被初始化,之后每次调用 test() 函数都会输出
<varname>$a</varname> 的值并加一。
</simpara>
<simpara>
静态变量也提供了一种处理递归函数的方法。以下这个简单的函数递归计数到
10,使用静态变量 <varname>$count</varname> 来判断何时停止:
</simpara>
<para>
<example>
<title>静态变量与递归函数</title>
<programlisting role="php">
<![CDATA[
<?php
function test()
{
static $count = 0;
$count++;
echo $count;
if ($count < 10) {
test();
}
$count--;
}
?>
]]>
</programlisting>
</example>
</para>
<para>
在 PHP 8.3.0 之前,静态变量只能使用常量表达式进行初始化。自 PHP 8.3.0 起,还允许使用动态表达式(例如函数调用):
</para>
<para>
<example>
<title>声明静态变量</title>
<programlisting role="php">
<![CDATA[
<?php
function foo(){
static $int = 0; // 正确
static $int = 1+2; // 正确
static $int = sqrt(121); // 自 PHP 8.3.0 起正确
$int++;
echo $int;
}
?>
]]>
</programlisting>
</example>
</para>
<simpara>
匿名函数内的静态变量仅在该特定函数实例中存在。如果每次调用时都重新创建匿名函数,则静态变量将重新初始化。
</simpara>
<example>
<title>匿名函数内的静态变量</title>
<programlisting role="php">
<![CDATA[
<?php
function exampleFunction($input) {
$result = (static function () use ($input) {
static $counter = 0;
$counter++;
return "Input: $input, Counter: $counter\n";
});
return $result();
}
// 调用 exampleFunction 将重新创建匿名函数,因此静态
// 变量不会保留其值。
echo exampleFunction('A'); // 输出:Input: A, Counter: 1
echo exampleFunction('B'); // 输出:Input: B, Counter: 1
?>
]]>
</programlisting>
</example>
<para>
从 PHP 8.1.0
开始,当继承(不是覆盖)使用有静态变量的方法时,继承的方法将会跟父级方法共享静态变量。这意味着方法中的静态变量现在跟静态属性有相同的行为。
</para>
<simpara>
自 PHP 8.3.0 起,可以使用任意表达式初始化静态变量。这意味着也说,像是方法调用可用于初始化静态变量。
</simpara>
<example>
<title>在继承方法中使用静态变量</title>
<programlisting role="php">
<![CDATA[
<?php
class Foo {
public static function counter() {
static $counter = 0;
$counter++;
return $counter;
}
}
class Bar extends Foo {}
var_dump(Foo::counter()); // int(1)
var_dump(Foo::counter()); // int(2)
var_dump(Bar::counter()); // int(3),PHP 8.1.0 之前 int(1)
var_dump(Bar::counter()); // int(4),PHP 8.1.0 之前 int(2)
?>
]]>
</programlisting>
</example>
</sect2>
<sect2 xml:id="language.variables.scope.references">
<title><literal>global</literal> 和 <literal>static</literal> 变量的引用</title>
<simpara>
对于变量的
<link linkend="language.variables.scope.static">static</link> 和
<link linkend="language.variables.scope.global">global</link>
定义是以<link linkend="language.references">引用</link>的方式实现的。例如,在一个函数作用域内部用
<literal>global</literal>
语句导入的一个真正的全局变量实际上是建立了一个到全局变量的引用。这有可能导致预料之外的行为,如以下例子所演示的:
</simpara>
<informalexample>
<programlisting role="php">
<![CDATA[
<?php
function test_global_ref() {
global $obj;
$new = new stdClass;
$obj = &$new;
}
function test_global_noref() {
global $obj;
$new = new stdClass;
$obj = $new;
}
test_global_ref();
var_dump($obj);
test_global_noref();
var_dump($obj);
?>
]]>
</programlisting>
</informalexample>
&example.outputs;
<screen>
<![CDATA[
NULL
object(stdClass)#1 (0) {
}
]]>
</screen>
<simpara>
类似的行为也适用于
<literal>static</literal> 语句。引用并不是静态地存储的:
</simpara>
<informalexample>
<programlisting role="php">
<![CDATA[
<?php
function &get_instance_ref() {
static $obj;
echo 'Static object: ';
var_dump($obj);
if (!isset($obj)) {
$new = new stdClass;
// 将一个引用赋值给静态变量
$obj = &$new;
}
if (!isset($obj->property)) {
$obj->property = 1;
} else {
$obj->property++;
}
return $obj;
}
function &get_instance_noref() {
static $obj;
echo 'Static object: ';
var_dump($obj);
if (!isset($obj)) {
$new = new stdClass;
// 将一个对象赋值给静态变量
$obj = $new;
}
if (!isset($obj->property)) {
$obj->property = 1;
} else {
$obj->property++;
}
return $obj;
}
$obj1 = get_instance_ref();
$still_obj1 = get_instance_ref();
echo "\n";
$obj2 = get_instance_noref();
$still_obj2 = get_instance_noref();
?>
]]>
</programlisting>
</informalexample>
&example.outputs;
<screen>
<![CDATA[
Static object: NULL
Static object: NULL
Static object: NULL
Static object: object(stdClass)#3 (1) {
["property"]=>
int(1)
}
]]>
</screen>
<simpara>
此示例说明,当把引用赋值给静态变量时,第二次调用
<literal>&get_instance_ref()</literal> 函数时不会<emphasis>记住</emphasis>该引用。
</simpara>
</sect2>
</sect1>
<sect1 xml:id="language.variables.variable">
<title>可变变量</title>
<simpara>
有时候使用可变变量名是很方便的。就是说,一个变量的变量名可以动态的设置和使用。一个普通的变量通过声明来设置,例如:
</simpara>
<informalexample>
<programlisting role="php">
<![CDATA[
<?php
$a = 'hello';
?>
]]>
</programlisting>
</informalexample>
<simpara>
一个可变变量获取了一个普通变量的值作为这个可变变量的变量名。在上面的例子中
<emphasis>hello</emphasis> 使用了两个美元符号($)以后,就可以作为一个可变变量的变量了。例如:
</simpara>
<informalexample>
<programlisting role="php">
<![CDATA[
<?php
$$a = 'world';
?>
]]>
</programlisting>
</informalexample>
<simpara>
这时,两个变量都被定义了:<varname>$a</varname> 的内容是“hello”并且
<varname>$hello</varname> 的内容是“world”。因此,以下语句:
</simpara>
<informalexample>
<programlisting role="php">
<![CDATA[
<?php
echo "$a {$$a}";
?>
]]>
</programlisting>
</informalexample>
<simpara>
与以下语句输出完全相同的结果:
</simpara>
<informalexample>
<programlisting role="php">
<![CDATA[
<?php
echo "$a $hello";
?>
]]>
</programlisting>
</informalexample>
<simpara>
它们都会输出:<computeroutput>hello world</computeroutput>。
</simpara>
<simpara>
要将可变变量用于数组,必须解决模棱两可的问题。也就是说,如果解析器看到 <varname>$$a[1]</varname>,那么解析器需要知道是将
<varname>$a[1]</varname> 作为变量,还是将 <varname>$$a</varname> 作为个变量并取出该变量中索引为
<literal>[1]</literal> 的值。解决此问题的语法是,对第一种情况用
<varname>${$a[1]}</varname>,对第二种情况用
<varname>${$a}[1]</varname>。
</simpara>
<simpara>
类的属性也可以通过可变属性名来访问。可变属性名将在该调用所处的作用域内被解析。例如,对于
<varname>$foo->$bar</varname> 表达式,则会在局部作用域来解析 <varname>$bar</varname>
并且其值将被用于 <varname>$foo</varname> 的属性名。对于 <varname>$bar</varname>
是数组单元时也是一样。
</simpara>
<simpara>
也可使用花括号来给属性名清晰定界。最有用是在属性位于数组中,或者属性名包含有多个部分或者属性名包含有非法字符时(例如来自
<function>json_decode</function> 或 <link linkend="book.simplexml">SimpleXML</link>)。
</simpara>
<para>
<example>
<title>可变属性示例</title>
<programlisting role="php">
<![CDATA[
<?php
class Foo {
public $bar = 'I am bar.';
public $arr = ['I am A.', 'I am B.', 'I am C.'];
public $r = 'I am r.';
}
$foo = new Foo();
$bar = 'bar';
$baz = ['foo', 'bar', 'baz', 'quux'];
echo $foo->$bar . "\n";
echo $foo->{$baz[1]} . "\n";
$start = 'b';
$end = 'ar';
echo $foo->{$start . $end} . "\n";
$arr = 'arr';
echo $foo->{$arr[1]} . "\n";
echo $foo->{$arr}[1] . "\n";
?>
]]>
</programlisting>
&example.outputs;
<screen>
<![CDATA[
I am bar.
I am bar.
I am bar.
I am r.
I am B.
]]>
</screen>
</example>
</para>
<warning>
<simpara>
注意,在 PHP 的函数和类的方法中,<link
linkend="language.variables.superglobals">超全局变量</link>不能用作可变变量。<literal>$this</literal>
变量也是一个特殊变量,不能被动态引用。
</simpara>
</warning>
</sect1>
<sect1 xml:id="language.variables.external">
<title>来自 PHP 之外的变量</title>
<sect2 xml:id="language.variables.external.form">
<title>HTML 表单(GET 和 POST)</title>
<simpara>
当一个表单提交给 PHP
脚本时,表单中的信息会自动在脚本中可用。有几个方法访问此信息,例如:
</simpara>
<para>
<example>
<title>一个简单的 HTML 表单</title>
<programlisting role="html">
<![CDATA[
<form action="foo.php" method="POST">
Name: <input type="text" name="username"><br />
Email: <input type="text" name="email"><br />
<input type="submit" name="submit" value="Submit me!" />
</form>
]]>
</programlisting>
</example>
</para>
<para>
只有两种方法可以访问 HTML
表单中的数据。以下列出了当前有效的方法:
</para>
<para>
<example>
<title>从简单的 POST HTML 表单访问数据</title>
<programlisting role="php">
<![CDATA[
<?php
echo $_POST['username'];
echo $_REQUEST['username'];
?>
]]>
</programlisting>
</example>
</para>
<para>
使用 GET 表单也类似,只不过要用适当的 GET 预定义变量。GET
也适用于
<literal>QUERY_STRING</literal>(URL
中在“?”之后的信息)。因此,举例说,<literal>http://www.example.com/test.php?id=3</literal>
包含有可用 <varname>$_GET['id']</varname>
来访问的 GET 数据。参见
<link linkend="reserved.variables.request">$_REQUEST</link>。
</para>
<note>
<para>
变量名中的点和空格被转换成下划线。例如
<literal><input name="a.b" /></literal> 变成了
<literal>$_REQUEST["a_b"]</literal>。
</para>
</note>
<simpara>
PHP 也理解表单变量上下文中的数组(参见<link
linkend="faq.html">相关常见问题</link>)。例如可以将相关的变量编成组,或者用此功能从多选输入框中取得值。例如,将表单
POST 给自己并在提交时显示数据:
</simpara>
<para>
<example>
<title>更复杂的表单变量</title>
<programlisting role="php">
<![CDATA[
<?php
if ($_POST) {
echo '<pre>';
echo htmlspecialchars(print_r($_POST, true));
echo '</pre>';
}
?>
<form action="" method="post">
Name: <input type="text" name="personal[name]" /><br />
Email: <input type="text" name="personal[email]" /><br />
Beer: <br />
<select multiple name="beer[]">
<option value="warthog">Warthog</option>
<option value="guinness">Guinness</option>
<option value="stuttgarter">Stuttgarter Schwabenbräu</option>
</select><br />
<input type="submit" value="submit me!" />
</form>
]]>
</programlisting>
</example>
</para>
<note>
<simpara>
如果外部变量名以有效的数组语法开头,则将会忽略尾随字符。例如,<literal><input name="foo[bar]baz"></literal>
变为 <literal>$_REQUEST['foo']['bar']</literal>。
</simpara>
</note>
<sect3 xml:id="language.variables.external.form.submit">
<title>IMAGE SUBMIT 变量名</title>
<simpara>
当提交表单时,可以用一幅图像代替标准的提交按钮,用类似这样的标记:
</simpara>
<informalexample>
<programlisting role="html">
<![CDATA[
<input type="image" src="image.gif" name="sub" />
]]>
</programlisting>
</informalexample>
<simpara>
当用户点击到图像中的某处时,相应的表单会被传送到服务器,并加上两个变量
<varname>sub_x</varname> 和
<varname>sub_y</varname>。它们包含了用户点击图像的坐标。有经验的用户可能会注意到被浏览器发送的实际变量名包含的是一个点而不是下划线(即
sub.x 和 sub.y),但 PHP 自动将点转换成了下划线。
</simpara>
</sect3>
</sect2>
<sect2 xml:id="language.variables.external.cookies">
<title>HTTP Cookie</title>
<simpara>
PHP 透明地支持 <link
xlink:href="&url.rfc;6265">RFC 6265</link>定义中的
HTTP cookies。Cookies
是一种在远端浏览器端存储数据并能追踪或识别再次访问的用户的机制。可以用
<function>setcookie</function>
函数设定 cookies。Cookies 是
HTTP 信息头中的一部分,因此
SetCookie 函数必须在向浏览器发送任何输出之前调用。对于
<function>header</function> 函数也有同样的限制。Cookie
数据会在相应的 cookie 数据数组中可用,例如
<varname>$_COOKIE</varname> 和
<varname>$_REQUEST</varname>。更多细节和例子见
<function>setcookie</function> 手册页面。
</simpara>
<note>
<simpara>
自 PHP 7.2.34、7.3.23 和 7.4.11 起,出于安全原因,传入 cookie 的
<emphasis>name</emphasis> 不再进行 url 解码。
</simpara>
</note>
<simpara>
如果要将多个值赋给单个 cookie 变量,可以将其赋成数组。例如:
</simpara>
<informalexample>
<programlisting role="php">
<![CDATA[
<?php
setcookie("MyCookie[foo]", 'Testing 1', time()+3600);
setcookie("MyCookie[bar]", 'Testing 2', time()+3600);
?>
]]>
</programlisting>
</informalexample>
<simpara>
尽管 <varname>MyCookie</varname> 在脚本中是单个数组,这将会建立两个单独的 cookie。如果只需为一个 cookie
设定多个值,考虑先在值上使用 <function>serialize</function> 或 <function>explode</function>。
</simpara>
<simpara>
注意,除非路径或者域不同,cookie 将替换浏览器中先前的同名
cookie。因此对于购物车程序,可以保留一个计数器并一起传递,即
</simpara>
<example>
<title>一个 <function>setcookie</function> 的示例</title>
<programlisting role="php">
<![CDATA[
<?php
if (isset($_COOKIE['count'])) {
$count = $_COOKIE['count'] + 1;
} else {
$count = 1;
}
setcookie('count', $count, time()+3600);
setcookie("Cart[$count]", $item, time()+3600);
?>
]]>
</programlisting>
</example>
</sect2>
<sect2 xml:id="language.variables.external.dot-in-names">
<title>变量名中的点</title>
<para>
通常,PHP 不会改变传递给脚本中的变量名。然而应该注意到点(句号)不是
PHP 变量名中的合法字符。至于原因,看看:
<programlisting role="php">
<![CDATA[
<?php
$varname.ext; /* 非法变量名 */
?>
]]>
</programlisting>
这时,解析器看到是一个名为
<varname>$varname</varname>
的变量,后面跟着一个字符串连接运算符,后面跟着一个裸字符串(即没有加引号的字符串,且不匹配任何已知的健名或保留字)'ext'。很明显这不是想要的结果。
</para>
<para>
出于此原因,要注意 PHP
将会自动将变量名中的点替换成下划线。
</para>