PHP开发中ORM的应用

1.什么是ORM?

对象关系映射(Object Relational Mapping,简称ORM)模式是一种为了解决面向对象与关系数据库存在的互不匹配的现象的技术。简单的说,ORM是通过使用描述对象和数据库之间映射的元数据,将程序中的对象自动持久化到关系数据库中。

ORM并不是PHP独有的东西,只要和数据库打交道的语言都可以使用ORM,比如Java Web三大框架里面Hibernate,还有Doctrine(PHP重量级的ORM) ,Eloquent(laravel框架默认ORM,也可以单独使用)。

ORM是完全采用面向对象的方式去操作数据库,不用去拼SQL,对于复杂的SQL,ORM也支持直接运行原生SQL,咱先回顾一下平时咱们都是怎么操作数据库?举个例子,现在有一个库blog,一张表article,大部分的时候都是这是方式:新建MySQL连接,然后执行数据库操作,需要手写SQL:

1
2
3
4
5
6
7
8
9
10
11
12
13
<?php
$connect = mysqli_connect("localhost", "root", "123456", "blog", "3306") or die("数据库连接失败!");
$connect->set_charset("utf8");

$id = 1;
$sql = "SELECT * FROM article WHERE id = $id";
$query = mysqli_query($connect, $sql);
if (!$query) {
die("数据库查询失败!");
}

$assoc = $query->fetch_assoc();
var_dump($assoc);

上面的写法有一些缺点,有一种更好的方式是使用PDO,扩展性更强,而且可以使用预处理防止SQL注入:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
<?php
try {
$pdo = new PDO("mysql:host=localhost;dbname=blog", "root", "123456");
} catch (PDOException $exception) {
echo "Connect Failed" . $exception->getMessage();
}
$pdo->exec("set names utf8");

$id = 1;
$prepare = $pdo->prepare("SELECT * FROM article WHERE id = ?");
$prepare->execute(array($id));
while ($row = $prepare->fetch()) {
var_dump($row);
}

不过实际开发中,大家都是使用一些封装好的类和方法,比如laravel框架里面称之为查询构造器,我们可以使用这样方法去查询数据库:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
<?php
$users = DB::table('users')->get();

$price = DB::table('orders')->where('finalized', 1)->avg('price');

$users = DB::table('users')
->join('contacts', 'users.id', '=', 'contacts.user_id')
->join('orders', 'users.id', '=', 'orders.user_id')
->select('users.*', 'contacts.phone', 'orders.price')
->get();

$orders = DB::table('orders')
->select('department', DB::raw('SUM(price) as total_sales'))
->groupBy('department')
->havingRaw('SUM(price) > 2500')
->get();

还有比如说TP框架里面M方法,这些类和方法大大简化了查询操作,但本质上还是拼SQL,只不过调用的时候看起来更像面向对象,方便很多。

但是这些并不是真正意义上的ORM,最多只算得上是O(object),它只是把数据库查询操作对象化了,但是没有解决对象之间的关系问题!


2.Doctrine

doctrine是symfony框架默认ORM,下面我就简单介绍一下,官网连接: https://www.doctrine-project.org/

一.安装

按照官方的教程,最好的方式是使用composer:

1
2
3
4
5
6
{
"require": {
"doctrine/orm": "^2.6.2",
"symfony/yaml": "2.*"
}
}

二.在项目根目录创建一个bootstrap.php文件:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
<?php
// bootstrap.php
use Doctrine\ORM\Tools\Setup;
use Doctrine\ORM\EntityManager;

require_once "vendor/autoload.php";

// Create a simple "default" Doctrine ORM configuration for Annotations
$isDevMode = true;
$config = Setup::createAnnotationMetadataConfiguration(array(__DIR__."/src"), $isDevMode);
// or if you prefer yaml or XML
//$config = Setup::createXMLMetadataConfiguration(array(__DIR__."/config/xml"), $isDevMode);
//$config = Setup::createYAMLMetadataConfiguration(array(__DIR__."/config/yaml"), $isDevMode);

// database configuration parameters
$conn = array(
'dbname' => 'blog',
'user' => 'root',
'password' => '123456',
'host' => 'localhost',
'driver' => 'pdo_mysql',
'charset' => 'utf8',
);
// obtaining the entity manager
$entityManager = EntityManager::create($conn, $config);

这里面有一些需要注意的地方,$idDevMode是配置是否开发模式.

$config按照官方说法现在推荐使用 Annotation 也就说注解的方式配置,还支持xml和yaml,但是yaml这种方式已经被deprecated了,还有需要把src替换成你自己项目的目录,在本例中,是app。

