鍍金池/ 教程/ PHP/ 如何從已存在的數(shù)據(jù)庫中生成實體
如何以非繼承方式自定義方法
如何創(chuàng)建事件監(jiān)聽器
如何以非繼承方式擴展一個類
如何記錄消息到不同的文件
如何掌握并創(chuàng)建新的環(huán)境
如何使用 Doctrine DBAL
"XXX is deprecated" E-USER-DEPRECATED 的警告是什么意思?
在登錄表單中使用 CSRF 保護
如何注冊自定義 DQL 函數(shù)
如何為表單類配置空數(shù)據(jù)
如何嵌入集合表單
如何創(chuàng)建自定義認證提供者
如何使用 Apache Router
如何組織配置文件
部署在 Microsoft Azure 云
如何在路由參數(shù)中允許"/"字符
如何在安全,路由,服務(wù)和驗證中使用表達式
如何對顯示控制臺信息配置 Monolog
如何為一個 Bundle 創(chuàng)建友好的配置
如何改變默認的目標(biāo)路徑行為
如何在運行測試之前自定義引導(dǎo)過程
如何從路由向控制器傳輸額外的信息
如何從數(shù)據(jù)庫(實體提供者)讀取安全用戶
如何從 Controller 調(diào)用一個命令
如何創(chuàng)建自定義表單密碼驗證器
如何使用內(nèi)建的 PHP Web 服務(wù)器
如何在功能測試中用 Token 模擬認證
配置 Session 文件的保存目錄
理解前端控制器、內(nèi)核及環(huán)境如何協(xié)同工作
如何實現(xiàn)一個簡單的注冊表單
如何使用 Doctrine 擴展:Timestampable, Sluggable, Translatable 等等
如何使用多個實體管理器和連接
如何自定義表單渲染
如何安裝第三方 Bundles
使用預(yù)認證安全防火墻
如何簡化多個 Bundle 的配置
會話代理實例
安裝 Composer
如何冒充一個用戶
如何注冊一個新的請求格式和 MIME 類型
如何在功能測試中使用分析器
如何在服務(wù)容器內(nèi)設(shè)置外部參數(shù)
如何重寫 Symfony 默認的目錄結(jié)構(gòu)
如何在一個 Symfony 控制器中創(chuàng)建一個 SOAP 的 Web 服務(wù)
如何使用序列化
部署在 Platform.sh
升級一個副版本
如何寫一個自定義的 Twig 擴展
如何在 SubVersion 中創(chuàng)建并保存一個 Symfony 項目
使用 PHP 庫聯(lián)合,編譯和最小化 Web 資產(chǎn)
如何創(chuàng)建一個自定義的數(shù)據(jù)收集器
如何使用和注冊命名空間路徑
如何使用 Monolog 記錄日志
如何建立一個傳統(tǒng)的登錄表單
如何強制路由總是使用 HTTPS 或者 HTTP
如何在模板中使用 PHP 而不是 Twig
如何動態(tài)選擇密碼加密算法
避免匿名用戶開始 Session 會話
如何測試 Doctrine 倉庫
如何在功能測試中測試一封電子郵件被發(fā)送
Symfony2 與 Symfony1 的區(qū)別
使用結(jié)尾反斜線重定向 URL
(configuration)如何在數(shù)據(jù)庫中使用 PdoSessionHandler 存儲 Sessions
如何使用匹配器有條件地啟用分析器
部署在 Heroku 云
如何不用自定義控制器配置重定向
如何在 Bundle 內(nèi)部加載服務(wù)配置
如何處理不同的錯誤級別
如何在應(yīng)用中保護服務(wù)和方法
如何對表單單元測試
如何把命令定義為服務(wù)
如何配置 Monolog 從日志中排除 404 錯誤
如何使用控制臺
如何測試與數(shù)據(jù)庫交互的代碼
如何在路由中使用除了 GET 和 POST 的 HTTP 方法
如何使用云服務(wù)發(fā)送電子郵件
如何創(chuàng)建一個控制臺命令
在遺留的應(yīng)用上使用 Symfony Session
如何使用高級的訪問控制列表
如何不用一個自定義的控制器渲染一個模板
如何重寫部分 Bundle
升級一個主版本
安全訪問控制是如何工作的
如何使用 Bundle 的繼承來重寫部分 Bundle
如何使用 Voter 檢查用戶權(quán)限
如何為多個 Doctrine 的實現(xiàn)提供模型類
如何使用作用域
如何部署一個 Symfony 應(yīng)用
如何用 "inherit-data" 減少代碼冗余
如何注冊事件監(jiān)聽器和訂閱
使用 Bower 安裝 Symfony
如何創(chuàng)建一個自定義路由加載器
如何創(chuàng)建一個自定義的驗證限制
在獨立注入類中使用參數(shù)
如何使用 Assetic 和 Twig Functions 進行圖像優(yōu)化
如何利用表單事件動態(tài)修改表單
如何在過濾器的前后設(shè)置事件分發(fā)器
如何在 Bundle 中使用 Compiler Passes
緩存包含 CSRF 保護表單的頁面
如何注入變量到所有的模板(如全局變量)
如何創(chuàng)建一個自定義表單域類型
如何限定防火墻使其只允許通過指定請求
如何把 Controller 定義為服務(wù)
如何使用 Gmail 發(fā)送郵件
升級一個補丁版本
如何創(chuàng)建一個表單類型擴展
如何對不同的 URL 強制使用 HTTPS 或者 HTTP
如何使用 Varnish 加速我的 Web 站點
如何定義虛擬類和接口之間的關(guān)系
如何自定義登錄表單
如何測試多個客戶端的交互
PSR-7 Bridge
如何使用 YUI Compressor 裁剪 Javascripts 和 Stylesheets
在用戶的 Session 中使用局部 "Sticky"
如何定制錯誤頁
如何從已存在的數(shù)據(jù)庫中生成實體
如何用 Doctrine 上傳文件
可復(fù)用 Bundles 的最佳實踐
如何發(fā)送一封電子郵件
如何將 Assetic Filter 應(yīng)用到具體的文件擴展名上
切換分析器存儲
如何上傳文件
限制 Session 元數(shù)據(jù)的寫入
如何使用多用戶提供者
如何使用數(shù)據(jù)轉(zhuǎn)換
配置一個 Web 服務(wù)器
如何編程訪問分析器數(shù)據(jù)
如何在路由中使用服務(wù)容器參數(shù)
如何在控制臺生成 URL 和發(fā)送郵件
如何將你的開發(fā)環(huán)境優(yōu)化為調(diào)試環(huán)境
如何在控制臺命令中啟用日志
如何在功能測試中模擬 HTTP 認證
如何使用 API 驗證用戶
如何移除 AcmeDemoBundle
控制臺命令
如何配置 Symfony 使其工作在負載均衡和反轉(zhuǎn)代理
如何添加“記住我”登錄功能
如何使用 Assetic 進行資產(chǎn)管理
如何限定防火墻使其接受指定主機
如何使用訪問控制列表(ACLs)
如何裁剪 CSS/JS 文件(使用 UglifyJS 和 UglifyCSS)
如何緩存電子郵件
如何使用 submit() 函數(shù)處理表單提交
如何在 Git 中創(chuàng)建并保存一個 Symfony 項目
如何創(chuàng)建自定義用戶提供者
如何對電子郵件錯誤配置 Monolog
如何在開發(fā)時使用電子郵件

