你如何使用Zend Framework中的关系数据库?

时间:2011-05-30 21:45:04

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

我正在使用zend框架来编写应用程序,并且我在使关系数据库模型工作时遇到了很多问题。我已经多次阅读了快速入门和文档,但我仍然不确定如何做到这一点。我发布了不同表格之间关系的图片,以避免解释所有内容,因为英语不是我的第一语言,当我试图解释......复杂的事情时,我倾向于不清楚。 enter image description here

表press_releases,social_networking,blog_posts,rss_feed,directorios,users和articulos都在名为planilla_users的表中设置为外键。我用快速入门显示的方式对表格模型进行了编码:

class Application_Model_DbTable_PlanillaUsers extends Zend_Db_Table_Abstract
{

protected $_name = 'planilla_users';
protected $_referenceMap = array(
    'User' => array(
        'columns' => 'users_id',
        'refTableClass' => 'Application_Model_DbTable_Users',
        'refColumns' => 'id'
    ),
    'Articulo' => array(
        'columns' => 'art_id',
        'refTableClass' => 'Application_Model_DbTable_Articulos',
        'refColumns' => 'id'
    ),...etc

......其余的格式为:

class Application_Model_DbTable_Users extends Zend_Db_Table_Abstract
{

protected $_name = 'users';
protected $_dependentTables = array('Application_Model_DbTable_PlanillaUsers'); 

我还有一个模型Planilla.php,其中包含所有用于存储/更新/检索/删除信息的setter和getter,...但是......我对于如何处理映射器完全空白。我不知道它应该如何工作,老实说我还没有找到一个关于如何做这样的事情的好例子。所以任何帮助都会受到赞赏。

1 个答案:

答案 0 :(得分:3)

我给你的建议是看看Doctrine,它是一个对象关系映射器,可以很好地与ZF集成。我个人使用版本1.2和ZF 1.11.4没有任何问题。

有一个很好的屏幕演员here,它解释了如何将Doctrine插入ZF。学说可以是一个学习的猪,但一旦你了解它,从长远来看,它将节省你的时间。 Doctrine还附带了一个命令行脚本,可用于将数据库反向工程为Doctrine类。如果你在数据库上设置了关系,那么Doc​​trine也可以选择它。

我也听说ZF 2将包含Doctrine 2.0。

我用来创建Doctrine类的方法是首先在appication.ini文件中设置doctrine,在生成标题下的某处添加这些行。

[production]

//...

; Doctrine settings
pluginpaths.Freedom_Zend_Application_Resource = "Freedom/Zend/Application/Resource"
resources.doctrine.connection_string = "mysql://username:password@localhost/database_name"
resources.doctrine.compiled = false ; use compiled version of Doctrine
resources.doctrine.cache = false ; use query cache

; Information required for models generator
resources.doctrine.models_path = APPLICATION_PATH "/modules/default/models/Doctrine"
resources.doctrine.module_directories[] = APPLICATION_PATH "/modules/default/models/Doctrine/base"  
resources.doctrine.module_directories[] = APPLICATION_PATH "/modules/default/models/Doctrine"

; Generator settings
resources.doctrine.generate_models_options.phpDocPackage = Your App Name
resources.doctrine.generate_models_options.phpDocSubpackage = Doctrine
resources.doctrine.generate_models_options.phpDocName = Your Company Name
resources.doctrine.generate_models_options.phpDocEmail = your@email.address
resources.doctrine.generate_models_options.pearStyle = true
resources.doctrine.generate_models_options.generateTableClasses = true
resources.doctrine.generate_models_options.generateBaseClasses = true
resources.doctrine.generate_models_options.classPrefix = "Model_Doctrine_"
resources.doctrine.generate_models_options.baseClassPrefix = "Base_"
resources.doctrine.generate_models_options.baseClassesDirectory =
resources.doctrine.generate_models_options.classPrefixFiles = false
resources.doctrine.generate_models_options.generateAccessors = false
//...

您会注意到顶部有一条线 pluginpaths.Freedom_Zend_Application_Resource Freedom是我库中的通用命名空间(参见下面的文件夹树)。我在这里有一个Zend文件夹,我可以放置额外的ZF代码,这包括在需要时覆盖现有的ZF功能。自由是我的公司名称,你的明显不同。此行需要更改为您公司的名称,例如pluginpaths.Yourcompany_Zend_Application_Resource = "Yourcompany/Zend/Application/Resource"

下一行是您放置数据库连接设置resources.doctrine.connection_string = "mysql://username:password@localhost/database_name"

的位置

接下来的三行:

resources.doctrine.models_path = APPLICATION_PATH "/modules/default/models/Doctrine"
resources.doctrine.module_directories[] = APPLICATION_PATH "/modules/default/models/Doctrine/Base"  
resources.doctrine.module_directories[] = APPLICATION_PATH "/modules/default/models/Doctrine"

告诉Doctrine在哪里放置生成的类,因为我使用的是模块化设置,它们分别进入我的application/modules/default/models/Doctrineapplication/modules/default/models/Doctrine/Base(参见下面的文件夹树)。

还有一些其他行需要改变,这些应该是不言而喻的。

您还需要在application.ini中的开发标题下添加以下行。

[development : production]

//...

; phpSettings
resources.doctrine.compiled = false ; use compiled version of Doctrine
resources.doctrine.cache = false ; use query cache

//...

您还需要资源文件Doctrine.php。这个位置取决于您的设置,我的如下。

-Application
  -configs
  -layouts
  -modules
    -default // ZF default controller
      -controllers
      -models
        -Doctrine // folder for you generated classes
          -Base // Do not change the files in this folder
    -views
      -scripts
-Library
  -Doctrine // the doctrine application as downloaded
   doctrine-cli.php // you will need to create this (see below)
   Doctrine.php // come with the doctrine application
  -Freedom // my generic namespace, shared across multiple apps
    -Zend // place to overide/add ZF classes
      -Application
        -Resource
          *Docrine.php // the file below
  -Zend // ZF 1.11.4 (yours may differ)
  -ZendX // ZF extras

* Doctrine.php文件如下(显然忽略了* !!)

/**
* Doctrine application resource
*
* @author Juozas Kaziukenas (juozas@juokaz.com)
*/
class Freedom_Zend_Application_Resource_Doctrine extends Zend_Application_Resource_ResourceAbstract
{
    /**
* Initialize
*/
    public function init()
    {
        $doctrineConfig = $this->getOptions();

        if (isset($doctrineConfig['compiled']) && $doctrineConfig['compiled'] == true &&
            file_exists(APPLICATION_PATH . '/../library/Doctrine.compiled.php'))
        {
            require_once 'Doctrine.compiled.php';
        }
        else
        {
            require_once 'Doctrine.php';
        }
        $loader = Zend_Loader_Autoloader::getInstance();
        $loader->pushAutoloader(array('Doctrine_Core', 'autoload'), 'Doctrine');

        $manager = Doctrine_Manager::getInstance();

        // set models to be autoloaded and not included (Doctrine_Core::MODEL_LOADING_AGGRESSIVE)
        $manager->setAttribute(
            Doctrine_Core::ATTR_MODEL_LOADING,
            Doctrine_Core::MODEL_LOADING_CONSERVATIVE
        );

        // enable ModelTable classes to be loaded automatically
        $manager->setAttribute(
            Doctrine_Core::ATTR_AUTOLOAD_TABLE_CLASSES,
            true
        );

        // enable validation on save()
        $manager->setAttribute(
            Doctrine_Core::ATTR_VALIDATE,
            Doctrine_Core::VALIDATE_ALL
        );

        // enable accessor override
        $manager->setAttribute(
            Doctrine_Core::ATTR_AUTO_ACCESSOR_OVERRIDE,
            true
        );

        // enable sql callbacks to make SoftDelete and other behaviours work transparently
        $manager->setAttribute(
            Doctrine_Core::ATTR_USE_DQL_CALLBACKS,
            true
        );

        // enable automatic queries resource freeing
        $manager->setAttribute(
            Doctrine_Core::ATTR_AUTO_FREE_QUERY_OBJECTS,
            true
        );

        // connect to database
        $manager->openConnection($doctrineConfig['connection_string']);

        // set to utf8
        $manager->connection()->setCharset('utf8');

        if (isset($doctrineConfig['cache']) && $doctrineConfig['cache'] == true)
        {
            $cacheDriver = new Doctrine_Cache_Apc();

            $manager->setAttribute(
                Doctrine_Core::ATTR_QUERY_CACHE,
                $cacheDriver
            );
        }

        return $manager;
    }
}

接下来,您将需要创建doctrine-cli.php文件来引导您的应用程序,这应该位于库文件的根目录中(参见上面的树),我的如下。

/**
 * Doctrine CLI script
 *
 * @author Juozas Kaziukenas (juozas@juokaz.com)
 */

define('APPLICATION_ENV', 'development');
define('APPLICATION_PATH', realpath(dirname(__FILE__) . '/../../your/application'));

set_include_path(implode(PATH_SEPARATOR, array(
    realpath(APPLICATION_PATH . '/../library'),
    './',
    get_include_path(),
)));

require_once 'Zend/Application.php';

// Create application, bootstrap, and run
$application = new Zend_Application(
    APPLICATION_ENV,
    APPLICATION_PATH . '/configs/application.ini'
);

$application->getBootstrap()
        ->bootstrap('doctrine')
        ->bootstrap('autoload');

// set aggressive loading to make sure migrations are working
Doctrine_Manager::getInstance()->setAttribute(
    Doctrine::ATTR_MODEL_LOADING,
    Doctrine_Core::MODEL_LOADING_AGGRESSIVE
);

$options = $application->getBootstrap()->getOptions();

$cli = new Doctrine_Cli($options['resources']['doctrine']);

$cli->run($_SERVER['argv']);

您需要更改的唯一一行是

define('APPLICATION_PATH', realpath(dirname(__FILE__) . '/../../your/application'));

所以它指向应用程序的根文件夹,类似于公用文件夹中index.php文件中的行。

现在希望您应该准备好生成数据库类文件。转到终端 窗口和以下内容。

cd /home/path/to/library
php doctrine-cli.php generate-models-db

如果一切顺利,文件夹application/modules/default/models/Doctrineapplication/modules/default/models/Doctrine/Base应包含数据库的类。如上所述,不要更改Base文件夹中的文件,可以使用父Doctrine文件夹中的类进行更改。您还会注意到Doctrine文件夹中每个数据库表有两个类,一个以Table为后缀。在这里,我倾向于放置我的DQL代码,使其远离我的模型/控制器。其他类可用于更改基本文件夹中的类,也可以在此处添加挂钩和侦听器以添加特定于表的代码,例如添加密码加密或预设日期等。

我希望我已经清楚地解释了它,因为它是一个工作的猪,但这就是我的设置。

我希望这会有所帮助。