仓酷云

 找回密码
 立即注册
搜索
热搜: 活动 交友 discuz
查看: 2220|回复: 20
打印 上一主题 下一主题

[学习教程] PHP教程之在数据库中利用对象的优点

[复制链接]
小魔女 该用户已被删除
跳转到指定楼层
楼主
发表于 2015-2-4 00:18:02 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有帐号?立即注册

x
就是管理员可以编辑,删除,回复 等功能,。加入管理员功能要加入登陆系统,慢慢你会想在线添加管理员,慢慢你会让自己的作品更漂亮些,慢慢1个完整的留言板就会出来了,对象|数据|数据库   咱们都晓得若何从Mysql获得咱们需求的行(纪录),读取数据,然后存取一些修改。很分明也很直接,在这个进程面前也没有甚么借题发挥的。但是关于咱们利用面临对象的法式设计(OOP)来办理咱们数据库中的数据时,这个进程就需求大大改善一下了。这篇文章将对若何设计一个面临对象的体例来办理数据库的纪录做一个复杂的描写。你的数据傍边的一切外部逻辑关系将被封装到一个十分层次的纪录对象,这个对象可以供应专门(专注)切实其实认代码体系,转化和数据处置。跟着Zend Engine2 和PHP5的宣布,PHP开辟者将会具有更壮大的面临对象的东西来帮助任务,这将使这个进程(面临对象地办理数据库)更有吸引力。


以以下出了一些利用对象来描叙你的数据库的有益方面:


存取办法(Accessor methods)将会使你对属性的读取和写入进程做到完整的掌握
每级的每一个纪录和属性(的操作)都有确认进程
从关系表中智能的获得对象
反复利用的逻辑办法意味着一切的数据交互都要经由过程不异的基本代码(codebase),这将使保护变得加倍复杂
代码复杂,由于分歧的纪录的外部逻辑都已包括在各自所处的类(class)傍边,而不是繁琐的库(lib)文件
在手工编写代码和SQL查询语句时,失足的时机将更少


存取办法(Accessor methods)

存取体例是经由过程类给实例(instance)的变量赋值。一个例子,我有一个叫User的类,而且有一个实例$username,我会写如许的存取办法(函数),User->username()和User->setUsername()用来前往和给实例赋值。



<?php
class User {
var $username;

function username() {
return $this->username;
}

function setUsername($newUsername) {
$this->username = $newUsername;
}
}
?>




这里有很好的来由让咱们编写如许的“出格的代码”。它将使开辟者更天真的改动类的繁琐的任务,由于这一进程将不需求其他的利用类的php代码。让咱们来看看上面这个加倍完美的可托赖的User类。


变量$username将不复存在,一切的器材都被整合的放在数组$_data傍边
假如username是空的话,username()函数将供应一个缺省(默许)的值给它
setUsername()进程将在承受值之前确认username是不是符合尺度格局(如字长等)

<?php
class User {
var $_data = array(); // associative array containing all the attributes for the User

function username() {
return !empty($this->_data['username']) ? $this->_data['username'] : '(no name!)';
}

function setUsername($newUsername) {
if ($this->validateUsername($newUsername)) {
$this->_data['username'] = $newUsername;
}
}

function validateUsername(&$someName) {
if (strlen($someName) > 12) {
throw new Exception('Your username is too long'); // PHP5 only
}
return true;
}
}
?>



不言而喻,这对咱们掌握存取对象的数据有很大匡助。假如一个法式员已直接地存取username的信息,以上代码的变更将会损坏他的代码。但是咱们可使用(类的)存取办法,就像下面代码中正文的那样,添加一个验证的功效而不需求改动任何其他的器材。注重username的验证(例子傍边是不克不及超越12字节)代码是自力在setUsername()办法以外的。从验证到存储到数据库的进程垂手可得。并且,这是个十分好的单凭经历的办法,一个办法或一个类需求做的越少,它的反复利用的时机将会越大。这在你入手下手写一个子类时加倍分明,假设你需求一个子类,而且又要跳过(疏忽)父类办法(行动)中的一些特别的细节,假如(针对这个细节的)办法很小而又精密,(修正它)只是一刹时的进程,而假如这个办法十分痴肥,针对多种目标,你能够将在复制子类中大批代码中愁闷而终。

例如说,假设Admin是User类的一个子类。咱们对adamin的用户能够会有分歧的,绝对刻薄一些的暗码验证办法。最好是跨过父类的验证办法和全部setUsername()办法(在子类中重写)。

更多关于存取器(Accessor)
上面是一些其他的例子来讲明若何使存取器用的更无效果。良多时分咱们能够要盘算了局,而不是复杂的前往数组中的静态数据。存取办法还能做的一个有效的工作就是更新(updating)缓存中的值。当一切的变化(对数据的一切操作)都要经由过程setX()办法的时分,这恰是咱们依据X来重置缓存中的值的时辰。

