分享好友 编程语言首页 频道列表

doctrine缘来 之 造轮子

PHP教程  2016-12-23 12:410

doctrine缘来 之 造轮子

本系列是读 php data persistence with doctrine2 orm 的笔记,本文是第一篇:自己造轮子。

最开始描述下需要构建的系统

一个User可以发表Post,一个Post只有一个作者,User和Post之间彼此引用

一个User可以有多个Roles,User有Roles的引用,但是不能通过Role找到Users

一个User有一个UserInfo,UserInfo中包含了用户的注册信息等,User和UserInfo彼此引用

一个User有一个ContactData,包含email、电话等信息,User单向引用ContactData

一个User可能会有一个life partner,彼此之间互相引用

一个User会有多个friends,关系是单向的

一个Post会有多个标签Tag,Post到Tag是双向关系

一个Post有一个Category,Post到Category时单向关系

一个Category会有subcategories,并且会有parent Category

一个User会有多个Categories,User到Categories是单向关系

doctrine缘来 之 造轮子

在起初这个阶段我们不会直接就是用Doctrine,而是会自己来打造一个ORM,让我们更清楚的了解一个好的ORM需要怎么做。

读数据

先来看Model:User,部分代码如下:

classUser{
    const GENDER_MALE                 = 0;

    const GENDER_FEMALE               = 1;

    const GENDER_MALE_DISPLAY_VALUE   = "Mr.";

    const GENDER_FEMALE_DISPLAY_VALUE = "Mrs.";
	 /**
     * @return string
     */
    public functionassembleDisplayName()
    {
        $displayName = '';
        if ( $this->gender == self::GENDER_MALE ) {
            $displayName .= self::GENDER_MALE_DISPLAY_VALUE;
        } elseif ( $this->gender == self::GENDER_FEMALE ) {
            $displayName .= self::GENDER_FEMALE_DISPLAY_VALUE;
        }
        if ( $this->namePrefix ) {
            $displayName .= ' ' . $this->namePrefix;
        }
        $displayName .= ' ' . $this->firstName . ' ' . $this->lastName;
        return $displayName;
    }
}
classUserTestextendsPHPUnit_Framework_TestCase{

    public functiontestAssembleDisplayName()
    {
        $user = new User();
        $user->setFirstName( 'Max' );
        $user->setLastName( 'Mustermann' );
        $user->setGender( 0 );
        $user->setNamePrefix( 'Prof. Dr' );
        $this->assertEquals("Mr. Prof. Dr Max Mustermann",$user->assembleDisplayName());
    }
}

上面测试了User的一个功能,一般来说User都是从数据库中获取的,我们来写一段代码,测试下从数据库中读取的方式

public functiontestLoadFromDataBase()
    {
        $db       = new \PDO( 'mysql:host=127.0.0.1;dbname=app;port=33060', 'root', 'root' );
        $userData = $db->query( 'SELECT * FROM users WHERE id = 1' )->fetch();
        $user = new Entity\User();
        $user->setId( $userData['id'] );
        $user->setFirstName( $userData['first_name'] );
        $user->setLastName( $userData['last_name'] );
        $user->setGender( $userData['gender'] );
        $user->setNamePrefix( $userData['name_prefix'] );
        $this->assertEquals( "Mr. Prof. Dr. Max Mustermann", $user->assembleDisplayName() );
    }

上面代码就是一个简易的ORM,从数据库中加载数据,然后将其转换为Object,让我们更进一步,将这些“data mapping”功能单独抽取出来,叫做Mapper:

<?php
namespace Mapper;

classUser{

    private $mapping = [
        'id'         => 'id',
        'firstName'  => 'first_name',
        'lastName'   => 'last_name',
        'gender'     => 'gender',
        'namePrefix' => 'name_prefix',
    ];
    public functionpopulate( $data, $user )
    {
        $mappingsFlipped = array_flip( $this->mapping );
        foreach ( $data as $key => $value ) {
            if ( isset( $mappingsFlipped[ $key ] ) ) {
                call_user_func_array(
                    [ $user, 'set' . ucfirst( $mappingsFlipped[ $key ] ) ],
                    [ $value ]
                );
            }
        }
        return $user;
    }
}

此处我们再来看测试代码:

