cncml手绘网

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

作者: admin    时间: 2020-7-1 23:37
标题: PHP命令空间namespace及use的用法实践总结
以下皆为本人自我理解内容,如有失误之处,请多多包涵。

1 Y. f& O- }% |
文章大纲:
使用namespace的目的
namespace的使用方法
使用use的目的
use的使用方法
4 l0 F4 N! q: T  M

0 s5 o5 f0 _4 F9 X, P使用namespace的目的:
团队合作项目时,避免与团队其它成员新建的类发生冲突;个人负责项目时,避免前后新建的类发生冲突;

( N  Z: W  @4 Z; b+ {0 ?6 g; N
据个人理解,用到所需要的类时,需要先require或include引入,所以会发生类重定义的错误的前提是:两个相同命名的类都有被引入。目前有些php框架会自动加载(即include)所有新建的model类,所以为了避免你新建的model类和项目框架原生的核心类发生重名冲突,采用了namespace。(想了想,与团队成员新建的类产生冲突应该通过沟通避免,即使事发后也应该重新调整类名即时维护,避免后期造成因为对类的理解混淆而带来维护上的复杂度提高)
% G% }, y1 t3 P8 q* j3 n" n$ q* U( x4 i
结合使用方法来进一步理解它的使用目的吧。
5 E2 h5 L9 f: J

* x! B  a; S) ^3 _namespace的使用方法:
为了进行测试,我将创建3个文件:1.php和name.php(此文件用来执行测试),后面将不再说明,请自行注意代码的变化。
1.namespace后命名的定义不区分大小写
  1. namespace one;  E$ r# |; t  V  B! ]
  2. namespace One;
    # [; \6 B  K' x) F3 q, t3 j" M
  3. namespace ONE;
复制代码
" T% r* k+ ?  U$ E0 p& Z
如上写法都可以,选择一种作为自己的规范即可。(后面代码我采用第一种进行测试哈)
% k- J1 T7 {- v, _- |$ i
2. 没有定义命名空间,就理解为使用顶级命名空间。new类时,可以在类前加上反斜杠\,也可以不加。
  1. //1.php   
    ' ?2 h$ N& E0 X: y" {! V6 e
  2. class Person{
    - G2 u, K8 V  a
  3.     function __construct(){/ \9 Q% X4 ~* M1 @4 a
  4.             echo 'I am one!';
    " r  Z6 k& D! P- F0 y: N6 |
  5.         }" y3 c$ Y' L1 z+ M
  6. }
复制代码
  1. //name.php
    6 u) t/ g  y' q( h
  2. require_once './1.php';, n& e5 G. i0 d' ~5 J6 e
  3. / q, V9 q2 G# H' P6 G2 [
  4. new Person();     //输出 I am one!;' T0 R* w. T6 }! F! o2 T) W. J4 s
  5. new \Person(); //输出 I am one!;
复制代码

+ u& q3 v- U# B8 A! y3. new类时,带上命名空间时,之间一定用反斜杠字符,而不是顺斜杠。
记忆方法:按找%中斜杠的顺序理解为顺斜杠。(有时说反斜杠,自己都不知道是哪种方向,以前按自左向右上升方向这种方向记忆,现在感觉这个太不靠谱了)
  1. //name.php; G6 X! x+ U4 U9 \
  2. require_once './1.php';
    # n2 X; ?5 A! G2 n4 G# d( E
  3. new /Person();  // 代码报错:Parse error: syntax error, unexpected '/'
复制代码

