您尚未登录,请登录后浏览更多内容! 登录 | 立即注册

QQ登录

只需一步,快速开始

 找回密码
 立即注册

QQ登录

只需一步,快速开始

查看: 8849|回复: 0
打印 上一主题 下一主题

[php学习资料] PHP命令空间namespace及use的用法实践总结

[复制链接]
跳转到指定楼层
楼主
发表于 2020-7-1 23:37:46 | 只看该作者 回帖奖励 |正序浏览 |阅读模式
以下皆为本人自我理解内容,如有失误之处,请多多包涵。
$ s7 M, P4 U* `8 b
文章大纲:
使用namespace的目的
namespace的使用方法
使用use的目的
use的使用方法
& l" @; O' s! E" Q  m  {. Y

, C2 u" Z1 H6 J: `5 k4 u0 o使用namespace的目的:
团队合作项目时,避免与团队其它成员新建的类发生冲突;个人负责项目时,避免前后新建的类发生冲突;

8 |9 r& N6 h2 G
据个人理解,用到所需要的类时,需要先require或include引入,所以会发生类重定义的错误的前提是:两个相同命名的类都有被引入。目前有些php框架会自动加载(即include)所有新建的model类,所以为了避免你新建的model类和项目框架原生的核心类发生重名冲突,采用了namespace。(想了想,与团队成员新建的类产生冲突应该通过沟通避免,即使事发后也应该重新调整类名即时维护,避免后期造成因为对类的理解混淆而带来维护上的复杂度提高)
, |1 ?  k) u: f& l4 A# w
结合使用方法来进一步理解它的使用目的吧。

( R+ \7 m: c2 U' s2 w" K$ Z! p. T; ^5 ~% J7 Q' d$ M
namespace的使用方法:
为了进行测试,我将创建3个文件:1.php和name.php(此文件用来执行测试),后面将不再说明,请自行注意代码的变化。
1.namespace后命名的定义不区分大小写
  1. namespace one;
    * A6 C7 U4 ]. y" ?( L
  2. namespace One;( b7 p. A" N) K9 s3 X
  3. namespace ONE;
复制代码

9 }0 \" k7 Y  E! M% x
如上写法都可以,选择一种作为自己的规范即可。(后面代码我采用第一种进行测试哈)

& J1 _3 \7 Z8 U. z9 ]( P# v0 a2. 没有定义命名空间,就理解为使用顶级命名空间。new类时,可以在类前加上反斜杠\,也可以不加。
  1. //1.php   . G. B( T, j. p6 [
  2. class Person{1 c* R# K7 ^# f8 _5 e1 d. s
  3.     function __construct(){
    5 o1 e7 _& c7 V3 }
  4.             echo 'I am one!';
    " r: u! D) v+ i
  5.         }/ g) Y% Y2 o. z: P
  6. }