下面还有数据库连接配置,官方给的案例是使用了sqlite,这里我改成了MySQL。

三.配置命令行工具

同样在项目根目录新建一个 cli-config.php 文件:

1
2
3
4
5
<?php
// cli-config.php
require_once "bootstrap.php";

return \Doctrine\ORM\Tools\Console\ConsoleRunner::createHelperSet($entityManager);

这样就可以使用命令行工具执行一些操作,比如说生成数据表,更新数据表

四.定义数据库实体,创建数据表

先来一个简单的,在app目录下创建一个 Product.php 文件,这个文件其实可以理解为是model,即数据库模型文件!内容如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<?php
namespace App;
/**
* @Entity @Table(name="products",options={"collate"="utf8mb4_unicode_ci", "charset"="utf8mb4"})
* Class Product
* @package App
*/
class Product
{
/**
* @ID @Column(type="integer") @GenerateDValue
* @var int
*/
protected $id;

/**
* @Column(type="string")
* @var string
*/
protected $name;

/**
* @return int
*/
......more code
}

后面的setter和getter这里省略了,如果有人对 annotation 这种注解方法比较熟悉的话应该可以看懂上面那些注释的意思。

首先在类的注释上,使用了@Entity表明这是一个数据库实体。@Table指定了表名,@ID表明的是主键,@Column表明是数据表字段,使用type声明了类型!

然后使用命令vendor/bin/doctrine orm:schema-tool:update --force --dump-sql就可以生成数据表:

1
2
3
4
5
6
7
8
9
The following SQL statements will be executed:

CREATE TABLE products (id INT NOT NULL, name VARCHAR(255) NOT NULL, PRIMARY KEY(id)) DEFAULT CHARACTER SET utf8 COLLATE utf8_unicode_ci ENGINE = InnoDB;

Updating database schema...

1 query was executed

[OK] Database schema updated successfully!

使用这种方式建表不用去写SQL语句,无论是mysql还是sql server,或者oracle,都没问题,一键迁移,ORM抹平了数据库之间的差异!

五.持久化数据到数据表

上面的步骤搞定了数据表创建的问题,下面来介绍一下如何插入数据到数据表,为了方便,这里我直接写在index.php里面:

1
2
3
4
5
6
7
8
9
10
11
12
<?php
require "vendor/autoload.php";
require "bootstrap.php";

$product = new \App\Product();
$product->setName("ORM的应用");

$entityManager->persist($product);
$entityManager->flush();

echo "Created Product Success with ID: ".$product->getId();
var_dump($product);

可以看出来这是一个完全OOP的写法,是先实例化一个数据表实体,然后通过setter去设置去属性,最后调用persist和flush持久化数据库里面。

六.查询数据

使用ORM查询数据也很简单,:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<?php
//查询所有
$productRepository = $entityManager->getRepository('\App\Product');
$products = $productRepository->findAll();

foreach ($products as $product) {
var_dump($product);
var_dump($product->getName());
}

//查询单个
$id = 3;
$product = $entityManager->find('Product', $id);
if ($product === null) {
echo "No product found.\n";
exit(1);
}
var_dump($product);

如果想对数据进行修改也很简单,比如在上面的例子里面,我们查询出id为3的数据,现在我们想修改这条数据:

1
2
3
<?php
$product->setName("ORM更新数据");
$entityManager->flush();

我们只需调用这个对象的setter方法,然后flush即可!

七.表与表之间的关系

数据表和数据表之间的关系总体来说可以分为下面几种:1对1,1对多,多对多,在doctrine里面有细分为下面几种:

划分的有点复杂和难理解,这里我就简单介绍其中一种:oneToMany,即1对多关系,这个其实很常见,比如说一个产品可以有多个评论。

从面向对象的思维来说,2个表之间的关系就是2个对象之间的关系,所谓1对多,其实1个对象包含(hasMany)多个其它对象, 在实际数据表设计,为了表达这种关系,也有好几种设计方式:

1
2
3
4
5
第一种: 在 product 表新增一个字段 comment_ids,用于存放所有评论ID,这种方式查询评论的时候简单,但是一旦要修改数据就头疼了,很少使用。

第二种: 在 comment 表新增一个product_id,用于表明当前评论所属的product,查询的时候稍微复杂点,但是便于修改数据。

第三种: 新建一个中间表,用来维护2个表之间的关系,中间表一般用来维护多对多的关系,但是也可以用于1对多的关系,这时候查询和修改都比较复杂,好处就是很容易扩展成多对多关系!

实际开发中,大部分时候都是使用第二种方式来表示1对多的关系。在doctrine里面,对于1对多,有3种形式:

