arrayDB,全新而且简单的PHP ORM库

我见过许多PHP ORM库。它们中大多数都要你为想保存在数据库里的每一项写一个类。无缘无故地继承这个、继承那个,而那些内容通常都是重复和反复无常的。

\

既然条目包含相似数据类型以及相似关系的字段,一个编写良好的类就可适用于所有情况。如果你需要一个库来简化这些事情,这就是我的方法。

\

arrayDB ORM库只包含5个类。你基本上只使用其中一个,而其余类则在内部使用,仅此而已。缓存及其与数据库之间的同步全是自动的,你无需对其进行跟踪。

\

要开始使用这个库,你只需简单地定义:

\
  • 你的数据模型(你需要保存哪几项,他们的字段以及字段间的关系)。\
  • 你的MySQL连接方式。\
  • 你的缓存配置。\

定义数据模型

\

所有的数据模型定义被写成像这样的数组:

\
\    $model=array(\        'user'=\u0026gt;array(\            'conf'=\u0026gt;array('len'=\u0026gt;7),\            'fields'=\u0026gt;array(\                'name'=\u0026gt;array('len'=\u0026gt;50)\            ),\            'has_many'=\u0026gt;array(\                'posts'=\u0026gt;array('type'=\u0026gt;'post', \'foreign_name'=\u0026gt;'writer')\            ),\            'many_to_many'=\u0026gt;array(\                'liked_posts'=\u0026gt;array('type'=\u0026gt;'post', \'foreign_name'=\u0026gt;'likers'),\            ),\            'self_ref'=\u0026gt;array('friends')\        ),\\        'post'=\u0026gt;array(\            'conf'=\u0026gt;array('len'=\u0026gt;10),\            'fields'=\u0026gt;array(\                'text'=\u0026gt;array('len'=\u0026gt;200),\                'view_count'=\u0026gt;array('type'=\u0026gt;'numeric', 'len'=\u0026gt;5)\                // 默认的字段类型是text,这里将类型定义成numeric\            )\        )\    );\
\

这里我们有两张表:用户和帖子。

\

用户有姓名,有些用户发帖子而有些用户关注帖子。帖子有文本内容,浏览量以及关注者。

\

用户还有其他许多用户作为好友。

\

使用这个模型,我们想通过$user['posts']获取用户发布的所有帖子,通过$post['writer']获取帖子的作者。这是一对多关系。

\

我们还想通过$user['liked_posts']获取用户关注过的帖子,通过$post['likers']获取帖子的关注者。这是多对多关系。

\

最后我们想通过$user['friends']获取用户的好友列表。这是自引用关系。

\

定义MySQL连接方式

\

定义MySQL连接方式也写成一个像这样的数组:

\
\    $db_config=array(\        'hostname'=\u0026gt;'localhost', 'database'=\u0026gt;'social', \'username'=\u0026gt;'root', 'password'=\u0026gt;''\    );\
\

定义缓存配置

\

目前,我们有三种缓存类型实现:APC、Memcached以及普通文本文件。 要使用APC,这样的配置数组就够了:

\
\    $cache_config=array('type'=\u0026gt;'apc');\
\

要使用Memcached,你需要提供一些参数:

\
\    $cache_config=array('type'=\u0026gt;'memcached', 'host'=\u0026gt;'127.0.0.1', \'port'=\u0026gt;11211, 'timeout'=\u0026gt;1);\
\

要使用普通文本文件,你需要创建一个可读可写的目录并提供其绝对路径:

\
\    $cache_config=array('type'=\u0026gt;'file', 'path'=\u0026gt;'/tmp/my_project_cache')\
\

还有一个可选参数“prefix”。如果给定,它将用作缓存的键名。

\

开始使用

\

现在是时候使用那些我们定义过的数据了。初始化该库,我们只需要这几行代码。

\
\    DB::init($db_config);\    CACHE::init($cache_config);\\    ADB::init($model);\    $adb=ADB::get_instance();\
\

创建表

\

这项任务只需执行一次。我们告诉库去创建所需的数据库表。它就会负责完成关系等相关的复杂工作。

\
\    $adb-\u0026gt;create_tables();\
\

我们只需在发布的时候执行该方法一次。如果在插入数据之后,调用该方法将导致数据丢失。

\

使用记录

\

我们手头拥有这个$adb实例。我们将通过它来获取所有数据。

\

创建记录

\

我们提供表名和由字段名称及数据组成的键值对数组来创建一条记录。

\
\    $uid1=$adb-\u0026gt;create('user', array('name'=\u0026gt;'John'));\    $uid2=$adb-\u0026gt;create('user', array('name'=\u0026gt;'Marry'));\\    $pid1=$adb-\u0026gt;create('post', array(\        'writer'=\u0026gt;$uid1,\        'text'=\u0026gt;'What a wonderful world!'\    ));\\    $pid2=$adb-\u0026gt;create('post', array(\        'writer'=\u0026gt;$uid2,\        'text'=\u0026gt;'Life is beautiful!'\    ));\
\

创建多对多关系

\

第一个参数是表名。第二个参数是被关联记录的局部名称。第三个参数是记录的ID,第三个参数是被关联记录的ID。

\
\    $adb-\u0026gt;relate('user', 'friends', $uid1, $uid2);\\    $adb-\u0026gt;relate('user', 'liked_posts', $uid1, $pid1); // 关注自己的帖子 :)\    $adb-\u0026gt;relate('user', 'liked_posts', $uid1, $pid2);\\    $adb-\u0026gt;relate('user', 'liked_posts', $uid2, $pid1);\
\

列举数据

\

我们可在一个简单的循环中列举所有写过帖子的用户以及帖子的关注者:

\
\    foreach ($adb-\u0026gt;id_list('user') as $uid) {\        // load user\        $user=$adb-\u0026gt;load('user', $uid);\        echo '\u0026lt;h1\u0026gt;' . $user['name'] . '\u0026lt;/h1\u0026gt;' . \"\\";\\        echo '\u0026lt;h2\u0026gt;Posts: \u0026lt;/h2\u0026gt;' . \"\\";\        echo '\u0026lt;ul\u0026gt;' . \"\\";\\        foreach ($user['posts'] as $pid) {\                //load post of user\                $post=$adb-\u0026gt;load('post', $pid);\                $likers=array();\\                foreach ($post['likers'] as $lid) {\                    // load liker of post\                    $liker=$adb-\u0026gt;load('user', $lid);\                    $likers[]=$liker['name'];\                }\                $likers=(count($likers)) ? '\u0026lt;br /\u0026gt;' . implode(', ', \$likers) . ' liked.' : '';\                echo '\u0026lt;li\u0026gt;' . $post['text'] . ' ' . $likers . '\u0026lt;/li\u0026gt;' . \\"\\";\            }\\            echo '\u0026lt;/ul\u0026gt;' . \"\\";\        }\
\

更新记录

\

我们能够像下面这样更新任意一个记录:

\
\    $user1=$adb-\u0026gt;load('user', $uid1);\    $user1['name']='Jack';\    // 无需调用任何保存方法,保存以及缓存的同步更新全是自动的。\
\

如果我们需要一次更新多个字段,这是另一种方法:

\
\    $post1=$adb-\u0026gt;load('post', $pid1);\    $post1-\u0026gt;update(array('writer'=\u0026gt;$uid2, 'text'=\u0026gt;'Not a wonderful \world!'));\
\

删除关联

\

这和创建关联一样:

\
\    $adb-\u0026gt;unrelate('user', 'friends', $uid1, $uid2);\    $adb-\u0026gt;unrelate('user', 'liked_posts', $uid1, $pid1);\
\

删除记录

\

我们可以删除记录,同时保存或删除与该记录相关联的数据。

\
\    $adb-\u0026gt;delete('user', $uid1);\    // 用户删掉了,帖子成为匿名的了。\\    $adb-\u0026gt;delete('user', $uid1, true);\    // 用户以及用户的帖子都删掉了。\
\

查询更多

\

作为示例,我想获得最受关注的5篇帖子。如下是我们所有要做的:

\
\    foreach ($adb-\u0026gt;id_list('post', false, 'likers DESC', 5) as $pid) {\        $post=$adb-\u0026gt;load('post', $pid);\        // 可对帖子做任何操作\    }\
\

抑或是我们想要获取用户被关注次数超过5次的所有帖子,我们可以执行以下操作:

\
\    $user=$adb-\u0026gt;load('user', $uid1);\    foreach ($user-\u0026gt;id_list('post', 'view_count\u0026gt;5') as $pid) {\        $post=$adb-\u0026gt;load('post', $pid);\        // 可对帖子做任何操作\    }\
\

主要目标

\

对于单独的帖子页面,我们的代码将会是这么简单:

\
\    $post=$adb-\u0026gt;load('post', $pid1);\    $writer=$adb-\u0026gt;load('user', $post['writer']);\\    echo $writer['name'] . ' wrote' . \"\u0026lt;br /\u0026gt;\\";\    echo $post['text'] . \"\u0026lt;br /\u0026gt;\\";\\    $post['view_count']++;\    // 是的,增大浏览量就是这么简单\
\

这些代码里有任何查询或者缓存逻辑吗?没有,主要目标是保持简单。

\

市面上有很多著名的替代产品。他们文档完备,支持得也更好。该库不是他们中的一员,它目前还不是一部状态良好的机器。在我看来,这是最简单且容易上手的方法。如果ORM库的目的是为了让编码人员不用关心数据库逻辑,这个库是其他新兴库中最自信的一个。jQuery是最简单的javascript框架,也因此成了标准。所以,易于使用的PHP ORM库也有这样的机会。

\

这个库的下载地址:[arrayDB github]

\

欢迎提出意见和建议。

\

关于作者

\

aafa10ba77ef7dc2abc374c511b3fbde.jpg

\

Mustafa Dokumaci是一位来自土耳其伊斯坦布尔的软件工程师。他的专业是环境工程和会计学,但他目前就职于sporcum.com和hipsin.com。Mustafa拥有六年的PHP、MySQL、Apache、Nginx、Python、CodeIgniter、Magento等的使用经验。

\\\

查看英文原文arrayDB, a New and Easy PHP ORM

\

感谢侯伯薇对本文的审校。

\

给InfoQ中文站投稿或者参与内容翻译工作,请邮件至editors@cn.infoq.com。也欢迎大家通过新浪微博(@InfoQ)或者腾讯微博(@InfoQ)关注我们,并与我们的编辑和其他读者朋友交流。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
自主封装的PHP ORM框架,面向对象的PDO数据操作,API框架,支持Get/Post/Put/Delete多种请求方式。 代码示例: <?php use Models\User; require '../application.php'; require '../loader-api.php'; //适合查询,如:获取用户列表或者单个用户信息 execute_request(HttpRequestMethod::Get, function() { $action = request_action(); //判断是否存在 if ($action == 1) { list($type, $value) = filter_request(array( request_int('type', 1, 2, 3), //1.用户名 2.邮箱 3.手机号 request_string('value'))); $type_field_map = array( 1 => User::$field_username, 2 => User::$field_email, 3 => User::$field_phone ); if ($type == 2 && !is_email($value) || $type == 3 && !is_mobilephone($value)) { die_error(USER_ERROR, $type_field_map[$type]['name'] . '格式无效'); } $user = new User(); $user->set_where_and($type_field_map[$type], SqlOperator::Equals, $value); $result = $user->exists(create_pdo()); echo_result($result ? 1 : 0); //存在返回1,不存在返回0 } //查询单条信息 if ($action == 2) { list($userid) = filter_request(array( request_userid())); //查询单条数据 $user = new User($userid); //set_query_fields可以指定查询字段,下面两种写法均可 //$user->set_query_fields('userid, username, email'); //$user->set_query_fields(array(User::$field_userid, User::$field_username, User::$field_email)); //还可设置where条件进行查询 //$user->set_where_and(User::$field_status, SqlOperator::Equals, 3); //$user->set_where_and(User::$field_truename, SqlOperator::IsNullOrEmpty); //$user->set_where_and(User::$field_age, SqlOperator::In, array(27, 29)); //$user->set_where_and(User::$field_regtime, SqlOperator::LessThan, '-6 month'); //创建数据连接 $db = create_pdo(); $result = $user->load($db, $user); //也可以用Model类的静态方法 //$result = Model::load_model($db, $user, $user); if (!$result[0]) die_error(PDO_ERROR_CODE, '获取用户信息时数据错误'); if (!$user) di
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值