[转]OOP的PHP长啥样

news/2024/7/5 17:47:24
// from:  http://bbs.chinaunix.net/forum/viewtopic.php?t=224412


在PHP的论坛中总是听到有人对PHP的OOP支持说三道四的,说这有缺陷,那里不足,但是都拿不出实际的例子。原来说过要和大家说说这事,但是一直很忙,现在算是抽了些时间了,所以把我刚刚做过的一个项目的框架拿出来和大家探讨一下。这个项目99%的代码是用oop方式编写的,感觉PHP对于OOP支持非常好,不是一般的好,是非常的好。有由于项目本身是一个商业项目所以源代码不好公布出来,但是基本框架还是可以说一说的,而且经过简化的例子更容易理解一些。如果你对PHP中的OOP还不太了解,还是就此打住吧,先去看看手册,或者基础读物再来看也不迟,反正这个是贴子没有长腿也跑不了。

长话短说,立刻开始吧。我这里会用到一个简单的例子,只有一个半的功能。一个是向浏览器发送一句"Hello, I can say OOP in PHP world!",另外半个功能是从数据库中进行一个查询然后输出到浏览器,说它是半个功能是因为只是作为一个例子讲讲没有实际的数据库操作。

首先从我的第一个文件index.php 开始介绍吧。我的index.php文件是这样的:
代码:

include_once ('config.php');
include_once ('class.Application.php');
$app = & new Application();
$app->run();
?>

这个就是全部了,虽然只有4行,但是如果用OOP的方式写这个应该就够了吧。
有一点经验的哥们会发现这里只用到了一个Application 对象,那么一定很想知道这个对象究竟长什么样呢?我们继续看看class.Application.php 这个文件的内部。从以上的代码中我们知道她应该至少包含两个方法
Application()

run()
所以大体上应该长成这样
代码:


class Application
{
   function Application()
   {

   }

   function run()
   {

   }
}

?>

现在就算知道Application 是什么样,它好像也没有办法完成我们预先设定的功能呀? 所以还要介绍一下如何运行这个程序,在我的结构中所有的页面都是通过index.php和一个action参数进行访问的例如第一个功能应该这样访问index.php?action=HelloPage,而第二个功能则是通过index.php?action=DatabasePage进行访问。这样的结构大家也许并不陌生吧。所以index.php 页面应当知道传进来的 action 参数是什么,也就是说Application对象应当知道 这个action 参数是什么。所以我们需要给Application增加一个方法 getAction()来获得action参数。既然知道action,知道了要做什么,那么方法 run()也就有知道如何去run了。

同时我还可以把(完成功能的)每一个页面作为一个对象来看待,所以我应该至少还需要两个类
class HelloPage 和
class DatabasePage
由于这两个对象最终都是向浏览器发送页面所以把他们共同的部分提出来作为他们的父类
class Page
以下是三个类文件的内容

class.Page.php
代码:


class Page
{
   function Page()
   {
   
   }

   function show()
   {
      //不能直接调用这个方法一定要在子类中去具体实现。
      die('You can not use this funciton directly from Page class');
   }
}
?>

其中这个show方法应该是所有页面对象都具有的方法,只是在实现上有所不同。

class.HelloPage.php
代码:

require_once ("class.Page.php");

class HelloPage extends Page
{
   function HelloPage()
   {
      parent::Page();
   }

   function show()
   {
      echo "Hello, I can say OOP in PHP world!";
   }
}
?>


class.DatabasePage.php
代码:

require_once ("class.Page.php");

class DatabasePage extends Page
{
   function DatabasePage()
   {
      parent::Page();
   }

   function show()
   {
      //做一些数据库操作然后将结果显示出来。
   }
}
?>


同时我们还遵守这样的一条规则:action的值和调用的页面类的名称保持一致,例如当action=HelloPage的时候程序就知道需要初始化一个HelloPage的对象,有了这样的规则和以上的几个文件我们就可以将 Application 类改进成这样。
代码:


class Application
{
   function Application()
   {

   }

   function getAction()
   {

   }

   function run()
   {
      $pageClass = $this->getAction();
      include_once ("class.".$pageClass.".php");
      $page = & new $pageClass();
      $page->show();
   }
}

?>

为什么getAction()空着?因为它太简单了,你自己可以轻松地把它写出来呀。

看到这里,如果你还不太明白,不用急,可以停下来重新再看一遍。

如果全明白了,我们就继续前进。我们还有半个任务没有完成,所以我们需要改进我们的Application和页面类,让它完成数据库操作功能。
进行数据库操作之前首先应当得到一个正确的数据库连接,如果让每个需要数据库连接的页面类去做这样的工作实在是一件非常费时费力的工作,不容易维护管理而且也破坏了oop的设计初衷,进行数据库操作的页面类例如 DatabasePage 只应当完成它份内工作即获得数据。 仔细看看我们的设计不难发现建立数据库连接的工作交给 Application 来做最合适不过了, 所以给Application 增加一个新的成员 $db 并且在初始化的时候将建立的数据库连接赋值给它。

代码:

require_once ("class.Database.php");

class Application
{
   var $db;//数据库对象

   function Application()
   {
      $this->db = & new Database(DB_HOST,DB_NAME,DB_LOGIN,DB_PASS);//$db 现在是一个数据库对象了
   }