1.双向(bidirectional),这个其实就是对应上面第二种的方式

2.单向结合中间表(Unidirectional with Join Table),这个就是对应上面所说的第三种的方式

3.自引用(Self-referencing),这个所谓的自引用,其实就是指类似在无限级分类表设计,有一个parent_id字段指向表本身的记录!

这里我就演示一下第二种方式,通过在 comment 表新建 product_id 字段这种方式。

首先,先定义一下评论实体comment, 基本结构和product差不多:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
<?php
namespace App;
/**
* @Entity @Table(name="comments",options={"collate"="utf8mb4_unicode_ci", "charset"="utf8mb4"})
* Class Product
* @package App
*/
class Comment
{
/**
* 这里通过注释设置了需要映射的实体和对应的字段
* @ManyToOne(targetEntity="Product", inversedBy="comments")
* @JoinColumn(name="product_id", referenceColumnName="id")
* @var Product
*/
protected $product;

/**
* @return Product
*/
public function getProduct(): Product
{
return $this->product;
}

/**
* @param Product $product
* @return self
*/
public function setProduct(Product $product): self
{
$this->product = $product;

return $this;
}

/**
* @ID @Column(type="integer") @GenerateDValue
* @var int
*/
protected $id;

/**
* @Column(type="string")
* @var string
*/
protected $content;

.......more code
}
```
但是多了一个属性 product, 因为这种1对多的关系对评论来说就是一个评论拥有一个产品,但是一个产品可以拥有多个评论。

同理,我们就需要对 product 实体做一些改动,加入了一个comments属性和一些注解!

```php
<?php
/**
* @oneToMany(targetEntity="Comment", mappedBy="product")
* @var
*/
protected $comments;

public function __construct()
{
$this->comments = new ArrayCollection();
}

....more code

执行 vendor/bin/doctrine orm:schema-tool:update --force --dump-sql更新数据库, 执行之后你会发现comments表会多一个product_id字段, 同时还会多出一个外键索引!

经过改造之后,现在如果我们需要给一个产品增加一条评论改怎么操作呢?举个例子,我们现在需要给id为3的这个产品增加一条评论,操作如下:

1
2
3
4
5
6
7
8
9
10
<?php
$id = 3;
$product = $entityManager->find('\App\Product', $id);

$comment = new \App\Comment();
$comment->setContent("这是一条评论!");
$comment->setProduct($product);

$entityManager->persist($comment);
$entityManager->flush();

执行以上代码,查看数据表你会发现comments表会自动增加一条记录,其product_id为3,在代码里面我们并没有手动去设置product_id,ORM替我们自动完成了这些操作!

下面再看查询一个产品的所有评论,操作也是相当简单的:
1
2
3
4
<?php
$id = 3;
$product = $entityManager->find('\App\Product', $id);
$comments = $product->getComments()->toArray();

3.总结

ORM试图把数据表之间的关系抽象成数据模型对象之间的关系,让开发人员少拼SQL,以更面向对象的方式开发,更加专注业务模型,有好处但是也有很多问题!

1.在doctrine这个ORM里面,表与表之间的关系是有物理外键索引的,这对于web项目来说并不是优势,虽然外键有利于保证数据安全和完整,但是大大影响了数据库插入速度!而现在大多数web项目都强调高并发,数据库往往都是瓶颈,
物理外键只适合一些对数据安全性和完整性要求非常高的项目,比如OA,企业SasS,或金融相关!但是在 Eloquent ORM 里面,数据表是没有物理外键的,表与表的之间关系是在模型层维护的。

2.比较占内存,ORM为了实现这些功能,在内存里面维护了很多对象,有很多额外处理逻辑,查询速度相比你直接运行原生sql慢一点

3.不利于优化SQL语句,虽然ORM最终也是生成sql去执行,但是其sql语句很多时候是不可控的,如果需要优化起来就很麻烦,无法针对每一条sql做优化!如果模型关系非常复杂的话就更难优化了!

以前参与了一个基于symfony框架的云采购系统的开发,symfony本身就是一个非常重量级的PHP框架,有人戏称用起来和某些Java框架一样,虽然其很多理念非常不错,但是PHP毕竟是脚本语言,并没有jvm的buff加持,导致项目做大了以后,速度非常慢(一个页面大概要1s多),也很占内存(最低2G内存),不过作为一个2b(面向企业)的系统来说,这点还可以接受,毕竟用户不多。

对于很多用户量巨大的web项目来说,数据库可能还有分库分表、读写分离等操作,使用ORM就可能会有很多问题要去解决了,直接拼SQL虽然难看了点,麻烦了点,但是可控性比较高!大家怎么看呢?