我以前有过使用框架的经验,但从未真正进入过模型。我不知道这是所有框架的情况还是只是zend,但目前我正在学习Zend Framework。
我一直在浏览示例应用程序并尝试阅读大量文章,但我无法找到一个简短而明确的答案,让我感到困惑。
3个班级之间有什么关系? (model,modelmapper,datatable)
假设我有一个名为users
的数据库表,它有3个字段userID, userName, userPassword
什么是示例代码?
是否有必要以这种方式构建我的应用程序的模型部分?如果我只是从数据库中检索数据并将结果作为数组返回,那么这是一个不好的做法吗?
请考虑这是一个非常简单的应用程序,它包含用户,图像库和消息传递功能。
提前致谢。
答案 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,并在相应的维基百科文章中详细描述。
对于您的示例应用程序,让我们考虑messages
和users
之间的关系。假设您有两个表,括号中给出了列:
users (id, username, email, password)
messages (recipient: users.id, sender: users.id, message)
哪个对应于“哑”对象User
和Message
。现在您要加载特定邮件。对象Message
提供方法getRecipient()
和getSender()
,它们都应返回User
的实例。但是,如果您读取数据库中的特定行,它将返回一个数组,而不是具有所需对其他对象的所有引用的对象。这正是映射器的用武之地:将数据库中的“平面”结果转换为对象,并确保对其他对象的所有必要引用都可用。
最后,另一个重要的功能是将持久性与普通模型分开。让我们假设有一天你想从经典的关系数据库引擎切换到XML存储。只要您的新持久层可以提供必要的对象,您就不必触及业务逻辑。
答案 2 :(得分:0)
我想确认一下我的理解。
Application_Model_User
和Application_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类来链接多个到多个关系所需的表,但我想如果我有了基础,我最终会弄明白。
请说明我误解的任何内容。