如何從已存在的數(shù)據(jù)庫中生成實體

當(dāng)開始使用一個工作于一個全新的項目的數(shù)據(jù)庫,自然而然就有兩個不同的結(jié)果。大部分情況下,數(shù)據(jù)庫模型的設(shè)計和建立從零開始。然而有些時候,您將是從一個已存在且不變的模型上開始。幸運的是,Doctrine 有一大堆的工具來幫助從您已存在的數(shù)據(jù)中生成模型類。

正如 Doctrine 工具文檔所說的,逆向工程是開始一個項目的一次性過程。Doctrine 能夠轉(zhuǎn)換 70 - 80% 基于領(lǐng)域、表單和外檢約束的必要映射信息。Doctrine 不能夠發(fā)現(xiàn)逆關(guān)聯(lián)、繼承類型、作為主鍵的外鍵實體或者語義操作關(guān)聯(lián)例如級聯(lián)或者生命周期事件。在生成實體之后還有一些必要的額外工作來設(shè)計每一個適合您的域模型特性。

本教程假設(shè)您正在使用一個有以下兩個表格的簡單的博客應(yīng)用程序:blog_postblog_comment。由于外鍵約束的原因,評論記錄與后續(xù)記錄相鏈接。

CREATE TABLE `blog_post` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `title` varchar(100) COLLATE utf8_unicode_ci NOT NULL,
  `content` longtext COLLATE utf8_unicode_ci NOT NULL,
  `created_at` datetime NOT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

CREATE TABLE `blog_comment` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `post_id` bigint(20) NOT NULL,
  `author` varchar(20) COLLATE utf8_unicode_ci NOT NULL,
  `content` longtext COLLATE utf8_unicode_ci NOT NULL,
  `created_at` datetime NOT NULL,
  PRIMARY KEY (`id`),
  KEY `blog_comment_post_id_idx` (`post_id`),
  CONSTRAINT `blog_post_id` FOREIGN KEY (`post_id`) REFERENCES `blog_post` (`id`) ON DELETE CASCADE
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