   function getAction()
   {
      return $_GET['action']; //简单的实现 getAction;
   }

   funciton & getDatabase()
   {
      return $this->db;
   }

   function run()
   {
      $pageClass = $this->getAction();
      include_once ("class.".$pageClass.".php");
      $page = & new $pageClass($this); //这里是唯一做了手脚的地方,将这个Application对象传给页面对象。
      $page->show();
   }
}

?>

你现在不用太关心这个 Database对象从何而来如何实现,知道它是一个含有数据库连接的对象就可以了,如果用过phplib, ADODB,或者Pear库的就很容易理解。
这个语句:
$this->db = & new Database(DB_HOST,DB_NAME,DB_LOGIN,DB_PASS);
就是建立一个数据库连接而已。

至于DB_HOST,DB_NAME,DB_LOGIN,DB_PASS 这些都是常量我们在config.php中已经预先设定。

由于数据库操作页面 DatabasePage 需要进行数据库连接所以它也需要一个变量 $db 来保存数据库对象,所以我们需要把DatabasePage改进成这样:

class.DatabasePage.php
代码:

require_once ("class.Page.php");

class DatabasePage extends Page
{
   var $db;

   function DatabasePage(&$app)//将Application对象作为参数接受。
   {
      parent::Page();
      $this->db = $app->getDatabase();//获得 Application 中的数据库对象。
   }

   function show()
   {
      $sql = 'SELECT * FROM sale_orders';//简单的一个 SQL 例子。
      $results = $this->db->query($sql);//query 是 Database对象的一个公共的方法,通过它向数据库提交SQL查询。
      ...;//做一些操作把得到的结果显示出来。
   }
}
?>

好了,一个半的功能算是完成了,PHP对于OOP支持得也很漂亮吧,结构清晰,维护方便,至于效率嘛,我可没看出来有什么损失,如果你有兴趣可以自己测试一下。用这样的框架可以轻松应对各种需求的变化:增加各种权限控制,分离数据库层,商业逻辑,和表象层,增加远程调用接口统统不成问题,只是这里实在写不完这么多的东西。真不知道谁还会有理由说PHP 中OOP 很烂呢?

另外,需要提醒大家的是传递对象和赋值的时候要使用 & 符号这样可以保证每次引用的是同一个对象。



// 原帖讨论非常精彩,建议参考: http://bbs.chinaunix.net/forum/viewtopic.php?t=224412




http://www.niftyadmin.cn/n/3652074.html

相关文章

分布式文件系统GFS

内容概要一、GlusterFS简介1、GlusterFS的概念2、GlusterFS特点3、GlusterFS术语4、GlusterFS的模块化堆栈式架构5、GlusterFS的工作流程6、后端存储如何定位文件7、GFS支持的七种卷(1)分布式卷(默认)(2)条…

zookeeper集群介绍及部署

内容概要一、Zookeeper概述二、Zookeeper工作机制三、Zookeeper的特点四、Zookeeper的应用场景五、Zookeeper选举机制六、部署Zookeeper服务器1、安装JDK环境,默认已经安装的了2、解压安装包,安装Zookeeper3、修改配置文件4、使用远程传输来配置另外两台…

PHP中静态页处理动态内容的思路

最近碰到一个问题,新闻系统里所有新闻是生成静态的,但是我要在静态页里显示访问次数等动态内容,左思右想,没有好的解决方法。今天猛然一想,是不是能够通过框架页来解决呢?总结的基本思路如下:首…

消息队列,Kafka+Zookeeper群集

内容概要一、使用消息队列的原因二、使用消息队列的好处三、消息队列的两种模式四、Kafka简介五、Kafka的特点六、Kafka 系统架构七、部署 Zookeeper 集群1、安装kafka2、修改配置文件3、修改环境变量4、配置 Zookeeper 启动脚本5、设置开机自启6、Kafka 命令行操作一、使用消息…

[转] PHPlib Template 模板类中文文档

// from: http://www.4kiki.net/php_lib_template/PHPlib Template 模板类中文文档 第一版 2004-10-16iwind译 (2004-10-24更新)第十章. 模板(Template)目录模板类实例变量 模板类实例方法 可调用的实例方法 内部实例方法 模板类应用举例 备注: 如果你认为它像FastTemplates,那…

容器化工具--------Docker的介绍以及部署

内容概要一、Docker介绍二、使用容器的原因三、Docker和虚拟机的区别四、Docker的核心概念五、安装Docker1、安装依赖包2、设置阿里云镜像源3、安装 Docker-CE并设置为开机自动启动六、Docker的镜像操作1、搜索镜像2、获取镜像3、查看镜像信息4、根据镜像的唯一标识 ID 号&…

限制IP访问的思路和PHP代码

那天写自己的网站,想到要有关于限制IP的功能,至少要包括能够在后台设置要限制的IP,还有,能够对IP段进行限制,比如 192.168.0.* 一样的就能够限制整个段,左思右想,用了好几个if判断才解决&#x…

容器化工具--------Docker的网络操作和资源控制

内容概要一、Docker网络操作1、Docker网络实现的原理2、Docker的网络模式3、host网络模式4、container模式5、none模式6、Bridge模式7、自定义网络二、Docker资源控制1、CPU控制2、内存控制3、磁盘IO限制一、Docker网络操作 1、Docker网络实现的原理 Docker使用Linux桥接&…