因而咱们的这个类条理变得加倍了然:

外部变量$_data的处置被交换成受回护的公有办法(private methods)_getData()和_setData()
这类办法被转移到被称作纪录(Record)的笼统的超等类(super class),固然它是User类下的子类
这个纪录类(Record class)把握一切存取数组$_data的细节,在内容被修正之前挪用验证的办法,和将变动的告诉发给纪录(Records),就像发给中间对象存储(ObjectStore)实例。

<?php
class User extends Record {

// --- OMITTED CODE --- //

/**
* Do not show the actual password for the user, only some asterixes with the same strlen as the password value.
*/
function password() {
$passLength = strlen($this->_getData('password'));
return str_repeat('*', $passLength);
}
/**
* Setting the user password is not affected.
*/
function setPassword($newPassword) {
$this->_setData('password', $newPassword);
}

/**
* fullName is a derived attribute from firstName and lastName
* and does not need to be stored as a variable.
* It is therefore read-only, and has no 'setFullname()' accessor method.
*/
function fullName() {
return $this->firstName() . " " . $this->lastName();
}

/**
* Spending limit returns the currency value of the user's spending limit.
* This value is stored as an INT in the database, eliminating the need
* for more expensive DECIMAL or DOUBLE column types.
*/
function spendingLimit() {
return $this->_getData('spendingLimit') / 100;
}

/**
* The set accessor multiplies the currency value by 100, so it can be stored in the database again
* as an INT value.
*/
function setSpendingLimit($newSpendLimit) {
$this->_setData('spendingLimit', $newSpendLimit * 100);
}

/**
* The validateSpendingLimit is not called in this class, but is called automatically by the _setData() method
* in the Record superclass, which in turn is called by the setSpendingLimit() method.
*/
function validateSpendingLimit(&$someLimit) {
if (is_numeric($someLimit) AND $someLimit >= 0) {
return true;
} else {
throw new Exception("Spending limit must be a non-negative integer"); //PHP5 only
}
}
}

/**
* Record is the superclass for all database objects.
*/
abstract class Record {
var $_data = array();
var $_modifiedKeys = array(); // keeps track of which fields have changed since record was created/fetched

/**
* Returns an element from the $_data associative array.
*/
function _getData($attributeName) {
return $this->_data[$attributeName];
}

/**
* If the supplied value passes validation, this
* sets the value in the $_data associative array.
*/
function _setData($attributeName, $value) {
if ($this->validateAttribute($attributeName, $value)) {
if ($value != $this->_data[$attributeName]) {
$this->_data[$attributeName] = $value;
$this->_modifiedKeys[] = $attributeName;
$this->didChange();
} else {
// the new value is identical to the current one
// no change necessary
}
}
}

/**
* For an attribute named "foo", this looks for a method named "validateFoo()"
* and calls it if it exists. Otherwise this returns true (meaning validation passed).
*/
function validateAttribute($attributeName, &$value) {
$methodName = 'validate' . $attributeName;
if (method_exists($this, $methodName)) {
return $this->$methodName($value);
} else {
return true;
}
}

function didChange() {
// notify the objectStore that this record changed
}
}
?>



如今咱们具有了一个笼统的超等类(Record),咱们可以将User类外面大批的代码转移出来,而让这个User的子类来存眷User的特别项目如存取和验证办法。你能够已注重到在咱们的这个记载类(Record class)没有任何的SQL代码。这并非忽略或漏掉!对象存储类(ObjectStore class)(埋没在第二局部)将担任一切和数据库的交互,还有咱们的超等类Record的实例化。如许使咱们的Record类加倍肥大而又无效率,而这关于评价咱们处置大批对象的效力的时分是个主要要素。

假如你有乐趣看看这篇文章基于的完全的代码(不会呈现如文中呈现的一切的语法毛病),可以给我的邮箱发信sam@360works.com

原文地址: http://www.phpbuilder.com/columns/barnum20030509.php3?page=1

  从刚开始练习的PHP基础语法练习,到PHP语言在WEB中的应用,再到实际的项目开发,如留言版,相册系统,中小型公司网站系统,以及期间做过的有关团队合作的小游戏,让我受益匪浅,学到了很多。