public functiontestPopulate()
    {

        $db       = new \PDO( 'mysql:host=127.0.0.1;dbname=app;port=33060', 'root', 'root' );
        $userData = $db->query( 'SELECT * FROM users WHERE id = 1' )->fetch();
        $user       = new Entity\User();
        $userMapper = new Mapper\User();
        $user       = $userMapper->populate( $userData, $user );
        $this->assertEquals( "Mr. Prof. Dr. Max Mustermann", $user->assembleDisplayName() );
    }

上面代码已经将数据映射的功能进行了封装,下一步,我们将sql语句抽离出来,封装到Repository中:

<?php namespace Repository;

use Mapper\User as UserMapper;
use Entity\User as UserEntity;

classUser{

    /**@var\EntityManager */
    private $em;
    private $mapper;

    public function__construct( $em )
    {
        $this->mapper = new UserMapper;
        $this->em     = $em;
    }

    public functionfindOneById( $id )
    {
        $userData = $this->em
            ->query( 'SELECT * FROM users WHERE id = ' . $id )
            ->fetch();
        return $this->mapper->populate( $userData, new UserEntity() );
    }
}

此处有个类叫 EntityManager ,其职责是作为数据库操作的Entry Point,负责所有的具体的数据库操作:

<?php
use Repository\User as UserRepository;
use Repository\Post as PostRepository;
use Mapper\User as UserMapper;

classEntityManager{

    private $host;
    private $db;
    private $user;
    private $pwd;
    private $port;
    private $connection;
    private $userRepository;
    private $postRepository;
    private $identityMap;

    public function__construct( $host, $db, $port, $user, $pwd )
    {
        $this->host           = $host;
        $this->user           = $user;
        $this->pwd            = $pwd;
        $this->connection     = new \PDO( "mysql:host=$host;port=$port;dbname=$db", $user, $pwd );
        $this->userRepository = null;
        $this->postRepository = null;
        $this->db             = $db;
        $this->identityMap    = [ 'users' => [] ];
        $this->port = $port;
    }

    public functionquery( $stmt )
    {
        return $this->connection->query( $stmt );
    }

    public functiongetUserRepository()
    {
        if ( !is_null( $this->userRepository ) ) {
            return $this->userRepository;
        } else {
            $this->userRepository = new UserRepository( $this );
            return $this->userRepository;
        }
    }
}

此时我们的测试代码变为了:

<?php

classUserRepositoryTestextends\PHPUnit_Framework_TestCase{

    public functiontestPopulate()
    {
        $em = new \EntityManager('127.0.0.1','app',33060,'root','root');
        $repository = new Repository\User($em);
        $user = $repository->findOneById(1);
        $this->assertEquals( "Mr. Prof. Dr. Max Mustermann", $user->assembleDisplayName() );
    }
}

到目前为止我们做的事情就是将数据从数据库中读取出来,然后根据数据构造出对象,下面我们再进一步,看怎么对对象进行持久化。

保存数据

保存操作有两种:insert、update,先来看准备动作,将数据从对象Entity中取出来:

// class Mapper\User 
public functionextract( $user )
    {
        $data = [];
        foreach ( $this->mapping as $keyObject => $keyColumn ) {
            if ( $keyColumn != $this->getIdColumn() ) {
                $data[ $keyColumn ] = call_user_func(
                    [ $user, 'get' . ucfirst( $keyObject ) ]
                );
            }
        }
        return $data;
    }

在EntityManager中新增saveUser方法:

public functionsaveUser( $user )
{
    $userMapper = new UserMapper();
    $data       = $userMapper->extract( $user );
    $userId     = call_user_func(
        [ $user, 'get' . ucfirst( $userMapper->getIdColumn() ) ]
    );
    if ( array_key_exists( $userId, $this->identityMap['users'] ) ) {
        $setString = '';
        foreach ( $data as $key => $value ) {
            $setString .= $key . "='$value',";
        }
        return $this->query(
            "UPDATE users SET " . substr( $setString, 0, -1 ) .
            " WHERE " . $userMapper->getIdColumn() . "=" . $userId
        );
    } else {
        $columnsString = implode( ", ", array_keys( $data ) );
        $valuesString  = implode( "', '", $data );
        return $this->query(
            "INSERT INTO users ($columnsString) VALUES('$valuesString')"
        );
    }
}

此时新增一个User的方法如下:

<?php

class EntityManagerTest extends PHPUnit_Framework_TestCase {

