当PHP5的出现面向对象思想,我们在构造PHP程序时候就有了对程序的新的总结,把对象特性封装到类,特别是当PHP框架应用到实际项目中,构造类的对象和调用类对象出现了很大的讲究。
我们很多时候会发现,我们在写一些程序的时候,期望只要第一次构造了该对象,以后操作就是直接对该对象进行其他动作的操作,其实,每个类在构造一个对象后都是可以实现对象对方法或属性的调用,这是面向对象最为基本的做法。但是如果你研究了一些框架集合后,特别是应用到MVC模式下的框架开发程序,你会发现一个Controller里面会有很多个action,也就是我们所说的控制器里的方法有很多,当然你也会发现当一个控制器控制一个事务对象操作的时候,里面会出现很多次构造对象,操作数据集合的动作,但是你也会发现如果每次都是靠NEW方式去构造对象的话,无疑就是一次次给自己的内存空间增加负担。因为构造一个新的对象无疑就是要分配一个新的内存空间给它,但是如果你其实只是要求同一个对象动作的延续,那么试图寻找已经构造好的对象,并调用它无疑是个很不错的想法。所谓,同一对象的动作的延续,这个问题其实也很好解释,比如我现在构造一个控制器类为Controller A,里面是对数据Model A的一个添加,删除等操作,那么我们的控制动作就会存在假设为action Add ,actionDelete ,按照常规,我们通常利用框架集的单点入口原理,通过路由分配器程序找到了Controller A,但是里面会如果要单独调用actionAdd或,actionDelete ,且两个动作都是对Model A进行的数据操作,每次操作按照常规都必须构造数据对象,然后调用SQL去执行数据库操作,假设构造两个动作都要去操作同一个数据对象时候调用的方法时候都是在使用同一个语句$a=new A()这种形式的话,两次操作其实就产生了两个对象,也就同时占用两倍的内存空间,试想不过就是要求同一个人去实施两个动作,有必要两个人分别去做两件事情么,显然这种条件下是等于资源浪费,而且当动作多了的时候就会出现大量对象资源占用内存严重导致系统性能下降,最终严重的就导致崩溃。
其实,我们在写很多程序的时候也都一再强调,技术完成功能只是最基础的一部分,一个好的程序或是网站,要研究的不仅仅是功能问题,更重要的是整体性能,特别是对于大型讲究效用的程序来说性能是非常重要的。这点上框架集合是非常不错的典范,它们善于合理使用地址引用。让如果能一个对象处理的事情,就让一个对象来处理,把足够的空间留给其他不同的类别的对象。
框架集合解决对象资源是否重用的最好的方法就是全局变量的使用,当你的对象建立存在的时候,一般是第一次建立时候,就把它装载到全局变量中,我们也可以叫做注册对象,所以以后考虑对象调用方法的时候就直接去Check全局变量中是该类对象,也就是说看其是否注册了该对象,如果存在就直接返回该类对象,并告知地址,以便引用。这样以来是很容易实现对象地址引用的。当建立所有对象变量指向同一内存空间的时候,我们说此刻他们只是名字不同,实际是同一对象。
于是,我们可以建立了这样的程序代码如下:
<?php $GLOBALS['objects']['classname']=null; //是否注册类,类名变量 $GLOBALS['objects']['obj']=null; //对象变量 function & getSingle($classname){ if($GLOBALS['objects']['classname']==$classname){ return $GLOBALS['objects']['obj']; }else { $object= new $classname(); $GLOBALS['objects']['classname']=$classname; $GLOBALS['objects']['obj']=&$object; return $GLOBALS['objects']['obj']; } } class Test{ var $p1; function Test(){ $this->p1=1; } function add() { $this->p1++; } function show() { return $this->p1; } } $test1=&getSingle('Test'); $test1->add(); $test2=&getSingle('Test'); echo $test2->show(); $test2->add(); $test3=&getSingle('Test'); echo $test3->show(); if($test1===$test2) echo 'yes'; else echo 'no'; ?>
根据第一次建立对象后,第二次调用读取数据时候的必然是以原对象为基础的再操作,可以由后面的程序看到$test2去数据时候结果是2,然后进行add操作后,由$test3再读取时候发现结果为3显然是因为动作2完成的效果。由此,我们看到了利用同一对象进行分步操作的好处所在,在这里是因为我们将代码放在了一起,当然我们在使用框架的使用经常会出现这样的情况就是将上面的$test1,$test2,$test3以及他们的动作分别放在同一个控制器的不同动作action中,因为动作块是相互独立的,因此,三个对象变量谁先谁后触发最早建立这个对象内存空间都是随机而定的。第一次先是检测全局变量中是否注册了该对象,并检查对象地址是否存储在全局变量中,以后就是对该对象地址进行比对和存取。