分手快乐 该用户已被删除
沙发
发表于 2015-2-4 10:56:15 | 只看该作者
建议加几个专业的phper的群,当然啦需要说话的人多,一处一点问题能有人回答你的,当然啦要让人回答你的问题,平时就得躲在里面聊天,大家混熟啦,愿意回答你问题的人自然就多啦。
金色的骷髅 该用户已被删除
板凳
发表于 2015-2-7 15:22:13 | 只看该作者
在我安装pear包的时候老是提示,缺少某某文件,才发现 那群extension 的排列是应该有一点的顺序,而我安装的版本的排序不是正常的排序。没办法我只好把那群冒号加了上去,只留下我需要使用的扩展。
admin 该用户已被删除
地板
发表于 2015-2-8 16:12:31 | 只看该作者
真正的方向了,如果将来要去开发团队,你一定要学好smarty ,phplib这样的模板引擎,
小女巫 该用户已被删除
5#
发表于 2015-2-25 20:37:42 | 只看该作者
学习php的目的往往是为了开发动态网站,phper就业的要求也涵盖了很多。我大致总结为:精通php和mysql
爱飞 该用户已被删除
6#
发表于 2015-2-27 05:47:46 | 只看该作者
学好程序语言,多些才是王道,写两个小时代码的作用绝对超过看一天书,这个我是深有体会(顺便还能练打字速度)。
柔情似水 该用户已被删除
7#
发表于 2015-3-7 09:59:24 | 只看该作者
首先我是坚决反对新手上来就用框架的,因为对底层的东西一点都不了解,造成知识上的真空,会对以后的发展不利。我的观点上手了解下框架就好,代码还是手写。当然啦如果是位别的编程语言的高手的话,这个就另当别论啦。
活着的死人 该用户已被删除
8#
发表于 2015-3-9 14:44:09 | 只看该作者
php里的数组为空的时候是不能拿来遍历的;(这个有点低级啊,不过我刚被这个边界问题墨迹了好长一会)
第二个灵魂 该用户已被删除
9#
发表于 2015-3-13 00:57:40 | 只看该作者
我学习了一段时间后,我发现效果并不好(估计是我自身的问题)。因为一个人的精力总是有限的,同时学习这么多,会导致每个的学习时间都得不到保证。
变相怪杰 该用户已被删除
10#
发表于 2015-3-17 05:48:52 | 只看该作者
再就是混迹于论坛啦,咱们的phpchina的论坛就很强大,提出的问题一般都是有达人去解答的,以前的帖子也要多看看也能学到不少前辈们的经验。别的不错的论坛例如php100,javaeye也是很不错的。
再见西城 该用户已被删除
11#
发表于 2015-3-23 21:56:04 | 只看该作者
其实也不算什么什么心得,在各位大侠算是小巫见大巫了吧,望大家不要见笑,若其中有错误的地方请各位大虾斧正。
飘灵儿 该用户已被删除
12#
发表于 2015-3-26 19:11:02 | 只看该作者
说php的话,首先得提一下数组,开始的时候我是最烦数组的,总是被弄的晕头转向,不过后来呢,我觉得数组里php里最强大的存储方法,所以建议新手们要学好数组。
老尸 该用户已被删除
13#
发表于 2015-4-5 15:11:02 | 只看该作者
爱上php,他也会爱上你。
精灵巫婆 该用户已被删除
14#
发表于 2015-4-12 21:51:11 | 只看该作者
学好程序语言,多些才是王道,写两个小时代码的作用绝对超过看一天书,这个我是深有体会(顺便还能练打字速度)。
若天明 该用户已被删除
15#
发表于 2015-4-24 15:09:22 | 只看该作者
Ps:以上纯属原创,如有雷同,纯属巧合
深爱那片海 该用户已被删除
16#
发表于 2015-4-26 20:57:19 | 只看该作者
微软最近出的新字体“微软雅黑”,虽然是挺漂亮的,不过firefox  支持的不是很好,所以能少用还是少用的好。
愤怒的大鸟 该用户已被删除
17#
发表于 2015-5-6 20:11:58 | 只看该作者
在学习的过程中不能怕麻烦,不能有懒惰的思想。学习php首先应该搭建一个lamp环境或者是wamp环境。这是学习php开发的根本。虽然网络上有很多集成的环境,安装很方便,使用起来也很稳定、
18#
发表于 2015-5-11 21:15:10 | 只看该作者
在我安装pear包的时候老是提示,缺少某某文件,才发现 那群extension 的排列是应该有一点的顺序,而我安装的版本的排序不是正常的排序。没办法我只好把那群冒号加了上去,只留下我需要使用的扩展。
若相依 该用户已被删除
19#
发表于 2015-6-6 11:43:55 | 只看该作者
基础有没有对学习php没有太大区别,关键是兴趣。
只想知道 该用户已被删除
20#
发表于 2015-6-7 11:03:14 | 只看该作者
php里的数组为空的时候是不能拿来遍历的;(这个有点低级啊,不过我刚被这个边界问题墨迹了好长一会)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|Archiver|手机版|仓酷云 鄂ICP备14007578号-2

GMT+8, 2025-1-1 22:06

Powered by Discuz! X3.2

© 2001-2013 Comsenz Inc.

快速回复 返回顶部 返回列表