    public function testSaveUser()
    {
        $em      = new \EntityManager( '127.0.0.1', 'app', 33060, 'root', 'root' );
        $newUser = new Entity\User();
        $newUser->setFirstName( 'Ute' );
        $newUser->setLastName( 'Musermann' );
        $newUser->setGender( 1 );
        $em->saveUser( $newUser );
        $this->assertEquals("Mrs. Ute Musermann",$newUser->assembleDisplayName());
    }
}

此处在saveUser中使用了identity map模式,通过记录已经load的entity,减少从数据库中重新加载数据。

关系

用户有多个Posts,通过User的getPosts方法可以获取posts,因此有下面的代码:

// class Entity\User
public function getPosts()
{
    if ( is_null( $this->posts ) ) {
        $this->posts = $this->postRepository->findByUser( $this );
    }
    return $this->posts;
}

此时为了能够获取posts,需要初始化postRepository,最好的初始化地方就是Repository\User中的findOneById,看代码:

public function findOneById( $id )
{
    $userData = $this->em->query('SELECT * FROM users WHERE id = ' . $id)->fetch();
    $newUser = new UserEntity();
    $newUser->setPostRepository($this->em->getPostRepository());
    return $this->em->registerUserEntity(
        $id,
        $this->mapper->populate($userData, $newUser)
    );
}

最后要配套的Post的Entity,Mapper,Repository,然后是findByUser方法的实现

// class Repository\Post
public function findByUser( UserEntity $user )
{
    $postsData = $this->em
        ->query( 'SELECT * FROM posts WHERE user_id = ' . $user->getId() )->fetchAll();
    $posts     = [];
    foreach ( $postsData as $postData ) {
        $newPost = new PostEntity();
        $posts[] = $this->mapper->populate( $postData, $newPost );
    }
    return $posts;
}

此时让我们回过头来看下项目结构:

src
├── Entity
│   ├── Post.php
│   └── User.php
├── EntityManager.php
├── Mapper
│   ├── Post.php
│   └── User.php
└── Repository
    ├── Post.php
    └── User.php

此时我们已经具备了基本的orm框架了,再往下就会越来越复杂了,下一篇让我们来看下doctrine的实现的。

本文完整的代码可以查看 https://github.com/zhuanxuhit/doctrine-learn

查看更多关于【PHP教程】的文章

展开全文
相关推荐
反对 0
举报 0
评论 0
图文资讯
热门推荐
优选好物
更多热点专题
更多推荐文章
【Rust】标准库-Result rust数据库
环境Rust 1.56.1VSCode 1.61.2概念参考:https://doc.rust-lang.org/stable/rust-by-example/std/result.html示例main.rsmod checked {#[derive(Debug)]pub enum MathError {DivisionByZero,NonPositiveLogarithm,NegativeSquareRoot,}pub type MathResult =

0评论2023-02-09978

【Rust】标准库-引用 rust 数据库框架
环境Rust 1.56.1VSCode 1.61.2概念参考:https://doc.rust-lang.org/stable/rust-by-example/std/rc.html示例rust 使用 Rc 来实现引用计数。main.rsuse std::rc::Rc;fn main() {let rc_examples = "Rc examples".to_string();{println!("--- rc_a is created

0评论2023-02-09638

Perl操作Mysql数据库 perl操作excel
一. 安装DBI模块步骤1:从TOOLS栏目中下载DBI.zip,下载完后用winzip解开到一个temp目录,共有三个文件:ReadmeDBI.ppdDBI.tar.gz步骤2: 在DOS窗口下,temp目录中运行下面的DOS命令:ppm install DBI.ppd 如果提示无效命令,可在perl/bin目录下运行 二. 安装DBD

0评论2023-02-09348

在OS X系统中配置Ruby on Rails使其可以访问Sql Server数据库
经过大半天的折腾,终于可以让RoR在OS X系统里访问Sql Server数据库了。这里记录一下操作的过程,免得以后忘了。第一步,安装FreeTDS从FreeTDS的官网上下载最新的稳定版的压缩包,然后,遵照这里的说明进行手工编译(好怀念微软的Setup.exe和*.msi啊),其中

0评论2023-02-09433

小程序-列表页跳详情页(不在数据库)
1.缓存localstorage,可以长期保存数据2.绑定到view层id='',只要显示历史记录,就能携带id到详情页e.currentTarget.id访问点击当前的view的id