在投入到教程之前,確保您的數(shù)據(jù)庫連接參數(shù)在 app/config/parameters.yml 文件中正確設(shè)置(或者不管您的數(shù)據(jù)庫配置在哪里),并確保您已經(jīng)初始化了一個將要群集您實體類的包。在本教程中假設(shè)存在并位于 src/Acme/BlogBundle 文件夾。

從已存在數(shù)據(jù)庫創(chuàng)建實體類的第一步是讓 Doctrine 內(nèi)省數(shù)據(jù)庫并生成相應(yīng)的元數(shù)據(jù)文件。元數(shù)據(jù)文件描述了在表字段生成的實體類。

$ php app/console doctrine:mapping:import --force AcmeBlogBundle xml

這個命令行工具讓 Doctrine 來內(nèi)省數(shù)據(jù)庫并在包的 src/Acme/BlogBundle/Resources/config/doctrine 文件夾中生成 XML 元數(shù)據(jù)文件。這生成兩個文件:BlogPost.orm.xmlBlogComment.orm.xml。

通過把最后一個參數(shù)改為 yml 來生成 YAML 格式的元數(shù)據(jù)文件也是可能的。

生成的 BlogPost.orm.xml 元數(shù)據(jù)文件看起來如下:

<?xml version="1.0" encoding="utf-8"?>
<doctrine-mapping xmlns="http://doctrine-project.org/schemas/orm/doctrine-mapping" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://doctrine-project.org/schemas/orm/doctrine-mapping http://doctrine-project.org/schemas/orm/doctrine-mapping.xsd">
  <entity name="Acme\BlogBundle\Entity\BlogPost" table="blog_post">
    <id name="id" type="bigint" column="id">
      <generator strategy="IDENTITY"/>
    </id>
    <field name="title" type="string" column="title" length="100" nullable="false"/>
    <field name="content" type="text" column="content" nullable="false"/>
    <field name="createdAt" type="datetime" column="created_at" nullable="false"/>
  </entity>
</doctrine-mapping>

一旦元數(shù)據(jù)文件生成,您可以通過執(zhí)行以下兩個命令讓 Doctrine 創(chuàng)建相關(guān)的實體類。

$ php app/console doctrine:mapping:convert annotation ./src
$ php app/console doctrine:generate:entities AcmeBlogBundle

第一個命令生成注釋映射的實體類。但是如果您想使用 YAML 或者 XML 而不是注釋,您應(yīng)該只執(zhí)行第二個命令。

如果您想使用注釋,您可以安全地在運行了這兩個命令后刪除 XML(或 YAML)文件。

例如,新創(chuàng)建的 BlogComment 實體類看起來如下:

// src/Acme/BlogBundle/Entity/BlogComment.php
namespace Acme\BlogBundle\Entity;

use Doctrine\ORM\Mapping as ORM;

/**
 * Acme\BlogBundle\Entity\BlogComment
 *
 * @ORM\Table(name="blog_comment")
 * @ORM\Entity
 */
class BlogComment
{
    /**
     * @var integer $id
     *
     * @ORM\Column(name="id", type="bigint")
     * @ORM\Id
     * @ORM\GeneratedValue(strategy="IDENTITY")
     */
    private $id;

    /**
     * @var string $author
     *
     * @ORM\Column(name="author", type="string", length=100, nullable=false)
     */
    private $author;

    /**
     * @var text $content
     *
     * @ORM\Column(name="content", type="text", nullable=false)
     */
    private $content;

    /**
     * @var datetime $createdAt
     *
     * @ORM\Column(name="created_at", type="datetime", nullable=false)
     */
    private $createdAt;

    /**
     * @var BlogPost
     *
     * @ORM\ManyToOne(targetEntity="BlogPost")
     * @ORM\JoinColumn(name="post_id", referencedColumnName="id")
     */
    private $post;
}

正如您可以看到的,Doctrine 將所有的表字段轉(zhuǎn)化成為純私有和帶注釋的類屬性。最令人印象深刻的是它同樣發(fā)現(xiàn)了與基于外鍵約束的 BlogPost 實體類的關(guān)系。因此,您可以在 $post 實體類中找到用 BlogPost 實體映射的一個私有的 $post 屬性。

如果您想有一對多的關(guān)系,您需要手動將其添加到實體或者生成的 XML 或 YAML 文件。在具體的實體上添加一個區(qū)段使一對多定義 inversedBymappedBy 塊。

現(xiàn)在可以使用生成的實體了。玩得開心!