如何使用数据表和数据映射器?他们有必要吗?

时间:2012-07-21 19:22:32

标签: php zend-framework php-5.3 zend-db zend-db-table

我以前有过使用框架的经验,但从未真正进入过模型。我不知道这是所有框架的情况还是只是zend,但目前我正在学习Zend Framework。

我一直在浏览示例应用程序并尝试阅读大量文章,但我无法找到一个简短而明确的答案,让我感到困惑。

3个班级之间有什么关系? (model,modelmapper,datatable)

假设我有一个名为users的数据库表,它有3个字段userID, userName, userPassword什么是示例代码?

是否有必要以这种方式构建我的应用程序的模型部分?如果我只是从数据库中检索数据并将结果作为数组返回,那么这是一个不好的做法吗?

请考虑这是一个非常简单的应用程序,它包含用户,图像库和消息传递功能。

提前致谢。

3 个答案:

答案 0 :(得分:5)

对于简单的应用程序而言,只需使用将数据库表绑定到数据库适配器的DbTable模型,就可以开始使用ZF。这不是最好的做法,buit很容易,可以让你开始。

使用Zend_Tool cli命令将具有此格式
 zf create db-table name actual-table-name module force-overwrite
 这将转化为:

zf create db-table Users users

分贝 这将在Users.php创建一个名为/application/models/DbTable/的文件,它看起来像:

<?php

class Application_Model_DbTable_Users extends Zend_Db_Table_Abstract
{
    protected $_name = 'users'; //name of table, does not have to match name of class
}

现在在fetchAll的控制器中使用它就像这样简单:

<?php

class IndexController extends Zend_Controller_Action {

   public function indexAction(){
       $db = new Application_Model_DbTable_Users();//instantiate model
       $result = $db->fetchAll();//perform query, see Zend_Db_Table_Abstract for API
       $this->view->users = $result;//send to view
    } 
} 

只需创建一个小类,您就可以访问所选数据库适配器的功能。您还可以在DbTable模型中构建方法以自定义您的访问需求。

<?php

class Application_Model_DbTable_Users extends Zend_Db_Table_Abstract
{
    protected $_name = 'users'; //name of table, does not have to match name of class

    public function getUser($id) {
       $select = $this->select();//see Zend_Db_Select for info, provides more secure interface for building queries.
       $select->where('id = ?', $id);//can be chained

       $result = $this->fetchRow($select);//returns a row object, if array is needed use ->toArray()
       return $result;
    }
} 

此方法的使用方式与fetchAll()方法类似:

<?php

class IndexController extends Zend_Controller_Action {

   public function indexAction(){
       $id = $this->getRequest()->getParam('id');//assumes the id is set somehow and posted
       $db = new Application_Model_DbTable_Users();//instantiate model
       $result = $db->getUser($id);//perform query, see Zend_Db_Table_Abstract for API
       //$this->view->user = $result;//send to view as object
       $this->view->user = $result->toArray();//send to view as an array
    } 
} 

希望这能让你开始,不要忘记阅读manual

答案 1 :(得分:2)

一般而言,与许多其他框架相比,Zend Framework不会强迫您使用特定的模型体系结构。因此,您可以自由使用最适合您应用的架构。因此,使用object-relational mapper只是其中一种可能的方法。

但要回答您的具体问题:使用映射器的原因仅仅是因为存储在关系数据库中的信息无法以1:1的形式转换为对象模型。这被称为Object-relational impedance mismatch,并在相应的维基百科文章中详细描述。

对于您的示例应用程序,让我们考虑messagesusers之间的关系。假设您有两个表,括号中给出了列:

users (id, username, email, password)
messages (recipient: users.id, sender: users.id, message)

哪个对应于“哑”对象UserMessage。现在您要加载特定邮件。对象Message提供方法getRecipient()getSender(),它们都应返回User的实例。但是,如果您读取数据库中的特定行,它将返回一个数组,而不是具有所需对其他对象的所有引用的对象。这正是映射器的用武之地:将数据库中的“平面”结果转换为对象,并确保对其他对象的所有必要引用都可用。

最后,另一个重要的功能是将持久性与普通模型分开。让我们假设有一天你想从经典的关系数据库引擎切换到XML存储。只要您的新持久层可以提供必要的对象,您就不必触及业务逻辑。

答案 2 :(得分:0)

我想确认一下我的理解。

Application_Model_UserApplication_Model_Message只有变量及其setter和getter方法。基本上它们是存储模型。

Application_Model_MessageMapper具有getSender之类的函数,可以返回Application_Model_User的实例。Mappers也使用db-table类来检索数据。

现在我可以在我的控制器中执行类似的操作;

$id = $this->getRequest()->getParam('id'); // I received the message ID
$messageMapper = new Application_Model_MessageMapper(); 
$message = $messageMapper->getMessageByID($id); // this method uses Application_Model_DbTable_Message to fetch data, 
// then creates an object type of Application_Model_Message to set its fields and return.

$sender = $messageMapper->getSender($message);  // returns an instance of Application_Model_User after doing the necessary processes.

因此,Mappers与db-table类进行对话,从中获取数据,并将该数据设置为要返回的模型对象。

到目前为止我是否正确理解了这一点?如果是这样,我将开始研究该项目,并在我去的过程中了解更多信息。

我仍然不知道如何使用这些db-table类来链接多个到多个关系所需的表,但我想如果我有了基础,我最终会弄明白。

请说明我误解的任何内容。