Ruang nama PHP
Apakah ruang nama?
Secara umum, ruang nama ialah cara untuk merangkum sesuatu. Abstraksi ini boleh didapati di banyak tempat.
Sebagai contoh, dalam sistem pengendalian, direktori digunakan untuk mengumpulkan fail berkaitan Untuk fail dalam direktori, ia memainkan peranan ruang nama.
Sebagai contoh, fail foo.txt boleh wujud dalam direktori /home/greg dan /home/other pada masa yang sama, tetapi dua fail foo.txt tidak boleh wujud dalam direktori yang sama.
Selain itu, apabila mengakses fail foo.txt di luar direktori /home/greg, kita mesti meletakkan nama direktori dan pemisah direktori sebelum nama fail untuk mendapatkan /home/greg/foo.txt. Aplikasi prinsip ini dalam bidang pengaturcaraan ialah konsep ruang nama.
Ruang nama PHP (ruang nama) telah ditambahkan dalam PHP 5.3 Jika anda telah mempelajari C# dan Java, maka ruang nama bukanlah perkara baharu. Walau bagaimanapun, ia masih mempunyai kepentingan yang sangat penting dalam PHP.
Peranan ruang nama PHP
Salah satu tujuan ruang nama yang paling jelas adalah untuk menyelesaikan masalah nama pendua atau Kelas dengan nama yang sama muncul, jika tidak ralat maut akan dihasilkan. Dalam kes ini, ia boleh diselesaikan selagi anda mengelak daripada menamakan pertindihan Cara yang paling biasa ialah bersetuju dengan awalan.
Ruang nama PHP boleh menyelesaikan dua jenis masalah berikut:
1 / Konflik nama antara pemalar.
2. Cipta nama alias (atau pendek) untuk nama pengecam yang sangat panjang (biasanya ditakrifkan untuk mengurangkan jenis masalah pertama) untuk meningkatkan kebolehbacaan kod sumber.
Cara PHP mentakrifkan ruang nama
Secara lalai, semua pemalar, kelas dan nama fungsi diletakkan dalam ruang global, sama seperti sokongan Ruang Nama PHP adalah sama seperti dahulu.
Ruang nama diisytiharkan menggunakan ruang nama kata kunci. Jika fail mengandungi ruang nama, ia mesti mengisytiharkan ruang nama sebelum semua kod lain. Format sintaks adalah seperti berikut;
<?php // 定义代码在 'MyProject' 命名空间中 namespace MyProject; // ... 代码 ...
Anda juga boleh menentukan kod ruang nama yang berbeza dalam fail yang sama, seperti:
<?php namespace MyProject1; // MyProject1 命名空间中的PHP代码 namespace MyProject2; // MyProject2 命名空间中的PHP代码 // 另一种语法 namespace MyProject3 { // MyProject3 命名空间中的PHP代码 } ?>
Satu-satunya kod undang-undang sebelum mengisytiharkan ruang nama adalah untuk menentukan fail sumber Pengekodan mengisytiharkan kenyataan. Semua kod bukan PHP, termasuk ruang putih, tidak boleh muncul sebelum pengisytiharan ruang nama.
<?php declare(encoding='UTF-8'); //定义多个命名空间和不包含在命名空间中的代码 namespace MyProject { const CONNECT_OK = 1; class Connection { /* ... */ } function connect() { /* ... */ } } namespace { // 全局代码 session_start(); $a = MyProject\connect(); echo MyProject\Connection::start(); } ?>
Kod berikut akan menyebabkan ralat sintaks:
<html> <?php namespace MyProject; // 命名空间前出现了“<html>” 会致命错误 - 命名空间必须是程序脚本的第一条语句 ?>
Ruang nama sub
sangat serupa dengan direktori dan fail ., ruang nama PHP juga membolehkan anda menentukan nama ruang nama hierarki. Oleh itu, nama ruang nama boleh ditakrifkan dalam cara hierarki:
<?php namespace MyProject\Sub\Level; //声明分层次的单个命名空间 const CONNECT_OK = 1; class Connection { /* ... */ } function Connect() { /* ... */ } ?>
Contoh di atas mencipta MyProjectSubLevelCONNECT_OK yang berterusan, kelas MyProjectSubLevelConnection dan fungsi MyProjectSubLevelConnect.
Penggunaan ruang nama
Nama kelas dalam ruang nama PHP boleh dirujuk dalam tiga cara:
1 Nama tidak layak, atau nama kelas tanpa awalan, seperti $a=new foo( atau foo:: staticmethod() ;. Jika ruang nama semasa ialah currentnamespace, foo akan diselesaikan kepada currentnamespacefoo. Jika kod yang menggunakan foo bersifat global dan tidak mengandungi kod dalam mana-mana ruang nama, foo akan diselesaikan sebagai foo. Amaran: Jika fungsi atau pemalar dalam ruang nama tidak ditentukan, fungsi tidak layak atau nama pemalar diselesaikan kepada fungsi global atau nama tetap.
2. Nama yang layak, atau nama yang mengandungi awalan, seperti $a = new subnamespacefoo( atau subnamespacefoo::staticmethod();. Jika ruang nama semasa ialah currentnamespace, foo akan diselesaikan kepada currentnamespacesubnamespacefoo. Jika kod yang menggunakan foo adalah global, kod tidak terkandung dalam mana-mana ruang nama, foo akan diselesaikan kepada subnamespacefoo.
3. Nama yang layak sepenuhnya, atau nama yang termasuk operator awalan global, contohnya, $a = new currentnamespacefoo( atau currentnamespacefoo::staticmethod();. Dalam kes ini, foo sentiasa diselesaikan kepada nama literal currentnamespacefoo dalam kod.
Berikut ialah contoh menggunakan tiga kaedah ini:
kod fail1.php
<?php namespace Foo\Bar\subnamespace; const FOO = 1; function foo() {} class foo { static function staticmethod() {} } ?>
kod fail fail2.php
<?php namespace Foo\Bar; include 'file1.php'; const FOO = 2; function foo() {} class foo { static function staticmethod() {} } /* 非限定名称 */ foo(); // 解析为 Foo\Bar\foo resolves to function Foo\Bar\foo foo::staticmethod(); // 解析为类 Foo\Bar\foo的静态方法staticmethod。resolves to class Foo\Bar\foo, method staticmethod echo FOO; // resolves to constant Foo\Bar\FOO /* 限定名称 */ subnamespace\foo(); // 解析为函数 Foo\Bar\subnamespace\foo subnamespace\foo::staticmethod(); // 解析为类 Foo\Bar\subnamespace\foo, // 以及类的方法 staticmethod echo subnamespace\FOO; // 解析为常量 Foo\Bar\subnamespace\FOO /* 完全限定名称 */ \Foo\Bar\foo(); // 解析为函数 Foo\Bar\foo \Foo\Bar\foo::staticmethod(); // 解析为类 Foo\Bar\foo, 以及类的方法 staticmethod echo \Foo\Bar\FOO; // 解析为常量 Foo\Bar\FOO ?>
Berhati-hati untuk akses mana-mana kelas, fungsi atau pemalar Global boleh menggunakan nama yang layak sepenuhnya, seperti strlen() atau Exception atau INI_ALL.
Akses kelas global, fungsi dan pemalar di dalam ruang nama:
<?php namespace Foo; function strlen() {} const INI_ALL = 3; class Exception {} $a = \strlen('hi'); // 调用全局函数strlen $b = \INI_ALL; // 访问全局常量 INI_ALL $c = new \Exception('error'); // 实例化全局类 Exception ?>
Ruang nama dan ciri bahasa dinamik
PHP Pelaksanaan ruang nama dipengaruhi oleh sifat dinamik bahasa itu sendiri. Jadi, jika anda ingin menukar kod di bawah kepada ruang nama, akses elemen secara dinamik.
example1.php kod fail:
<?php class classname { function __construct() { echo __METHOD__,"\n"; } } function funcname() { echo __FUNCTION__,"\n"; } const constname = "global"; $a = 'classname'; $obj = new $a; // prints classname::__construct $b = 'funcname'; $b(); // prints funcname echo constant('constname'), "\n"; // prints global ?>
Mesti menggunakan nama yang layak sepenuhnya (nama kelas termasuk awalan ruang nama). Ambil perhatian bahawa garis miring ke belakang utama tidak diperlukan kerana tiada perbezaan antara nama layak dan layak sepenuhnya dalam nama kelas dinamik, nama fungsi atau nama tetap.
Akses dinamik kepada elemen ruang nama
<?php namespace namespacename; class classname { function __construct() { echo __METHOD__,"\n"; } } function funcname() { echo __FUNCTION__,"\n"; } const constname = "namespaced"; include 'example1.php'; $a = 'classname'; $obj = new $a; // prints classname::__construct $b = 'funcname'; $b(); // prints funcname echo constant('constname'), "\n"; // prints global /* note that if using double quotes, "\namespacename\classname" must be used */ $a = '\namespacename\classname'; $obj = new $a; // prints namespacename\classname::__construct $a = 'namespacename\classname'; $obj = new $a; // also prints namespacename\classname::__construct $b = 'namespacename\funcname'; $b(); // prints namespacename\funcname $b = '\namespacename\funcname'; $b(); // also prints namespacename\funcname echo constant('\namespacename\constname'), "\n"; // prints namespaced echo constant('namespacename\constname'), "\n"; // also prints namespaced ?>
kata kunci ruang nama dan pemalar __NAMESPACE__
PHP menyokong dua abstraksi Kaedah untuk mengakses elemen dalam ruang nama semasa, pemalar ajaib __NAMESPACE__ dan kata kunci ruang nama.
Nilai pemalar __NAMESPACE__ ialah rentetan yang mengandungi nama ruang nama semasa. Dalam kod global, tidak termasuk dalam mana-mana ruang nama, ia mengandungi rentetan kosong.
__NAMESPACE__ contoh, kod dalam ruang nama
<?php namespace MyProject; echo '"', __NAMESPACE__, '"'; // 输出 "MyProject" ?> __NAMESPACE__ 示例,全局代码 <?php echo '"', __NAMESPACE__, '"'; // 输出 "" ?>
Malar __NAMESPACE__ berguna apabila mencipta nama secara dinamik, contohnya:
Buat nama secara dinamik menggunakan __NAMESPACE__
<?php namespace MyProject; function get($classname) { $a = __NAMESPACE__ . '\' . $classname; return new $a; } ?><🎜 Ruang nama kata kunci boleh digunakan untuk mengakses elemen secara eksplisit dalam ruang nama atau sub-ruang nama semasa. Ia bersamaan dengan pengendali diri dalam kelas. pengendali ruang nama, kod dalam ruang nama
<?php namespace MyProject; use blah\blah as mine; // see "Using namespaces: importing/aliasing" blah\mine(); // calls function blah\blah\mine() namespace\blah\mine(); // calls function MyProject\blah\mine() namespace\func(); // calls function MyProject\func() namespace\sub\func(); // calls function MyProject\sub\func() namespace\cname::method(); // calls static method "method" of class MyProject\cname $a = new namespace\sub\cname(); // instantiates object of class MyProject\sub\cname $b = namespace\CONSTANT; // assigns value of constant MyProject\CONSTANT to $b ?>pengendali ruang nama, kod global
<?php namespace\func(); // calls function func() namespace\sub\func(); // calls function sub\func() namespace\cname::method(); // calls static method "method" of class cname $a = new namespace\sub\cname(); // instantiates object of class sub\cname $b = namespace\CONSTANT; // assigns value of constant CONSTANT to $b ?>
Menggunakan ruang nama: Alias/Import
Sokongan ruang nama PHP mempunyai dua cara menggunakan alias atau import: menggunakan alias untuk nama kelas atau penggunaan nama Ruang Nama alias. Ambil perhatian bahawa PHP tidak menyokong fungsi pengimportan atau pemalar.
Dalam PHP, alias dilaksanakan melalui penggunaan operator Berikut ialah contoh menggunakan ketiga-tiga kaedah import yang mungkin:
1. Gunakan operator untuk mengimport/menggunakan alias
<?php namespace foo; use My\Full\Classname as Another; // 下面的例子与 use My\Full\NSname as NSname 相同 use My\Full\NSname; // 导入一个全局类 use \ArrayObject; $obj = new namespace\Another; // 实例化 foo\Another 对象 $obj = new Another; // 实例化 My\Full\Classname 对象 NSname\subns\func(); // 调用函数 My\Full\NSname\subns\func $a = new ArrayObject(array(1)); // 实例化 ArrayObject 对象 // 如果不使用 "use \ArrayObject" ,则实例化一个 foo\ArrayObject 对象 ?>
2. Satu baris mengandungi berbilang pernyataan penggunaan
<?php use My\Full\Classname as Another, My\Full\NSname; $obj = new Another; // 实例化 My\Full\Classname 对象 NSname\subns\func(); // 调用函数 My\Full\NSname\subns\func ?>
Operasi import dilaksanakan semasa penyusunan, tetapi nama kelas dinamik, nama fungsi atau nama tetap tidak.
3. Import dan nama dinamik
<?php use My\Full\Classname as Another, My\Full\NSname; $obj = new Another; // 实例化一个 My\Full\Classname 对象 $a = 'Another'; $obj = new $a; // 实际化一个 Another 对象 ?>
Selain itu, operasi import hanya mempengaruhi nama yang tidak layak dan nama yang layak. Nama yang layak sepenuhnya tidak terjejas oleh import kerana ia bersifat deterministik.
4. Import dan nama yang layak sepenuhnya
<?php use My\Full\Classname as Another, My\Full\NSname; $obj = new Another; // instantiates object of class My\Full\Classname $obj = new \Another; // instantiates object of class Another $obj = new Another\thing; // instantiates object of class My\Full\Classname\thing $obj = new \Another\thing; // instantiates object of class Another\thing ?>
Gunakan ruang nama: fungsi/pemalar global sandaran
dalam ruang bernama , apabila PHP menemui kelas, fungsi atau nama tetap yang tidak layak, ia menggunakan strategi keutamaan yang berbeza untuk menyelesaikan nama tersebut. Nama kelas sentiasa diselesaikan kepada nama dalam ruang nama semasa. Oleh itu, apabila mengakses nama kelas di dalam sistem atau tidak termasuk dalam ruang nama, anda mesti menggunakan nama yang layak sepenuhnya, contohnya:
1 Akses kelas global dalam ruang nama
<?php namespace A\B\C; class Exception extends \Exception {} $a = new Exception('hi'); // $a 是类 A\B\C\Exception 的一个对象 $b = new \Exception('hi'); // $b 是类 Exception 的一个对象 $c = new ArrayObject; // 致命错误, 找不到 A\B\C\ArrayObject 类 ?>
Untuk fungsi dan Untuk pemalar, jika fungsi atau pemalar tidak wujud dalam ruang nama semasa, PHP akan kembali menggunakan fungsi atau pemalar dalam ruang global.
2. Sandarkan fungsi/pemalar global dalam ruang nama
<?php namespace A\B\C; const E_ERROR = 45; function strlen($str) { return \strlen($str) - 1; } echo E_ERROR, "\n"; // 输出 "45" echo INI_ALL, "\n"; // 输出 "7" - 使用全局常量 INI_ALL echo strlen('hi'), "\n"; // 输出 "1" if (is_array('hi')) { // 输出 "is not array" echo "is array\n"; } else { echo "is not array\n"; } ?>
Ruang global
Jika tiada ruang nama ditakrifkan , semua kelas dan fungsi ditakrifkan dalam ruang global, sama seperti sebelum PHP memperkenalkan konsep ruang nama. Awalan nama menunjukkan bahawa nama itu berada dalam ruang global, walaupun nama itu berada dalam ruang nama lain.
Menggunakan arahan ruang global
<?php namespace A\B\C; /* 这个函数是 A\B\C\fopen */ function fopen() { /* ... */ $f = \fopen(...); // 调用全局的fopen函数 return $f; } ?>
Turutan ruang nama
Sejak pengenalan ruang nama, ruang nama yang paling terdedah kepada ralat Apabila tiba masanya untuk menggunakan kelas, sudah tiba masanya untuk mengetahui rupa laluan carian untuk kelas tersebut.
<?php namespace A; use B\D, C\E as F; // 函数调用 foo(); // 首先尝试调用定义在命名空间"A"中的函数foo() // 再尝试调用全局函数 "foo" \foo(); // 调用全局空间函数 "foo" my\foo(); // 调用定义在命名空间"A\my"中函数 "foo" F(); // 首先尝试调用定义在命名空间"A"中的函数 "F" // 再尝试调用全局函数 "F" // 类引用 new B(); // 创建命名空间 "A" 中定义的类 "B" 的一个对象 // 如果未找到,则尝试自动装载类 "A\B" new D(); // 使用导入规则,创建命名空间 "B" 中定义的类 "D" 的一个对象 // 如果未找到,则尝试自动装载类 "B\D" new F(); // 使用导入规则,创建命名空间 "C" 中定义的类 "E" 的一个对象 // 如果未找到,则尝试自动装载类 "C\E" new \B(); // 创建定义在全局空间中的类 "B" 的一个对象 // 如果未发现,则尝试自动装载类 "B" new \D(); // 创建定义在全局空间中的类 "D" 的一个对象 // 如果未发现,则尝试自动装载类 "D" new \F(); // 创建定义在全局空间中的类 "F" 的一个对象 // 如果未发现,则尝试自动装载类 "F" // 调用另一个命名空间中的静态方法或命名空间函数 B\foo(); // 调用命名空间 "A\B" 中函数 "foo" B::foo(); // 调用命名空间 "A" 中定义的类 "B" 的 "foo" 方法 // 如果未找到类 "A\B" ,则尝试自动装载类 "A\B" D::foo(); // 使用导入规则,调用命名空间 "B" 中定义的类 "D" 的 "foo" 方法 // 如果类 "B\D" 未找到,则尝试自动装载类 "B\D" \B\foo(); // 调用命名空间 "B" 中的函数 "foo" \B::foo(); // 调用全局空间中的类 "B" 的 "foo" 方法 // 如果类 "B" 未找到,则尝试自动装载类 "B" // 当前命名空间中的静态方法或函数 A\B::foo(); // 调用命名空间 "A\A" 中定义的类 "B" 的 "foo" 方法 // 如果类 "A\A\B" 未找到,则尝试自动装载类 "A\A\B" \A\B::foo(); // 调用命名空间 "A\B" 中定义的类 "B" 的 "foo" 方法 // 如果类 "A\B" 未找到,则尝试自动装载类 "A\B" ?>
Resolusi nama mengikut peraturan berikut:
1. Panggilan ke fungsi, kelas dan pemalar dengan nama yang layak sepenuhnya diselesaikan semasa penyusunan masa. Contohnya AB baru memutuskan ke kelas AB.
2. Semua nama yang tidak layak dan nama yang layak (nama tidak layak sepenuhnya) ditukar pada masa penyusunan mengikut peraturan import semasa. Sebagai contoh, jika ruang nama ABC diimport sebagai C, maka panggilan ke CDe() akan ditukar kepada ABCDe().
3. Dalam ruang nama, semua nama yang layak yang tidak ditukar mengikut peraturan import akan didahului dengan nama ruang nama semasa. Sebagai contoh, jika CDe() dipanggil dalam ruang nama AB, CDe() akan ditukar kepada ABCDe().
4. Nama kelas yang tidak layak ditukar pada masa penyusunan mengikut peraturan import semasa (nama penuh digunakan dan bukannya nama import pendek). Sebagai contoh, jika ruang nama ABC diimport sebagai C, maka C() baharu ditukar kepada ABC() baharu.
5. Dalam ruang nama (mis. AB), panggilan fungsi kepada nama yang tidak layak diselesaikan pada masa jalan. Contohnya, panggilan ke fungsi foo() dihuraikan seperti berikut:
a Cari fungsi bernama ABfoo() dalam ruang nama semasa
b () di angkasa.
6. Panggilan kepada nama yang tidak layak atau kelas nama yang layak (nama tidak layak sepenuhnya) di dalam ruang nama (mis. AB) diselesaikan pada masa jalan. Berikut ialah proses penghuraian untuk memanggil C() baharu dan DE() baharu: Penghuraian C() baharu: Penghuraian DE baharu(): Untuk merujuk kepada kelas global dalam ruang nama global, nama yang layak sepenuhnya baharu C() mesti digunakan.
a. Tambahkan nama ruang nama semasa di hadapan nama kelas untuk menjadi: ABDE, dan kemudian cari kelas.
b. Cuba muatkan kelas ABDE secara automatik.
c. Cari kelas ABC dalam ruang nama semasa.
d. Cuba muatkan kelas ABC secara automatik.