复制代码
  1. //name.php8 b3 R3 ]4 q3 n( J
  2. require_once './1.php';1 d6 v- ?9 K( Y5 ^1 J
  3. # ?; y- m  X* ]$ U. w' K: j( W4 u
  4. new Person();     //输出 I am one!;. n( [; I) ^. {- \9 k& q4 J# b
  5. new \Person(); //输出 I am one!;
复制代码
  v: C/ ], q2 o( i$ C# F3 v4 T5 n
3. new类时,带上命名空间时,之间一定用反斜杠字符,而不是顺斜杠。
记忆方法:按找%中斜杠的顺序理解为顺斜杠。(有时说反斜杠,自己都不知道是哪种方向,以前按自左向右上升方向这种方向记忆,现在感觉这个太不靠谱了)
  1. //name.php3 \& u4 |, B$ ^
  2. require_once './1.php';
    8 M7 v( o1 b% D5 A+ ^1 E4 y6 L: B
  3. new /Person();  // 代码报错:Parse error: syntax error, unexpected '/'
复制代码

; `  i2 a( Z7 p4 ~; H+ `. o4.类在指定命名空间下, new类时,一定要带上指定的命名空间。
没有带上指定的命名空间,按照第2点,php就会从顶级命名空间里找这个类。切记:这里不能按照顶级命名空间包含一切其它的命名空间来理解。而应该将顶级命名空间完完全全与其他命名空间区分开。
  1. //1.php   0 Q- y* i3 c6 ?5 c
  2. namespace one;
    6 l: a0 F% P9 q( ?' L; r7 j
  3. class Person{
    # N; {/ u4 U9 y7 I" c
  4.     function __construct(){' h9 ]/ Z) }5 S6 _3 c
  5.             echo 'I am one!';
    & X4 E3 _/ \% c
  6.         }& z$ G3 W' C) h1 b( w
  7. }
复制代码
  1. //name.php/ s/ ~# ^/ ?2 |  i2 Q' [
  2. require_once './1.php';
    / v; x8 c+ E: X- J# z
  3. new \one\Person(); //输出 I am one!;9 k4 f% _" t4 v4 v* {" s/ i! ?
  4. new \Person(); //代码报错:Fatal error: Class 'Person' not found
复制代码
0 \: b. f/ z2 {+ l9 y8 z
可以举个这个通俗例子来理解:带上指定命名空间代表某人的苹果(在他手里),顶级命名空间代表苹果箱里的苹果(在箱子里)。现在要找某人的苹果,就将某人的命名空间带上,否则就会从箱子里找某人的苹果,结果当然是找不到。
" |% a3 G2 ~6 o( z! \* g
5.命名空间声明后的代码便属于这个命名空间,即使有include或require也不影响(重点是对后半句的理解,具体看代码)。9 o4 l* u6 ]$ N2 P& c& y& M
  1. //1.php   1 N, o  W2 y) ~! v: Y
  2. namespace one;
    0 T/ G0 E3 C( _+ |7 p, p* H
  3. class Person{% n3 ?$ r2 k+ }$ d9 j
  4.     function __construct(){* T( ~# E/ ~' W* z& u
  5.             echo 'I am one!';
    8 f3 W; b- e0 s3 g3 @. u; i
  6.         }7 l! w0 C: J  ]
  7. }
复制代码
  1. //name.php0 `# s% J: v# m3 l" R
  2. namespace test;
    2 I0 R: m7 l( n/ ^+ ?9 g* F; M
  3. require './1.php'; 1 [1 a8 b9 n( v) y) f
  4. new \one\Person(); //输出 I am one!;
    " K2 a0 F4 F& Z) X
  5. new Person();      //这里结果会是什么呢,猜猜看
    5 V" b! g6 u! D& v
  6. 最后一行结果报错:
    , @/ a! q. I$ ~# h8 _0 T
  7. Fatal error:  Class 'test\Person' not found
复制代码

7 z1 j" z3 M/ v6 l3 Y0 }
5 q6 j1 z, F6 y2 r: l$ Z
首先,这里与第2点比较一下:
第2点,我说,没有命名空间时,new类时,有没有反斜杠意义一样。
在这里,有了命名空间,有和没有反斜杠的意义就不一样了。
最后一行换成
  1. new \Person();
复制代码

0 h$ R1 l7 }$ C' {
结果报错:
  1. Fatal error:  Class 'Person' not found
复制代码
; ?- T# i) t" j7 a  b2 N
接着,就说说当前这点。
我们可以发现,最后一行代码对应的命名空间为test,并没有受到require文件里的命名空间的影响。
进一步加强验证,我修改了name.php文件如下:
  1. //name.php
    % U$ o) P2 g( B1 H$ j
  2. namespace test;" b$ O2 O  h5 i+ z# |9 D, i
  3. require './1.php'; * O' P6 P5 ]6 T; p' X' m: A$ V
  4. 8 G) ?9 S: `( L& h/ _" P5 z
  5. class Person{
    4 L0 y0 T+ D% Z1 ~2 `$ [7 ^
  6.     function __construct(){
    ) \; i0 \" C* D
  7.             echo 'I am test!';
    ( ?: A" d, x2 ^1 z. w6 {2 S
  8.         }
    0 U! _. A' |) l% F
  9. }. f; V1 D: E+ r/ ^, t
  10. - x2 Y. {5 g5 q# k
  11. new \one\Person(); //输出 I am one!;
    3 K7 J+ X' S/ Y0 }$ c8 C4 \
  12. new Person();    //这里结果会是什么,自己猜猜看
复制代码
* V7 P2 ~% U/ M
最后,这个例子刷新了我对require的认识了。
按照我以前对require的理解:PHP 程序在执行前,就会先读入 require 所指定引入的文件,使它变成 PHP 程序网页的一部分。所以我常常就简单的理解为替换,只不过把抽离出来的代码再放回原处而已。然后我试了将1.php文件内容放到name.php里:
  1. //name.php3 Y/ y* p: U+ _. m5 K# L! Y- E
  2. namespace test;! y! s7 d1 w) \) L, x
  3. namespace one;
    0 m, @- O7 [+ a# B% H
  4. class Person{
    0 n* _2 v. M' G- C% x# d9 K
  5.     function __construct(){
    9 Y2 _* d: a5 o& q
  6.             echo 'I am one!';
    3 [5 W  F: K! e0 \. J
  7.         }( K3 Z1 ]9 N0 c2 ]; [
  8. }- j# ^" m, S5 o% H+ A

  9. / }* `+ F5 p0 G' D6 f; U! x# Z
  10. class Person{( H  _/ Y# ~) Z+ F/ v, U+ R# N
  11.     function __construct(){
    + d" r3 ~6 b* W. y( O
  12.             echo 'I am test!';- j$ F3 a: Y2 g3 P: Y
  13.         }
    , p) p& [" {6 g5 Z8 V( I, E% T
  14. }
复制代码

3 `9 Y: K4 W9 v
无需new类,该文件就会报错:
  1. Fatal error:  Cannot redeclare class one\Person
复制代码

; p: h9 ]) _- R2 v# {
看来简单的把require理解为替换,在这里行不通。
! G! I  G; A0 ~) z
6.namespace里不包含类名称,即使存在与类名称同名的部分,那也不代表类。new类时,还是得将这部分带上。
  u6 s' |! |- R9 m% x
  1. //name.php3 I) g) @7 n: m" P0 u. ^0 L8 P
  2. ; n% s9 U0 i& o4 i! G4 ~* n
  3. namespace test\person;
    $ y" B, ?6 i* a  M

  4. . ^, Y$ `$ K) N" f) \! {: x
  5. class Person{; h4 q9 F$ ?# l) C
  6.     function __construct(){! _3 n6 @5 ?4 D4 W" I
  7.             echo 'I am test!';5 i+ G/ Q9 F5 {% U( E4 \
  8.         }
    , B0 B( D- E# y& H$ e7 y* o5 Q
  9. }
    ; ?# S2 R* l$ v$ ?# }  s% l% A
  10. new \test\person\Person();  //命名空间里person无法代表类名
复制代码
# v, P+ A. F% w: E  v% g" Q+ g
不过这样纯粹是画蛇添足,直接干脆点,namespace里不要带类名称就好。

. ^- ?$ t  T5 o* t6 h5 \7.一个php文件中可以存在多个命名空间,第一个命名空间前不能有任何代码。
只说第一个命名空间前不能有任何代码,之后的命名空间之前可以有代码。这个自行测试即可。
  1. //name.php
    8 z3 ~0 I8 a5 v
  2. & U: h1 |3 L- @+ ^! y
  3. namespace test;
    7 G0 @* K5 O  T4 d. [$ g1 S7 @
  4. echo 'zhai14';
    8 o0 Y$ m( |: R. \3 C+ Z) `
  5. namespace zhai;" x' P6 _5 d" [& x
  6. require './1.php';
复制代码
* v8 f5 x! x6 X4 d/ P. X2 P) a
php命名空间namespace告一段落了,接下来说说use的用途。
使用use的目的:
在命名空间字符串过长时,使用use可以相应的缩短命名空间。
4 K( S: L; W! ~; A, W
use的使用方法:1.new类时,最前面无需用反斜杠。此外,use后没有as时,缩短的命名空间默认为最后一个反斜杠后的内容。/ P: s# M" k7 [5 r5 R, w
  1. //name.php- B' {; q9 D! n! R' F" l+ A5 L

  2. : E5 Y, _; \1 ^6 k: z
  3. namespace animal\dog;
    , x8 p$ u! c0 K3 G

  4. 4 ]. J: e  J+ r+ K8 k
  5. class Life{  N0 K6 G, t' V  ~6 c; g' s
  6.     function __construct(){4 x' j& H. V. D" s
  7.             echo 'dog life!';
    : ]( |/ c" ^: Y0 L$ ]( z
  8.         }
    % N( P7 U0 I. C& {6 H- N
  9. }% z' s) W4 H; j' y

  10. 6 t2 ^+ Q, Y2 N. x+ h6 j
  11. namespace animal\cat;
    . P* I4 e9 r& c+ S2 @: O
  12. : O& N+ L0 R! l
  13. class Life{' m( K/ E6 q! t& N7 Z4 r
  14.     function __construct(){
    , @  \) c; a9 R, q, n8 K
  15.             echo 'cat life!';
    ' c$ U# }$ t6 K* Q) }7 `4 Z9 G% U4 i
  16.         }
    * K6 h7 \$ J9 v" o2 ~
  17. }
    * h' F% L! X, a8 s' N5 S) N5 T

  18. 6 Y  Y( f9 g/ W1 {7 H0 ^0 a3 g
  19. new Life();  //按照代码执行顺序,这里默认animal\cat这个命名空间/ Z% i7 O1 n1 e; f  R3 n8 m8 Y- ~" e. Z
  20. new \animal\dog\Life();  //A( y  X: U# n0 W* n. [) [. Z

  21. 6 w: A7 J& L( ?- f5 g5 L
  22. use animal\dog;  //a! @3 P( r3 b# o& O
  23. new dog\Life();  //B/ y7 f6 t: ?4 c" Z7 `
  24. + i" D1 w) m8 {# W
  25. use animal\dog as d;  //b' k/ N0 B" C2 M! `* P. h  ?- \# d
  26. new d\Life();
复制代码

3 l$ ?* d7 H$ N& l/ V
通过A、B行代码比较,需要注意:
使用use后,new类时,最前面没有反斜杠。
没使用use时,命名空间最前面有反斜杠

$ V% \; X! g1 ^+ o* `- f7 y
通过a、b行代码比较,可以理解:
use后没有as时,缩短的命名空间默认为最后一个反斜杠后的内容。如上的:
  1. use animal\dog;
复制代码

! R3 ^" s+ g9 t) p" W1 P* D( r
相当于
  1. use animal\dog as dog;
复制代码
* E" e7 I' b; }7 F
2.namespace后面不建议加类名,但use后可以。
' I, j: |* v+ G+ U7 G! o5 ~, E" r! o
  1. //name.php
    ; \: \9 @+ P0 ^
  2. 5 F3 Q8 W4 D* t; n# d, s6 q
  3. namespace animal\dog;6 U2 {4 W8 N( k9 W4 l

  4. 2 R; T# y, p4 I  E
  5. class Life{
    , \! @! J" i% S# c. n: V
  6.     function __construct(){
    , {9 F' J" q8 Z1 W; _3 R+ ~! q2 h
  7.             echo 'dog life!';) {7 j' b8 q' p- v
  8.         }
    6 _. k; ?5 e9 X" X
  9. }8 h% X; f5 k4 u) w- i4 Z2 x

  10. 9 r9 ]) B- s! {5 s& ?
  11. namespace animal\cat;3 m) i9 t% W' g/ V) k

  12. 9 l# ?) i3 \: G% C! B7 ~& F" O, j
  13. class Life{
    ; \8 D# Y$ S/ H5 m
  14.     function __construct(){0 b7 I6 g# C; j( f
  15.             echo 'cat life!';+ M: y3 f) v% [" g
  16.         }
    ! e; O4 n, N4 ?# g- v7 ^/ l* h3 o" P
  17. }) L+ J# F( F6 ?8 G$ Q7 S6 L/ S# ?" o
  18. + L% Z2 w6 P6 G4 f
  19. use animal\dog\Life as dog;  
    & W& X+ m) m) L% r4 S( I
  20. new dog();
复制代码

) W/ w* t0 v( [, _8 U; n
如上所示,use后加上类名后,就相当于把类改了个名称:由Life改为dog了。
上面不用as dog就会报错:
  1. Fatal error:  Cannot use animal\dog\Life as Life because the name is already in use
复制代码

$ f- `$ [. F. B- @5 b
因为cat下也有个一样名称的Life类。
可以理解为,使用use后,这个昵称对应的类只能归当前命名空间占有,其它命名空间下不允许存在该类。
  1. //name.php6 l; E3 i! ^* R9 v' ?
  2. ) @6 z5 \9 c2 Q$ g
  3. namespace animal\dog;" c# U$ n. ?0 U* D( w; \- Q3 F

  4. & ~3 S5 P2 C3 O" k$ `
  5. class Life{! J6 q( B: d! W# e* b; |% Y
  6.     function __construct(){- P! v* z2 x9 O9 z: b& m
  7.             echo 'dog life!';
    . G6 c/ ~* y" l7 c, S; D: k1 T
  8.         }
    ( l7 P; y7 U7 Z& s; Z8 Q7 @1 M' v$ ]
  9. }
    " ^3 Y* h4 z* A: j) Q8 q
  10. class Dog{
    * h( ~( K- Z' b0 Y. O
  11.     function __construct(){
    & Q4 C8 k, g4 l4 X
  12.             echo 'dog in dog!';
    6 h& d2 o) f7 }8 `
  13.         }
    " b) f! |* T  y& a4 r9 [0 t) S4 z4 S
  14. }  e; `9 v2 g  {% f( r

  15. 1 H+ v" O% d  L. c( u2 a* N/ {. Q, p
  16. namespace animal\cat;! L/ n0 m. [$ ^
  17. ' m! q. `9 K8 r. t6 @+ [" R
  18. // class Dog{, T$ ]  O3 b, t1 j0 F
  19. //     function __construct(){0 W6 T/ q, k/ ^& `6 Z* h: U) _" ]
  20. //             echo 'dog in cat!';
    4 ?3 D9 {8 C, e$ Y
  21. //         }
    . o5 ?- _# h& y! f. o! N
  22. // }
    5 x- R- w$ y! |1 s' y' U2 m: @
  23. class Life{
    8 p5 u0 A4 Q# j9 |- J' Q8 N' i
  24.     function __construct(){& I, M( W3 O) {" Z! r: I
  25.             echo 'cat life!';- c  X" K% }7 s( q0 A8 }3 s; N& K
  26.         }8 L5 E5 r( e0 T8 k
  27. }
    ' V  @: f, a' z
  28. , W1 g* k1 Q- o3 D$ S
  29. use animal\dog;  
    ' I3 J0 {6 H+ k. u5 T
  30. new dog\Dog();
复制代码
6 |  j+ I. k% a& w' q
如上,使用了
  1. use animal\dog;
复制代码
# a- h  S% R6 u+ u) D( Q
cat
通过上面代码,我想使用use的目的效果(缩短命名空间名称)就很明显了。

+ H5 u5 j* A( m6 Y' t
简单总结一下:
namespace就是划分领域的作用,代表这些东西是属于某个命名空间下的。
use就是起小名的作用,不论写起来还是说起来都可以省不少事儿。

% Y9 f: L. Y3 d' Y+ U2 W/ n+ l# n6 I

& }5 ]  q. K* H, E. g
3 [2 ^6 W- W, R1 i+ s9 s
# i! v$ ~. S, s) V
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享分享 支持支持 反对反对
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

GMT+8, 2024-5-3 00:16 , Processed in 0.127829 second(s), 20 queries .

Copyright © 2001-2024 Powered by cncml! X3.2. Theme By cncml!