扫二维码与项目经理沟通
我们在微信上24小时期待你的声音
解答本文疑问/技术咨询/运营咨询/技术建议/互联网交流
在php中,自动加载一般都是通过魔术方法__autoload来实现的,在程序中,当需要一个类的定义文件时(如实例化一个对象,集成自一个类),如找不到类定义文件时,就会自动触发__autoload方法,所以我们可以在该方法中编写代码完成类的加载。要编写代码实现类的自动加载,很重要的两点一是文件结构规划,二是类的命名,尤其是命名,从命名中要体现出该类的所属文件,比如PersonController,一看就知道是一个控制器,UserModel,一看便知是模型,这样就可以到对应的目录下去加载该类。针对你所列出的文件结构和文件命名,显然不符合上述规则,所以想通过__autoload实现自动加载有难度。
创新互联建站-专业网站定制、快速模板网站建设、高性价比寻乌网站开发、企业建站全套包干低至880元,成熟完善的模板库,直接使用。一站式寻乌网站制作公司更省心,省钱,快速模板网站建设找我们,业务覆盖寻乌地区。费用合理售后完善,10余年实体公司更值得信赖。
很多开发者写面向对象的应用程序时对每个类的定义建立一个 PHP 源文件。一个很大的烦恼是不得不在每个脚本开头写一个长长的包含文件列表(每个类一个文件)。
在 PHP 5 中,不再需要这样了。可以定义一个 __autoload() 函数,它会在试图使用尚未被定义的类时自动调用。通过调用此函数,脚本引擎在 PHP 出错失败前有了最后一个机会加载所需的类。
Tip
spl_autoload_register() 提供了一种更加灵活的方式来实现类的自动加载。因此,不再建议使用 __autoload() 函数,在以后的版本中它可能被弃用。
Note:
在 5.3.0 版之前,__autoload 函数抛出的异常不能被 catch 语句块捕获并会导致一个致命错误。从 5.3.0+ 之后,__autoload 函数抛出的异常可以被 catch 语句块捕获,但需要遵循一个条件。如果抛出的是一个自定义异常,那么必须存在相应的自定义异常类。__autoload 函数可以递归的自动加载自定义异常类。
Note:
自动加载不可用于 PHP 的 CLI 交互模式。
Note:
如果类名比如被用于 call_user_func(),则它可能包含一些危险的字符,比如 ../。 建议您在这样的函数中不要使用用户的输入,起码需要在__autoload() 时验证下输入。
Example #1 自动加载示例
本例尝试分别从 MyClass1.php 和 MyClass2.php 文件中加载 MyClass1 和 MyClass2 类。
?php
function __autoload($class_name) {
require_once $class_name . '.php';
}
$obj = new MyClass1();
$obj2 = new MyClass2();
?
Example #2 另一个例子
本例尝试加载接口 ITest。
?php
function __autoload($name) {
var_dump($name);
}
class Foo implements ITest {
}
/*
string(5) "ITest"
Fatal error: Interface 'ITest' not found in ...
*/
?
Example #3 自动加载在 PHP 5.3.0+ 中的异常处理
本例抛出一个异常并在 try/catch 语句块中演示。
?php
function __autoload($name) {
echo "Want to load $name.\n";
throw new Exception("Unable to load $name.");
}
try {
$obj = new NonLoadableClass();
} catch (Exception $e) {
echo $e-getMessage(), "\n";
}
?
以上例程会输出:
Want to load NonLoadableClass.
Unable to load NonLoadableClass.
Example #4 自动加载在 PHP 5.3.0+ 中的异常处理 - 没有自定义异常机制
本例将一个异常抛给不存在的自定义异常处理函数。
?php
function __autoload($name) {
echo "Want to load $name.\n";
throw new MissingException("Unable to load $name.");
}
try {
$obj = new NonLoadableClass();
} catch (Exception $e) {
echo $e-getMessage(), "\n";
}
?
以上例程会输出:
Want to load NonLoadableClass.
Want to load MissingException.
Fatal error: Class 'MissingException' not found in testMissingException.php on line 4
不仅仅thinkphp,MVC框架思路是一样的。主要在一个核心类完成。
tp 是在App.class.php里完成。
1,读取配置
配置文件里的配置
2,自动加载
核心类文件比如数据库类文件,模板文件等,模型类文件等
自动加载,模块,控制器等。
3,请求分发
实例化控制器。执行逻辑业务。
总之,这个文件完成把得到数据层数据,再分发到模板层。
说来简单,其实也有点小复杂。
php 中有个魔术方法__autoload ,这个函数在找不到类的时候就会调用,自动加载就是在这里实现的。通过指定自动加载类的路径,只要保证文件名和类名一样。就可以自动加载。这也是为什么你看很多源码中类的名字和文件名一样的原因,这样可以实现自动加载,不需要include.
这个和ajax技术密切相关.
ajax技术通俗来讲 就是保证页面不刷新的情况下.吧数据通过服务端展现给浏览器.
所以这里楼主可以使用ajax来实现.一下给出一个ajax例子
select id="select"
option/option
/select
//此时select框是没有任何数据的
给他加上value值改变触发的事件
$("#select").change(function(){
$.get('index.php',data,function(res){
$("#select").append('option value="'+res.val+'"'+res.content+'/option');//把获取到的信息append到select框下
},'json');
});
//或者楼主是要这种情况.选中select的其中某个选项在加载数据
select id="select"
option value="1"/option
/select
$("#select").change(function(){
//这里假设选中了value =1的option
var val = $("#select").val();
$.get('index.php',{value:val},function(res){
sonsole.log(res);//打印从服务器获取到的信息
},'json');
});
传统上,在PHP里,当我们要用到一个class文件的时候,我们都得在文档头部require或者include一下:
?php
require_once('../includes/functions.php');
require_once('../includes/database.php');
require_once('../includes/user.php');
...
但是一旦要调用的文档多了,就得每次都写一行,瞅着也不美观,有什么办法能让PHP文档自动加载呢?
?php
function
__autoload($class_name)
{
require
"./{$class_name}.php";
}
对,可以使用PHP的魔法函数__autoload(),上面的示例就是自动加载当前目录下的PHP文件。当然,实际当中,我们更可能会这么来使用:
?php
function
__autoload($class_name)
{
$name
=
strtolower($class_name);
$path
=
"../includes/{$name}.php";
if(file_exists($path)){
require_once($path);
}else{
die("the
file
{$class_name}
could
not
be
found");
}
}
也即是做了一定的文件名大小写处理,然后在require之前检查文件是否存在,不存在的话显示自定义的信息。
类似用法经常在私人项目,或者说是单一项目的框架中见到,为什么呢?因为你只能定义一个__autoload
function,在多人开发中,做不到不同的developer使用不同的自定义的autoloader,除非大家都提前说好了,都使用一个__autoload,涉及到改动了就进行版本同步,这很麻烦。
也主要是因为此,有个好消息,就是这个__autoload函数马上要在7.2版本的PHP中弃用了。
Warning
This
feature
has
been
DEPRECATED
as
of
PHP
7.2.0.
Relying
on
this
feature
is
highly
discouraged.
那么取而代之的是一个叫spl_autoload_register()的东东,它的好处是可以自定义多个autoloader.
//使用匿名函数来autoload
spl_autoload_register(function($class_name){
require_once('...');
});
//使用一个全局函数
function
Custom()
{
require_once('...');
}
spl_autoload_register('Custom');
//使用一个class当中的static方法
class
MyCustomAutoloader
{
static
public
function
myLoader($class_name)
{
require_once('...');
}
}
//传array进来,第一个是class名,第二个是方法名
spl_autoload_register(['MyCustomAutoloader','myLoader']);
//甚至也可以用在实例化的object上
class
MyCustomAutoloader
{
public
function
myLoader($class_name)
{
}
}
$object
=
new
MyCustomAutoloader;
spl_autoload_register([$object,'myLoader']);
值得一提的是,使用autoload,无论是__autoload(),还是spl_autoload_register(),相比于require或include,好处就是autoload机制是lazy
loading,也即是并不是你一运行就给你调用所有的那些文件,而是只有你用到了哪个,比如说new了哪个文件以后,才会通过autoload机制去加载相应文件。
当然,laravel包括各个package里也是经常用到spl_autoload_register,比如这里:
/**
*
Prepend
the
load
method
to
the
auto-loader
stack.
*
*
@return
void
*/
protected
function
prependToLoaderStack()
{
spl_autoload_register([$this,
'load'],
true,
true);
}
我们在微信上24小时期待你的声音
解答本文疑问/技术咨询/运营咨询/技术建议/互联网交流