1.命名空间到底解决了什么问题?
PHP在5.3.0入了namespace关键字以及__NAMESPACE__魔术常量,命名空间的作用是为了解决重名问题。当然,这也是它唯一的作用了。
2.使用命名空间需要注意的点在哪?
你在当前文件下引入两个名字相同,命名空间不同的类,你可以通过use命名空间时,as出来两个别名的方式使类实例化的时候区分开使用。但你如果没有引入这两个类,仅仅use命名空间了,那么这两个类是不能实例化的。
这里你可以简单的测试一下,看看已有命名空间的情况下,引入类文件和没有引入能否found到类文件。
3.我们使用ide(phpstorm)的时候,直接敲出来这个类,然后ide会use需要的类命名空间,似乎就完成类文件的引入了?
重点:命名空间并没有做include/require文件的工作。
你要知道使用框架和没有使用框架是不一样的,框架会帮助你完成很多事。
相信你在第二步的时候。发现了php脚本情况下只引入命名空间是找不到类的这个事实。
so,回归php本身,甭管你有没有use这个类,当进程运行到需要使用到这个类的时候,php都会把它当做一个陌生的类。如果框架没有帮助引入类的设计,那么就会直接报错not found class了。
4.命名空间use在何时解析?
先上例子:
test.php
Print1();
namespace1.php
namespace my\name;
class A {
public function Print1(){
echo 11;
}
}
虽然require_once在use下面,也是可以正常运行的,因为程序只有在new mynameA()的时候才去加载命名空间myname,use相当于一种声明。
5.有命名空间的类,框架怎么实现引入的工作呢?
此时有两个函数可以帮到你,__autoload()和spl_autoload_register().
当你引用不存在的类时,__autoload()就会被调用,并且你的类名会被作为参数传送过去(当你同时使用命名空间,包含命名空间部分会一起作为参数传送)。
首先有一个类文件Aritst.class.php:
在你的index.php文件里有个全局函数,可以简单的理解成这个样子:
function __autoload($classname){
$classpath="./".$classname.'.php';
if(file_exists($classpath)){
require_once($classpath);
}
else{
echo 'class file'.$classpath.'not found!';
}
}
在index.php中去实例化Artist类
$obj = new Artist();
$obj->doThing();
运行index.php后正常输出hello aritst!。
在index.php中,由于没有包含Aritst.class.php,在实例化Aritst时,自动调用__autoload函数,参数$class的值即为类名Aritst,此时Aritst.class.php就被引进来了。
__autoload可以有很多写法和不同的用法,php框架在此基础上可以添加命名空间的映射,实现根据命名空间加载对应目录的工作。
再看spl_autoload_register(),这个函数与__autoload有与曲同工之妙,当看个简单的例子:
function autoRequire( $class ) {
$file = $class . '.class.php';
if (is_file($file)) {
require_once($file);
}
}
spl_autoload_register( 'autoRequire' );
$obj = new PRINTIT();
$obj->doPrint();
?>
它告诉PHP碰到没有定义的类就执行autoRequire()。
6.composer内部的命名空间又是怎么维护的?
兄弟,引入几个有现代php编码风格的开发包,然后打开你的composer.json文件,认真的阅读它,
"autoload": {
"psr-4": {
"App\\": "app/"
}
},
在这里,指明了命名空间和目录结构映射的关系,如果你想个性化定制命名空间,可以在这里添加上,比如:
"autoload": {
"psr-4": {
"App\\": "app/",
"Martistcn\\" : "app/martistcn"
}
},
运行composer命令:
composer dumpautoload
就会增加一条映射关系:
// vendor/composer/autoload_psr4.php
return array(
'App\\' => array($baseDir . '/app'),
/ ***** /
'Martistcn\\' => array($baseDir . '/app/martistcn'),
);
so,有址可循。