0评论2023-02-09710

Mysql数据库一个小程序实现自动创建分表。
每当跨月的时候也是系统出问题最多的时候,没有表和字段缺失是两个最常见的错误。为了解决这个问题,研究了一下mysql的 information_schema 表:information_schema这张数据表保存了MySQL服务器所有数据库的信息。如数据库名,数据库的表,表栏的数据类型与访

0评论2023-02-09699

解决小程序云函数操作数据库回调不执行
背景最近写个微信小程序,在云函数中操作数据库时,明明操作成功了,理应回调success,却没有;而在小程序端,一样的代码,却能成功回调。 问题原因参见官方文档:https://developers.weixin.qq.com/miniprogram/dev/wxcloud/reference-server-api/init.html

0评论2023-02-09598

Delphi XE2 之 FireMonkey 入门(31) - 数据绑定: 绑定数据库
Delphi XE2 之 FireMonkey 入门(31) - 数据绑定: 绑定数据库一、全设计时操作:先在窗体上放置控件:DataSource1    : TDataSource;ClientDataSet1 : TClientDataSet;Label1       : TLabel;Edit1          : TEdit;Memo1          : TMemo;Ima

0评论2023-02-09516

Delphi XE中使用dbExpress连接MySQL数据库疑难问题解决(对三层的例子配置有帮助)
Delphi IDE中包含一个Data Explorer的组件,如下图所示:  该组件基于dbExpress(包含TSQLConnection、TSQLDataSet、TSQLQuery、TSQLStoredProc、TSQLTable、TsqlServerMethod、TSQLMonitor、TSimpleDataSet)。但是因为该组件只提供了各种数据库的抽象驱动

0评论2023-02-09410

Delphi7数据库编程之TDataSet(转)
TDataSet类由TBDEDataSet(BDE组件)、TCustomADODataSet(ADO组件)、TIBCustomDataSet(InterBase组件)、TCustomSQLDataSet(dbExpress组件)和TCustomClientDataSet子类组成。下面介绍DataSet类中比较重要的属性(可能会有取舍,等到真正做项目用到的时候

0评论2023-02-09554

悟透delphi 第十一章 面向对象数据库基础
本书原著李战(leadzen)大牛,由tingsking18整理,本人blog发布的版本经过战哥同意,转载请著名出处和原作者!第十一章面向对象数据库基础第二节 数据对象的标识我们在关系数据库的设计和开发中,可能经常需要一些唯一的编号或标识,用来作为关键字,以区别每

0评论2023-02-09380

数据库服务器 之 在Linux下使用perl通过unixODBC连接SQLServer2000
作者:tonyvicky来自:LinuxSir.Org摘要:MS从来没有提供过SQLServer for Linux,所以大家也不要去尝试在Linux系统安装SQLServer,但是可以通过ODBC连接Windows系统的SQLServer数据库;目录一、关于测试环境及Linux连接SQL Server 的说明;二、下载相关软件 uni

0评论2023-02-09326

DBI 数据库模块剖析:Perl DBI 数据库通讯模块规范,工作原理和实例
原文:http://tech.ddvip.com/2010-06/1275904179154934.html     本文详细地介绍了 Perl 语言中用于和数据库通讯的 DBI 模块。以细腻的笔法和生动地示例给读者讲述了 DBI 模块的主要组成部分,结构和供用户编程时调用的方法。同时,本文也涉及了一些 DBI

0评论2023-02-09986

用perl做数据库迁移,从MSSQL到MYSQL(三)--V1.1版~多线程+handlerSocket
从前边的程序的运行情况来看,程序是可以运行的,但速度太扯了,在读写1000W条之前速度还是可以的(大概2000条/秒左右),但过了1000W之后(变成400条/秒左右),当然这个与SQL SERVER读取,网络还有服务器等性能都是有关系的,但,这速度,不晓得有测试过的

0评论2023-02-09304

SQL Server数据库(SQL Sever语言 函数以及SQL编程)
1、数学函数:操作一个数据,返回一个结果--去上限: ceiling ☆select ceiling(price) from car--去下限:floor ☆select floor(price) from car--ABS 绝对值--PI(),圆周率,括号里不加任何东西--round() 四舍五入 ☆select round(3.76 ,0)--sqrt() 开根号--

0评论2023-02-09951

更多推荐