& j) C; ]) v2 Z6 O# c: v, }2 m4.类在指定命名空间下, new类时,一定要带上指定的命名空间。
没有带上指定的命名空间,按照第2点,php就会从顶级命名空间里找这个类。切记:这里不能按照顶级命名空间包含一切其它的命名空间来理解。而应该将顶级命名空间完完全全与其他命名空间区分开。
  1. //1.php   
    ) g# D5 f# C! g; J8 O5 W
  2. namespace one;
    $ A, ^' f+ ?# S, r5 u
  3. class Person{- G% F# n4 v. `  I. Q/ k
  4.     function __construct(){) e6 ^) J* O. `. D- @
  5.             echo 'I am one!';9 }) q) ], a( b$ R
  6.         }
    " t3 s" i$ r  I" ]6 s  B  l3 [. j
  7. }
复制代码
  1. //name.php5 \4 N1 j. h& v: _5 B0 t$ ?2 d
  2. require_once './1.php';
    ' P) U& Z3 e# `
  3. new \one\Person(); //输出 I am one!;
    5 R1 p/ |2 R+ R
  4. new \Person(); //代码报错:Fatal error: Class 'Person' not found
复制代码
! f+ v% e* B/ U. g9 G7 m
可以举个这个通俗例子来理解:带上指定命名空间代表某人的苹果(在他手里),顶级命名空间代表苹果箱里的苹果(在箱子里)。现在要找某人的苹果,就将某人的命名空间带上,否则就会从箱子里找某人的苹果,结果当然是找不到。

