2

我以前有过使用框架的经验,但从未真正深入了解过模型。我不知道这是否适用于所有框架或只是 zend,但目前我正在学习 Zend Framework。

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

3个类之间有什么关系?(模型、模型映射器、数据表)

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

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

请考虑这是一个非常简单的应用程序,它具有用户、他们的图片库和消息传递功能。

提前致谢。

4

3 回答 3

5

对于简单的应用程序并在 ZF 中涉足,只需使用将数据库表绑定到数据库适配器的 DbTable 模型即可开始。这不是最佳实践,但它很容易,可以帮助您入门。

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

zf create db-table Users users

db 这将创建一个名为Users.phpat的文件/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
    } 
} 

希望这可以帮助您入门,不要忘记阅读手册

于 2012-07-22T11:47:10.440 回答
2

一般来说,与许多其他框架相比,Zend Framework 不会强制您使用特定的模型架构。因此,您可以自由使用最适合您的应用程序的架构。因此,使用对象关系映射器只是众多方法中的一种。

但是要回答您的具体问题:使用映射器的原因仅仅是因为存储在关系数据库中的信息无法 1:1 转换为对象模型。这称为对象关系阻抗失配,并在相应的 Wikipedia 文章中进行了详细描述。

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

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

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

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

于 2012-07-22T09:55:11.433 回答
0

我想确认我的理解。

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

Application_Model_MessageMapper具有类似的函数getSender可能会返回一个实例,映射器Application_Model_User也使用 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 类与多对多关系所需的链接表一起使用,但我想如果我得到了基础,我最终会弄明白的。

请指出我误解的任何内容。

于 2012-07-22T22:02:20.347 回答