当前位置:首 页 > 攻城湿 > 查看文章

过程式编程vs.面向对象编程

攻城湿 你是第2072个围观者 0条评论 供稿者: 标签:, , ,

过程式编程

维基百科是这样定义过程式编程的(Procedural Programming):过程式编程某种意义上等同于命令式编程(为了达到预定的状态而执行指定的步骤)的同义词,同时也是一种编程范例(正如本文中所述)——由结构化编程衍生而来,遵循过程调用的观念。

这是一个很恰当的定义,但我们还可以改进它。我更赞同“过程式编程只是一系列为了实现需求功能的特定步骤的命令”这一观点。它究竟是如何实现的只是细节,与范例无关,重要的是它是工作所必要的。先来看几个例子:

这个很明显是过程式编程:

$m = mysqli_connect(…);

$res = mysqli_query($m, $query);

$results = array();

while ($row = mysqli_fetch_assoc($res)) {

$results[] = $row;

}

虽然用到了对象,但它实际上也是过程式编程:

$m = new MySQLi(…);

$res = $m->query($query);

$results = array();

while ($row = $m->fetch_assoc($res)) {

$results[] = $row;

}

即使使用了类,它还是过程式编程:

class GetResults {

public function getResults() {

$m = new MySQLi(…);

$res = $m->query($query);

$results = array();

while ($row = $m->fetch_assoc($res)) {

$results[] = $row;

}

return $results;

}

}

注意:上述几个例子使用了完全相同的代码框,它们之间的不同点在于如何实现的,但都是过程式编程,都包含必须的独立步骤。再来看看什么才是面向对象编程,它们之间的不同之处又在哪?

面向对象编程

维基百科上是这样定义面向对象编程(Object-Oriented Programming)的:面向对象编程是使用对象的编程范式——包含数据域、方法以及它们之间的交互——来设计应用和程序。编程技术包括包括数据抽象、封装、通信、模块化、多态和继承。

这个定义也不错,但我只同意它的第二部分。第一部分所说的“必须使用对象来做面向对象编程”很明显是错误的,你完全可以通过数据抽象、封装、通信、模块化、多态和继承等方式实现数据抽象。

我对于面向对象编程的几点理解是:

  • 首先,它必须将数据抽象为模块结构;
  • 其次,必须存在某种方式来实现代码的多态执行;
  • 最后,它至少能部分压缩代码和函数。

下面再看看几个例子:

经典面向对象编程模式:

class Mediator {

protected $events = array();

public function attach($eventName, $callback) {

if (!isset($this->events[$eventName])) {

$this->events[$eventName] = array();

}

$this->events[$eventName][] = $callback;

}

public function trigger($eventName, $data = null) {

foreach ($this->events[$eventName] as $callback) {

$callback($eventName, $data);

}

}

}

$mediator = new Mediator;

$mediator->attach(‘load’, function() { echo “Loading”; });

$mediator->attach(‘stop’, function() { echo “Stopping”; });

$mediator->attach(‘stop’, function() { echo “Stopped”; });

$mediator->trigger(‘load’); // prints “Loading”

$mediator->trigger(‘stop’); // prints “StoppingStopped”

相同的模式,但使用函数:

$hooks = array();

function hook_register($eventName, $callback) {

if (!isset($GLOBALS[‘hooks’][$eventName])) {

$GLOBALS[‘hooks’][$eventName] = array();

}

$GLOBALS[‘hooks’][$eventName][] = $callback;

}

function hook_trigger($eventName, $data = null) {

foreach ($GLOBALS[‘hooks’][$eventName] as $callback) {

$callback($eventName, $data);

}

}

如你所见,它们都遵循传递者模式(Mediator Pattern),并且被设计为从sender中解耦caller,所以都是面向对象的。都提供状态、都是模块化的。不同点在于:第一个是通过传统的类实现的(因此可重用,这也是使用类的一个优势),而第二个使用了全局变量,并不可重用。我在这里使用“hook”,这是一个Drupal使用的事件系统。

Drupal是一个很好的例子,它的模块系统、“hook”系统、结构系统都是面向对象的,但都不是使用对象实现的,它是使用函数和动态分配,这导致了很多尴尬的折中,我并不是说这是一个好的面向对象,只是证明类并不是面向对象编程所必须的因素。

为什么这很重要?

很简单,因为很多开发者认为他们使用了类就是在做面向对线编程;另一些人认为他们使用函数就是在做过程式编程了,这并不正确。过程式编程和面向对象编程都是一种写代码的途径,而不是你写代码的手段。你会遵循步骤,按照设定好的方式去编写程序吗?你看起来是在函数式编程,但是如果你专注于状态改变和密封抽象,你就是在用面向对象编程。

类只是帮助简化面向对象编程的工具,并不是面向对象编程的要求或指示器。

面向对象编程与数据库存取

那面向对象编程里的数据库存取又是什么样的呢?面向对象编程的数据库存取是完全抽象的,我的方法是:

$mapper = new PersonDataMapper(new MySQLi(…));

$people = $mapper->getAll();

people是一个person对象的数组。注意:像这样抽象很有必要,所以事物对象无法直接对数据库操作,你需要一个映射器来翻译事物对象和数据存储之间的转换。一个专门的映射器会在内部创建请求,执行并返回结果。但这完全是抽象的,我们可以简单地换掉映射器来改变数据库层实现细节。

数据持久化的责任变成了封装抽象,这也就是为什么它是面向对象编程而不是过程式编程。

这家伙很懒,什么都没写!

—— zhaorong

zhaorong
你可能也喜欢Related Posts
众说纷纭Comments
大眼 可爱 大笑 坏笑 害羞 发怒 折磨 快哭了 大哭 白眼 晕 流汗 困 腼腆 惊讶 憨笑 色 得意 骷髅 囧 睡觉 眨眼 亲亲 疑问 闭嘴 难过 淡定 抗议 鄙视 猪头
小提示:直接粘贴图片到输入框试试
努力发送中...
  • 评论最多
  • 最新评论
  • 随机文章
footer logo
未经许可请勿自行使用、转载、修改、复制、发行、出售、发表或以其它方式利用本网站之内容
Copyright © zhaorong All Rights Reserved. 滇ICP备15006105号-1