cncml手绘网

标题: PHP命令空间namespace及use的用法实践总结 [打印本页]

作者: admin    时间: 2020-7-1 23:37
标题: PHP命令空间namespace及use的用法实践总结
以下皆为本人自我理解内容,如有失误之处,请多多包涵。
$ H( |: G5 Z- f6 f1 Q! Z" x$ k. N
文章大纲:
使用namespace的目的
namespace的使用方法
使用use的目的
use的使用方法
4 W, d1 n3 G9 ~' s/ M# |
* {" d8 R9 b3 M2 y6 f
使用namespace的目的:
团队合作项目时,避免与团队其它成员新建的类发生冲突;个人负责项目时,避免前后新建的类发生冲突;
/ ?7 r; x0 k4 y9 n1 K0 m$ U
据个人理解,用到所需要的类时,需要先require或include引入,所以会发生类重定义的错误的前提是:两个相同命名的类都有被引入。目前有些php框架会自动加载(即include)所有新建的model类,所以为了避免你新建的model类和项目框架原生的核心类发生重名冲突,采用了namespace。(想了想,与团队成员新建的类产生冲突应该通过沟通避免,即使事发后也应该重新调整类名即时维护,避免后期造成因为对类的理解混淆而带来维护上的复杂度提高)

4 z+ }3 k+ r! W  x% K0 `结合使用方法来进一步理解它的使用目的吧。

4 w- X+ M, E; o( B% k+ O5 m; E: M( I" }9 j: }' R: }, j, Y
namespace的使用方法:
为了进行测试,我将创建3个文件:1.php和name.php(此文件用来执行测试),后面将不再说明,请自行注意代码的变化。
1.namespace后命名的定义不区分大小写
  1. namespace one;
    + `8 ]' u) g7 @# Z
  2. namespace One;1 r+ O$ g: A  H: O
  3. namespace ONE;
复制代码
" o/ K! c: ~) }/ W# O2 r
如上写法都可以,选择一种作为自己的规范即可。(后面代码我采用第一种进行测试哈)