% L/ j2 Y- i. j8 [. _% B. ?5.命名空间声明后的代码便属于这个命名空间,即使有include或require也不影响(重点是对后半句的理解,具体看代码)。) {( G8 I* s/ p
  1. //1.php   
    ' C% f- f$ A' N9 T( E
  2. namespace one;/ O) Z% Q/ B9 ^( k% P9 p0 F. X
  3. class Person{' [0 s  v% x3 O
  4.     function __construct(){
    5 A/ W: |$ E$ r4 R! O5 |* [
  5.             echo 'I am one!';
    % A. O5 [: c4 [4 b" _* q) G; v( R
  6.         }$ c5 c8 F- [2 I6 A: l
  7. }
复制代码
  1. //name.php
    $ n+ q, k/ T" y5 V
  2. namespace test;
    - D* |- S: s  D' i
  3. require './1.php';
    7 e, @. D1 W" ~, w5 z
  4. new \one\Person(); //输出 I am one!;
    7 Y- c! ]! W" X; L0 T
  5. new Person();      //这里结果会是什么呢,猜猜看
    , ^' o2 k& }! f7 u( Y7 Y* z* m
  6. 最后一行结果报错:8 {/ U4 ], k6 X4 E0 X; ?8 F" |
  7. Fatal error:  Class 'test\Person' not found
复制代码

2 f% C% K- g9 @  a4 X) o3 @0 G
0 c' f$ Z* g% y) d( N& V
首先,这里与第2点比较一下:
第2点,我说,没有命名空间时,new类时,有没有反斜杠意义一样。
在这里,有了命名空间,有和没有反斜杠的意义就不一样了。
最后一行换成
  1. new \Person();
复制代码
8 F0 u6 z4 {2 r# i& E" l2 L
结果报错:
  1. Fatal error:  Class 'Person' not found
复制代码
; m$ z4 E, V8 F
接着,就说说当前这点。
我们可以发现,最后一行代码对应的命名空间为test,并没有受到require文件里的命名空间的影响。
进一步加强验证,我修改了name.php文件如下:
  1. //name.php
    $ Z( a$ }4 o8 v
  2. namespace test;/ N' s! t" t. j# t! {, w2 [! `
  3. require './1.php'; 8 V; S5 z: x. T. c9 t' X9 z$ b3 W
  4. 5 q8 U5 j  P$ |& h* H  J
  5. class Person{
    8 B4 s. s; d3 f& ~9 p" U
  6.     function __construct(){
    ; y% l3 V5 s# L6 O
  7.             echo 'I am test!';
    # B) F2 O; z$ _5 y; q9 J
  8.         }
    $ n6 F5 X7 S2 _
  9. }& Z4 C' m6 N2 ^0 U! D8 i. x, V7 d& M7 ?

  10.   X0 D  z/ {  O+ N7 u4 @
  11. new \one\Person(); //输出 I am one!;
    6 F& c. P; A. E; d7 D$ ~
  12. new Person();    //这里结果会是什么,自己猜猜看
复制代码

: a$ I# p0 H+ \8 T: s
最后,这个例子刷新了我对require的认识了。
按照我以前对require的理解:PHP 程序在执行前,就会先读入 require 所指定引入的文件,使它变成 PHP 程序网页的一部分。所以我常常就简单的理解为替换,只不过把抽离出来的代码再放回原处而已。然后我试了将1.php文件内容放到name.php里:
  1. //name.php) ^% b7 F) D' ?
  2. namespace test;
    1 U) i( q% `" X6 Y' @/ T
  3. namespace one;# K4 X; C* h, M8 ?1 J% B2 F
  4. class Person{+ A) h" i& ^* A
  5.     function __construct(){
    & l1 r. j$ d7 z7 d& C& a
  6.             echo 'I am one!';
    0 @' e2 c& N7 y6 H( A
  7.         }4 x( V; z- `/ R) S% R9 h  `
  8. }) Z) X1 p0 G: x2 T& ?0 w. }, g
  9. 2 F' \5 \/ k/ P+ w* {/ u
  10. class Person{8 |5 g: \+ [+ @$ A+ v+ I) f5 j
  11.     function __construct(){
    . ?) i0 }/ [8 Y7 W- f
  12.             echo 'I am test!';0 C' @' _. b2 Y  }
  13.         }) Q# T* \" S7 ~) U  R
  14. }
复制代码

" ^& A% ]0 }; [+ x# e& Q
无需new类,该文件就会报错:
  1. Fatal error:  Cannot redeclare class one\Person
复制代码

0 p. A! [5 b$ @* j1 T
看来简单的把require理解为替换,在这里行不通。

3 M/ i0 e- X4 `) _) t% e6.namespace里不包含类名称,即使存在与类名称同名的部分,那也不代表类。new类时,还是得将这部分带上。1 Q* J  \, n8 V# n0 \" V0 G4 ?- b
  1. //name.php6 D# }2 g0 b  T% A

  2. 6 }0 J: R& h3 j, ]
  3. namespace test\person;3 d3 |% c) \! J) \  g

  4. 9 w% q. T5 T: K
  5. class Person{
    & l# t6 v" U2 y# [* e3 V" C, a8 y
  6.     function __construct(){: j3 m+ U- u+ P0 g# ?
  7.             echo 'I am test!';
    ( w0 w/ a2 p( v
  8.         }
    % O' d/ v1 k! K/ i
  9. }
    & E  q& p$ c/ W* j% M3 I
  10. new \test\person\Person();  //命名空间里person无法代表类名
复制代码

1 f. F' C) ~' m8 |- c
不过这样纯粹是画蛇添足,直接干脆点,namespace里不要带类名称就好。
- `, q; S' V5 {# q7 U2 Y4 M
7.一个php文件中可以存在多个命名空间,第一个命名空间前不能有任何代码。
只说第一个命名空间前不能有任何代码,之后的命名空间之前可以有代码。这个自行测试即可。
  1. //name.php8 R" }* `" }/ G& m" h) L/ }
  2. * L1 D! Q5 }' s
  3. namespace test;
    & {3 Z. ?2 J9 \, c7 w6 K0 k
  4. echo 'zhai14';0 P& j6 |, H7 N
  5. namespace zhai;0 O1 S9 S( Z8 Y
  6. require './1.php';
复制代码
" p+ w7 Z9 c0 b! l( u$ y6 _; G# v
php命名空间namespace告一段落了,接下来说说use的用途。
使用use的目的:
在命名空间字符串过长时,使用use可以相应的缩短命名空间。

! w6 a0 Y5 f" u) p6 xuse的使用方法:1.new类时,最前面无需用反斜杠。此外,use后没有as时,缩短的命名空间默认为最后一个反斜杠后的内容。
! O! e3 b8 f* F; g: E- v( Z5 n
  1. //name.php7 n5 ~+ }) C- s* ?! C' n. _& Q  ]

  2. ) `% p& K. U& K  P8 t! }  \6 ~
  3. namespace animal\dog;0 {% H" [; x- V' j, Z

  4. , n% p2 l" y+ w" t, z+ G$ v
  5. class Life{
    " `1 [5 W+ r  y, i9 A; M& H
  6.     function __construct(){% ?' Q* I9 K1 t) e& x
  7.             echo 'dog life!';: }1 b# p5 C+ ]. F9 v) ?
  8.         }
    4 m. i- n9 E" v0 ^
  9. }: ~: m) J/ U( O* |

  10. . F, n: j! @  \' U0 g! }
  11. namespace animal\cat;
    ' @0 K7 K& a/ @, f

  12. . u' {( [7 Q4 o2 x
  13. class Life{' y$ E, @" V0 ~2 X+ K9 G
  14.     function __construct(){" b; A7 S4 f7 T, S1 y; T2 C
  15.             echo 'cat life!';" b  C! _: l+ q( @+ A+ m9 t
  16.         }# T% v5 y0 ?; U6 q# ?
  17. }
    $ H0 u; Y3 L7 B6 f0 Y; T# b

  18. ; N* x" W* v  ~# i
  19. new Life();  //按照代码执行顺序,这里默认animal\cat这个命名空间. h; m+ r3 E+ ]. [6 l+ f, V
  20. new \animal\dog\Life();  //A; n8 T8 {  x8 E/ F2 A2 k0 B
  21. - O5 z2 Q5 v; k7 ~2 M
  22. use animal\dog;  //a  e5 H* S4 Y7 m
  23. new dog\Life();  //B6 R' d/ `, w) o! D* @- a
  24. % z. p. q( H4 C5 W9 c
  25. use animal\dog as d;  //b
    / `% R% m# G- b# h& [
  26. new d\Life();
复制代码

* Z" V: o# p1 k
通过A、B行代码比较,需要注意:
使用use后,new类时,最前面没有反斜杠。
没使用use时,命名空间最前面有反斜杠
: t! B8 x, ^0 v0 {5 t
通过a、b行代码比较,可以理解:
use后没有as时,缩短的命名空间默认为最后一个反斜杠后的内容。如上的:
  1. use animal\dog;
复制代码

1 m; ]% I, P$ P7 g
相当于
  1. use animal\dog as dog;
复制代码
; s7 g6 C" u0 ~. t# o
2.namespace后面不建议加类名,但use后可以。
0 g* Y3 _. o) [: L7 z
  1. //name.php4 H6 D3 q( s, o& a2 W& ^! n/ \# W

  2. - {: G( x2 w6 l& p
  3. namespace animal\dog;
    ) f7 `* v' M8 H2 g) I& E
  4. & V3 V1 T6 O: C) c
  5. class Life{
    0 y% B$ c. B+ Z6 R+ I! G3 j
  6.     function __construct(){/ V$ ~$ t5 y& M6 _' t$ R
  7.             echo 'dog life!';
    ! o2 y1 a$ E6 D* X
  8.         }
    $ U& d) k7 b4 F9 p6 k; {2 ]
  9. }
    2 _' }- F; E- r3 r0 o8 d% Z6 R# v
  10. 4 m  D: S* @& a' W* t& T
  11. namespace animal\cat;
    6 U' B' S6 g, L# U

  12. 9 r$ X5 @/ ~+ c2 r8 ~' P
  13. class Life{
    . M* J" k$ s5 Q; ?% d: d+ E
  14.     function __construct(){* a9 u% F+ V% J0 Y* }( l
  15.             echo 'cat life!';
      O/ B2 L( {( o0 O% @1 X  K- O& y
  16.         }
    7 T: l7 x9 W0 [* r, d- j2 k$ C
  17. }3 y; J( Y1 q1 Q6 \

  18. 2 R) X% o* {# s0 S' j
  19. use animal\dog\Life as dog;  
    , a) \3 [) G9 j9 s) z
  20. new dog();
复制代码
: q, J4 S# S* ^( z4 X! j* ~: M
如上所示,use后加上类名后,就相当于把类改了个名称:由Life改为dog了。
上面不用as dog就会报错:
  1. Fatal error:  Cannot use animal\dog\Life as Life because the name is already in use
复制代码
( v" b5 j# Z2 z( b
因为cat下也有个一样名称的Life类。
可以理解为,使用use后,这个昵称对应的类只能归当前命名空间占有,其它命名空间下不允许存在该类。
  1. //name.php+ A$ x. W. {; [& K2 Q7 p2 S* k
  2. - ~$ b- l4 g5 t+ ^6 c
  3. namespace animal\dog;
    ) x+ A% N: R: n6 u8 u
  4. 7 R+ [/ w" L+ N" N7 ^: S) |* `  u" |
  5. class Life{; m+ P, @3 d2 i8 m
  6.     function __construct(){$ \0 G) h- k1 v) v2 w
  7.             echo 'dog life!';
    / J( E  E4 s; d
  8.         }  p' K: X5 [0 J  L, G; I
  9. }0 H3 }  L2 n! v. a0 D: {
  10. class Dog{* Y4 G/ m% z+ p' a# y
  11.     function __construct(){
    " S3 ^1 Y% i0 I- ~+ S
  12.             echo 'dog in dog!';1 g6 c% ]7 e$ p& e2 _% L6 U
  13.         }  k$ g  B7 j/ B" E- J- o. g
  14. }, n: S  K0 P' F$ q% I
  15. ' x# a3 B7 [5 r8 I
  16. namespace animal\cat;
    + L  ?& I5 p- J; A
  17. 3 f/ B1 F7 ?! r. j. r+ S
  18. // class Dog{
    5 Z2 k2 Q1 L# |3 Z/ W0 V
  19. //     function __construct(){
    , S4 A$ v" r4 y% f" ^
  20. //             echo 'dog in cat!';) P8 g9 B: P/ Z+ H* e4 M3 ~
  21. //         }
    & r5 d3 n" C8 f( W  |
  22. // }# Z' V3 c5 K0 l8 c/ ~- \9 @) N
  23. class Life{
    ! S% M  e6 g  I+ _/ ~
  24.     function __construct(){6 ^1 o& z! e8 @- R, r
  25.             echo 'cat life!';
    + g* z1 G+ C- q# N6 K  d5 x
  26.         }
    , E9 m1 E( h8 J
  27. }' z" H* k* h1 c0 }3 N* O0 P
  28. , X0 Y- z0 t0 Z7 c* F! Z
  29. use animal\dog;  ; g: |% H5 v* U+ ]* ]4 [, [
  30. new dog\Dog();
复制代码

  Q! J  w! p7 k. L$ R$ j6 X
如上,使用了
  1. use animal\dog;
复制代码

! p- p, K( Z$ Z2 T6 n
cat
通过上面代码,我想使用use的目的效果(缩短命名空间名称)就很明显了。

' K9 C5 F+ v$ ]8 f2 J; I
简单总结一下:
namespace就是划分领域的作用,代表这些东西是属于某个命名空间下的。
use就是起小名的作用,不论写起来还是说起来都可以省不少事儿。
/ t4 y! ?3 {% g3 \+ s  ?. b

, g$ v# k2 Z( }9 E- q6 @' e( D% n" l1 B( _
( X& T+ C* t8 E1 F/ M& R$ y

* s/ g- X% s# o4 I5 J8 t9 H& k




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