快速入门
对于任何应用程序来说,一个最常见和最具挑战的任务,就是从数据库中读取和持久化数据信息。在 Ecshopx 中我们采用 Doctrine ORM 替代 Laravel 自带的 Eloquent ORM ,前者更灵活,且自带 Repository 模式。本节将为大家介绍如何使用 Doctrine ORM。
几个概念
什么是 Doctrine ORM ?
Doctrine ORM 是 PHP 7.1+ 的对象关系映射器(ORM),可以将PHP对象持久化到数据库中。它以Data Mapper模式为核心,旨在将您的业务逻辑与数据库中的持久化完全分离。
什么是 Entities ?
实体是简单的PHP对象,其含可持久化的属性,可持久属性是实体的一个实例变量,通过Doctrine的数据映射功能可以将其保存到数据库中或从数据库中检索出来。实体类不需要扩展任何抽象基类或接口。
实体类不能为final,尽管它可以包含final方法。
什么是 DQL ?
DQL 是 Doctrine Query Language 的缩写,代表 Doctrine 查询语言,并且是对象查询语言(Object Query Language)的派生类,它与 Hibernate 查询语言(HQL)或Java持久化查询语言(JPQL)非常相似。
定义数据库 Schema
在 ECOS 中可以在 dbschema 文件夹下,使用数组定义数据库 Schema,laravel 中可以书写数据库迁移(database migration)文件,而在 ecshopx 中,则需要定义 Doctrine ORM 的实体类来完成数据库 Schema定义。
我们从最简单的实体 Product 开始。创建一个 src/DemoBundle/Entities/Product.php包含 Product 实体定义的文件:
创建实体类时,所有字段都应为 private。 这个类仅仅是一个普通的 php 类,还不能称之为 Entity 类。我们需要通过为这个类添加注解使之成为 Entity 类。
在添加完注解之后,可以通过以下命令,生成 getter 和 setter 方法:
生成之后,一个实体类就完整了,完整的实体类如下:
生成数据表
一个实体类定义了一张数据表,如何将数据表同步到数据库呢? 首先需要先运行:
运行以上命令成功后,会生成数据库迁移文件:
文件的路径在database/migrations/Version20191225161708.php 生成之后,即可运行一下命令,将数据表写入到数据库:
持久化对象到数据库
现在我们有了Product实体和与之映射的product数据库表。接下里我们演示如何把数据持久化到数据库里。 为了方便演示,我们采用测试用例的方式来演示此功能:
当 flush() 方法被调用时,Doctrine会遍历它管理的所有对象以确定是否需要被持久化到数据库。本例中, $product 对象的数据在库中并不存在,因此entity manager要执行 INSERT 请求,在 product 表中创建一个新行。
从数据库中获取对象
从数据库中取回对象就更简单了:
当你要查询某个特定类型的对象时,你总是要使用它的”respository”。你可以认为Respository是一个PHP类,它的唯一工作就是帮助你从那个特定的类中取出entity。对于一个entity类,要访问其宝库,通过:
一旦有了Repository对象,你就可以访问它的全部有用的方法了。
你也可以有效利用 findBy 和 findOneBy 方法,基于多个条件来轻松获取对象:
对象更新
一旦从Doctrine中获取了一个对象,更新它就很容易了:
更新一个对象包括三步:
1、从Doctrine中取出对象;
2、修改对象;
3、调用entity manager的 flush() 方法。
注意调用 $em->persist($product) 是不必要的。回想一下,这个方法只是告诉Doctrine去管理或者“观察” $product 对象。此处,因为你已经取到了 $product 对象了,它已经被管理了。
删除对象
删除一个对象十分类似,但需要从entity manager调用 remove() 方法:
你可能已经预期,remove() 方法通知Doctrine你想从数据库中删除指定的entity。真正的 DELETE 查询不会被真正执行,直到 flush() 方法被调用。
对象查询
你已经看到 repository 对象是如何让你执行一些基本查询而毋须做任何工作了:
当然,Doctrine 也允许你使用Doctrine Query Language(DQL)来写一些复杂的查询,DQL类似于SQL,只是它用于查询一个或者多个entity类的对象(如 product),而SQL则是查询一个数据表中的行(如 product )。
在Doctrine中查询时,你有两个主要选择:
编写纯正的Doctrine查询(DQL)
使用Doctrine的Query Builder
使用DQL进行对象查询
假设你要查询价格高于 19.99 的产品,并且按价格从低到高排列。你可以使用DQL,Doctrine中类似原生SQL的语法,来构造一个用于此场景的查询:
如果你习惯了写SQL,那么对于DQL也会非常自然。它们之间最大的不同就是你需要就“select PHP对象”来进行思考,而不是数据表的行。正因为如此,你要 从 Product 这个 entity 来select,然后给entity一个 p 的别名。
注意 `setParameter()`` 方法。当使用 Doctrine 时,通过“占位符”来设置任意的外部值(上面例子的 :price),是一个好办法,因为它可以防止SQL注入攻击。
getResult() 方法返回一个结果数组。要得到一个结果,可以使用getSingleResult()(这个方法在没有结果时会抛出一个异常)或者 getOneOrNullResult() :
DQL语法强大到令人难以置信,允许轻松地在entity之间进行join(稍后会覆盖relations)和group等。参考 Doctrine Query Language 文档以了解更多。
使用Doctrine's Query Builder进行对象查询
除了去写DQL,你还可以使用一个非常有用的QueryBuilder对象,来构建查询 SQL。当你的查询取决于动态条件时,这很有用,因为随着你的连接字符串不断增加,DQL代码会越来越难以阅读:
QueryBuilder对象包含了创建查询时的所有必要方法。通过调用getQuery()方法,query builder将返回一个标准的Query对象,可用于取得请求的结果集。
Query Builder更多信息,参考 Doctrine 的 Query Builder文档。
Last updated
Was this helpful?