+ h- d1 z  j/ T- h3 @( P' \1 e2. 没有定义命名空间,就理解为使用顶级命名空间。new类时,可以在类前加上反斜杠\,也可以不加。
  1. //1.php   
    2 M! t  J3 X7 o
  2. class Person{
    % g, O% |( |, r4 n' m4 ^
  3.     function __construct(){$ I4 H( \8 u& I
  4.             echo 'I am one!';
    9 G( [/ N$ X( u2 y
  5.         }
    4 U% Z  L4 z% L1 S: _
  6. }
复制代码
  1. //name.php
    6 c6 ?; C  r7 \$ Z' s4 t
  2. require_once './1.php';6 r3 u) Z- R3 l6 b$ ~9 A+ x

  3. 8 {0 P( o- o8 E. B% h
  4. new Person();     //输出 I am one!;* N. q, t! s% a& `8 Z7 |3 k# I2 x- z- G
  5. new \Person(); //输出 I am one!;
复制代码

* E& N/ x( N; A  ]3. new类时,带上命名空间时,之间一定用反斜杠字符,而不是顺斜杠。
记忆方法:按找%中斜杠的顺序理解为顺斜杠。(有时说反斜杠,自己都不知道是哪种方向,以前按自左向右上升方向这种方向记忆,现在感觉这个太不靠谱了)
  1. //name.php
      b5 Z: G+ f! |( ], p/ R
  2. require_once './1.php';  O. W7 R% s4 H! Q
  3. new /Person();  // 代码报错:Parse error: syntax error, unexpected '/'
复制代码
# v6 x  A+ J& d
4.类在指定命名空间下, new类时,一定要带上指定的命名空间。
没有带上指定的命名空间,按照第2点,php就会从顶级命名空间里找这个类。切记:这里不能按照顶级命名空间包含一切其它的命名空间来理解。而应该将顶级命名空间完完全全与其他命名空间区分开。
  1. //1.php   4 k1 s5 m" l; J$ T9 t
  2. namespace one;
    - C  s" L3 \, t4 L2 F0 _# I' V: m6 G
  3. class Person{
    1 K* q# N1 C) K2 n# _* M
  4.     function __construct(){
    $ [( Y& s( e7 m4 T' u9 r/ E  E
  5.             echo 'I am one!';
    1 r5 |6 J" X4 x& ?, G2 N. o( k
  6.         }
    9 [* d- c; Q& w9 C+ k
  7. }
复制代码
  1. //name.php
    7 [; v$ F/ H+ v5 ?- i! M/ W
  2. require_once './1.php';& S1 G2 m$ r2 F# e$ j
  3. new \one\Person(); //输出 I am one!;
    & V5 O: E" E7 \# j4 n1 I
  4. new \Person(); //代码报错:Fatal error: Class 'Person' not found
复制代码

1 W. n! T: S$ M* J/ w2 {
可以举个这个通俗例子来理解:带上指定命名空间代表某人的苹果(在他手里),顶级命名空间代表苹果箱里的苹果(在箱子里)。现在要找某人的苹果,就将某人的命名空间带上,否则就会从箱子里找某人的苹果,结果当然是找不到。

7 w; L$ n0 ^' N8 }! ?5.命名空间声明后的代码便属于这个命名空间,即使有include或require也不影响(重点是对后半句的理解,具体看代码)。
& K. G7 a9 E# s  ?& v$ z: m
  1. //1.php   0 }* \1 m8 J; i' M8 t
  2. namespace one;
    4 O" W* K8 S  U! M; w% n
  3. class Person{
    " [& \) f0 B0 u! T
  4.     function __construct(){
    ) o/ |; ?2 N! I. i
  5.             echo 'I am one!';
    % z8 J+ }7 W' }6 J
  6.         }
    + r8 ]4 E/ N" s. ~6 e
  7. }
复制代码
  1. //name.php' x! P8 G! Z2 w" ~8 x. b
  2. namespace test;
    % I% i/ B- @$ c' E# w
  3. require './1.php'; . L+ [* x' T/ ]& T9 `- H
  4. new \one\Person(); //输出 I am one!;
    6 ~" c* X: B- C* M: @
  5. new Person();      //这里结果会是什么呢,猜猜看
    ( W! d) B+ U3 H& J1 F
  6. 最后一行结果报错:$ d5 n: |; ]2 ~4 p
  7. Fatal error:  Class 'test\Person' not found
复制代码

" g5 j. q8 x! \) F# M9 n
7 X" D" d; S$ B( p& v3 C1 R; B; A$ S$ b
首先,这里与第2点比较一下:
第2点,我说,没有命名空间时,new类时,有没有反斜杠意义一样。
在这里,有了命名空间,有和没有反斜杠的意义就不一样了。
最后一行换成
  1. new \Person();
复制代码

* Q  Q1 ?0 M4 i; A5 w2 r
结果报错:
  1. Fatal error:  Class 'Person' not found
复制代码

+ N. d) H6 n+ y2 v# L
接着,就说说当前这点。
我们可以发现,最后一行代码对应的命名空间为test,并没有受到require文件里的命名空间的影响。
进一步加强验证,我修改了name.php文件如下:
  1. //name.php
    6 _5 B' a( Y" S: `
  2. namespace test;
    - y! @1 L( [7 |  K0 F
  3. require './1.php'; + z8 M" Z  E6 D5 h
  4. # s- m2 p/ }) Z7 G% O+ h) n
  5. class Person{
    % y  \4 X8 z4 s/ M
  6.     function __construct(){9 H& @. g* E* ^! B3 u1 C, E! v
  7.             echo 'I am test!';3 j1 t4 j# e( N/ f; \- l1 B
  8.         }
      y2 k( m9 x# m+ u4 h1 n) `
  9. }
    ( q3 `8 A+ r0 b! H2 r  r
  10. ! P5 c2 f& \( _: B1 p
  11. new \one\Person(); //输出 I am one!;2 f3 t7 q5 e, `4 n
  12. new Person();    //这里结果会是什么,自己猜猜看
复制代码
% M4 m/ v5 i% B: T; ^
最后,这个例子刷新了我对require的认识了。
按照我以前对require的理解:PHP 程序在执行前,就会先读入 require 所指定引入的文件,使它变成 PHP 程序网页的一部分。所以我常常就简单的理解为替换,只不过把抽离出来的代码再放回原处而已。然后我试了将1.php文件内容放到name.php里:
  1. //name.php
    . A6 X/ D8 U: x/ C
  2. namespace test;
    0 ^. _0 _+ ^( M0 L( F( f
  3. namespace one;- r! A- S4 A' t6 ~, n2 {3 {
  4. class Person{
    : K- }( E1 I4 O4 j
  5.     function __construct(){$ D2 X- m4 X3 y' K
  6.             echo 'I am one!';
    7 r% h6 D1 R9 ^& Q8 Z+ u! D8 q
  7.         }* v1 H( `4 ^' w. M; J+ `- }
  8. }2 S2 U3 s8 @  E. \) |
  9. % k1 t/ h0 R: ]& `
  10. class Person{
    ' ^, X* E8 G6 o) r8 `- q" g
  11.     function __construct(){
    : J( j' g' f7 r. A! {
  12.             echo 'I am test!';  K& E( G' F+ G
  13.         }2 q. x. X# R6 i0 {# m8 E, O/ e1 M
  14. }
复制代码
/ E. e9 I( j# b6 d
无需new类,该文件就会报错:
  1. Fatal error:  Cannot redeclare class one\Person
复制代码

% ]1 E" ~- l( x/ a: k
看来简单的把require理解为替换,在这里行不通。
8 C7 C8 s6 B2 H& a6 z0 y; n
6.namespace里不包含类名称,即使存在与类名称同名的部分,那也不代表类。new类时,还是得将这部分带上。
1 E$ [( A6 V& T8 R8 G( j  `' r
  1. //name.php
    6 D# I- D2 G" G) A/ y) k1 `: S7 S7 x

  2. 6 s, R; I1 y. V5 V9 e- L
  3. namespace test\person;* X7 E: S6 y/ [$ H
  4. + m' g5 D: y: S4 |+ y0 D
  5. class Person{" Y: J5 Z. p( @: \+ Q" r5 n
  6.     function __construct(){
    - W' P; E; ?' K% t! ?& ^
  7.             echo 'I am test!';
    0 m( g2 V; E' b& c/ ?% f, h
  8.         }
    % ?1 `5 M. L6 `; }) U+ k
  9. }, B& B. m$ W1 M
  10. new \test\person\Person();  //命名空间里person无法代表类名
复制代码
, J! G) g) q* F) _. x
不过这样纯粹是画蛇添足,直接干脆点,namespace里不要带类名称就好。

. X8 [. R' \+ R' l' G' D# I7.一个php文件中可以存在多个命名空间,第一个命名空间前不能有任何代码。
只说第一个命名空间前不能有任何代码,之后的命名空间之前可以有代码。这个自行测试即可。
  1. //name.php  c8 Q/ s$ ~+ i3 T. m* w0 e( B! G

  2. / d. q  O* J" j7 ^: w  G
  3. namespace test;
    $ E- D- h- h- Y1 s3 d
  4. echo 'zhai14';& f. z1 Y3 S( B5 c6 i8 ?' h
  5. namespace zhai;6 g" E( F( h# d
  6. require './1.php';
复制代码

/ E' T' S. T1 [3 P
php命名空间namespace告一段落了,接下来说说use的用途。
使用use的目的:
在命名空间字符串过长时,使用use可以相应的缩短命名空间。

" r* l6 W! F, r7 L) E  p% ouse的使用方法:1.new类时,最前面无需用反斜杠。此外,use后没有as时,缩短的命名空间默认为最后一个反斜杠后的内容。; a% ~1 a) U& H. t0 `2 L9 h3 I
  1. //name.php& v2 W# I2 t' Y

  2. 5 ~$ s# d2 W# k2 o
  3. namespace animal\dog;
    5 y% ]* O- Q# G" Z' b
  4. , |* d# R& u# [1 d- A
  5. class Life{
    . m- X* Q1 q0 K! W- [% B
  6.     function __construct(){* f2 H# D8 p. P; D# q: r# s
  7.             echo 'dog life!';. M7 [! P4 y4 V* G% w; {
  8.         }
    5 {$ o( d) {  g' I" f
  9. }
    ! u/ J3 X- W$ l/ R/ F: m1 X

  10. ) D7 E5 T3 ?" r/ A9 G3 k6 ]
  11. namespace animal\cat;- t2 I# I3 J6 b- n; A
  12. " v: v1 C( R3 O9 B: w7 O
  13. class Life{
    5 j1 K' `! b% C! C) ~" |
  14.     function __construct(){
    & {# m4 Y- {3 ~) A( x& ]
  15.             echo 'cat life!';3 m6 r- W7 l- M  h! |
  16.         }
    1 P' _1 c6 ~: u/ E; G5 I5 f" `
  17. }8 J% _4 P3 @" U0 ~- e
  18. ; `1 O0 A. g' L% n: U& T8 Q
  19. new Life();  //按照代码执行顺序,这里默认animal\cat这个命名空间: d8 ?8 R+ U5 l
  20. new \animal\dog\Life();  //A  V2 F+ b' X3 x) F1 o3 G

  21. ' m7 T1 n! p4 g
  22. use animal\dog;  //a+ T% {+ c  u5 D/ `! k6 t( J
  23. new dog\Life();  //B
    3 L& c; |0 F2 n; H

  24. ! J4 Q1 @+ W8 N7 o
  25. use animal\dog as d;  //b
    5 P! K5 i: {# t/ h9 Y( B
  26. new d\Life();
复制代码

8 Z4 S$ ~; I6 x5 b! @" x$ q2 h
通过A、B行代码比较,需要注意:
使用use后,new类时,最前面没有反斜杠。
没使用use时,命名空间最前面有反斜杠
& H" r7 O! u' c5 n
通过a、b行代码比较,可以理解:
use后没有as时,缩短的命名空间默认为最后一个反斜杠后的内容。如上的:
  1. use animal\dog;
复制代码
9 ?8 t2 a# f, g+ C/ y
相当于
  1. use animal\dog as dog;
复制代码

/ l( e! s" G9 d' W2.namespace后面不建议加类名,但use后可以。
4 G, ?& D$ L! E. y- o
  1. //name.php3 e  p2 t, X' B( y- o& i  E: D

  2. . i& y3 D: v5 u( ~* X# {8 C  ^
  3. namespace animal\dog;/ D# Q% ~) Q+ i# v) Q' p8 u; T
  4. 3 x7 t( C5 E+ F4 _" v
  5. class Life{4 w/ B& M! L+ L: l2 s" q
  6.     function __construct(){/ i. q- \+ D  z2 X7 K, n$ {9 r
  7.             echo 'dog life!';+ h7 m# Y, f" y8 `
  8.         }0 g* q6 u4 a6 `7 H  h$ S: `* ?+ s
  9. }
    & v/ s" a. N/ x" e9 F
  10. ; a. V2 B( A. [8 G' w1 B5 L
  11. namespace animal\cat;
    + o4 o; |9 ?2 O3 K3 n
  12. ' W8 a( i: K  S. `/ h0 |- Z
  13. class Life{3 w8 R, L% ?. @0 T. Q, h# A
  14.     function __construct(){/ O5 Y( }- V; w, ^: L
  15.             echo 'cat life!';
    - E' }' x5 {* q
  16.         }
    " \+ E2 o8 h( W/ j  f
  17. }
    - ?" w3 v" ^# O, M/ o7 L+ A
  18. 6 i; f" C0 g. ]
  19. use animal\dog\Life as dog;  
      n  q; C. r& {3 h( V" Z
  20. new dog();
复制代码

$ H! ?# M8 f% _  o
如上所示,use后加上类名后,就相当于把类改了个名称:由Life改为dog了。
上面不用as dog就会报错:
  1. Fatal error:  Cannot use animal\dog\Life as Life because the name is already in use
复制代码

) L" R& W6 W. l" q! ^
因为cat下也有个一样名称的Life类。
可以理解为,使用use后,这个昵称对应的类只能归当前命名空间占有,其它命名空间下不允许存在该类。
  1. //name.php" a2 [, K, y$ v. D- h

  2. ' k  y+ d  u' J& P# S) \. N/ Z
  3. namespace animal\dog;
    3 K2 {1 g4 _" \* a8 r1 ^) U) }" m
  4. ) V% p( ^( G" b/ r3 K( P& b
  5. class Life{; r) B+ ~. S- x$ K  _4 m
  6.     function __construct(){
    1 y; `4 l  ]: m. l  v; z4 b( B
  7.             echo 'dog life!';
    ' _- O9 ~; ^9 B4 G
  8.         }
    8 v4 M' }5 q3 g) x) |# I
  9. }
    . f: S9 \" [) O; i
  10. class Dog{- R7 q1 G) `" X- t
  11.     function __construct(){
    6 T+ B- R) x9 Y! V
  12.             echo 'dog in dog!';* z! h. k* t6 Z6 H- f
  13.         }
    3 e% k: j2 b5 L7 W- T
  14. }
    ; q7 W+ o) \0 @* h' {% Y

  15.   p( p8 Y; ?% }+ P3 h6 l9 V) J/ A
  16. namespace animal\cat;  r. o2 t% \4 @+ O8 |
  17. $ u' X/ R$ Q$ A, L4 s7 A3 z
  18. // class Dog{, L4 O; B1 j+ _9 \" e7 s
  19. //     function __construct(){0 Z$ Z% {" _% n+ T
  20. //             echo 'dog in cat!';
    ' ]4 X, t) P6 F0 F/ ?
  21. //         }
    2 r" M, `0 M% g1 n6 E) L0 K7 J
  22. // }$ G3 x2 V$ V5 ?+ j3 I
  23. class Life{
    . j" r/ x: H% W' d5 W) j
  24.     function __construct(){
    7 `1 k1 x. b5 w; N& b# d
  25.             echo 'cat life!';* M9 p5 T. }$ I! x8 z+ p
  26.         }  l! q4 Z$ }6 B. Z* J
  27. }' i& O$ \0 ?5 I1 m
  28. ; o" {4 _& g5 q) P! }+ P! n5 x2 h# T
  29. use animal\dog;  
    2 V% H/ k1 O( K2 s' e; b& h# M
  30. new dog\Dog();
复制代码
: W7 r  ~. x/ z7 j4 t. q: M
如上,使用了
  1. use animal\dog;
复制代码

2 x* _8 ~$ z! C% `5 s
cat
通过上面代码,我想使用use的目的效果(缩短命名空间名称)就很明显了。

4 |, F& g0 r/ W$ S( v, m8 p) ^
简单总结一下:
namespace就是划分领域的作用,代表这些东西是属于某个命名空间下的。
use就是起小名的作用,不论写起来还是说起来都可以省不少事儿。

& m5 h' Q8 S( y/ T& f+ e( i& U4 [( N+ v% C. t! J& v
; [( v. f4 p9 P# p3 d" V6 o
+ n9 e2 y0 q! U; u; u4 N5 n5 c6 X
' S5 S$ l. }3 g- z% ?





欢迎光临 cncml手绘网 (http://www.cncml.com/) Powered by Discuz! X3.2