首页 > 开发 > PHP > 正文

FleaPHP 开发指南 - 7. 数据表关联

2024-05-04 22:59:27
字体:
来源:转载
供稿:网友

数据表关联是指两个或者多个数据表的记录之间的逻辑关系。

例如:

  • 每一个公民都有一个身份证号码
  • 每一位作者都写了多本(0-n)书籍,而每一本书籍都有多个(1-n)作者
  • 每一篇文章都有多个(0-n)评论
  • 每一个评论都属于一篇文章

目前,fleaphp 支持四种类型的数据表关联,分别是:

  • has_one: 当前表的每一条记录都拥有最多一条(0–1)关联记录
  • has_many: 当前表的每一条记录都拥有多条(0-n)关联记录
  • many_to_many: 当前表的每一条记录都和其他表的多条(0-n)记录关联
  • belongs_to: 当前表的每一条记录都属于另一个表的某条记录

在 fleaphp 中,可以为每一个表数据入口定义多个不同的关联,例如:

<?phpload_class('flea_db_tabledatagateway');class model_productclass extends flea_db_tabledatagateway{    var $tablename = 'product_class';    var $primarykey = 'pclass_id';    var $hasmany = array(        array(            'tableclass' => 'model_permissions',            'foreignkey' => 'pclass_id',            'mappingname' => 'permissions',        ),        array(            'tableclass' => 'model_products',            'foreignkey' => 'pclass_id',            'mappingname' => 'products',            'enabled' => false,        ),    );    var $hasone = array(        array(            'tableclass' => 'model_productclassadverts',            'foreignkey' => 'pclass_id',            'mappingname' => 'advert',        )    );}?>


术语

在详细介绍这四种关联之前,先了解一些后文将会用到的术语。

  • 关联: 一个关联是一个关系,这个关系属于某一个数据表。例如 users 表可能就拥有一个或者多个关联。
  • 主表: 对于一个关联,拥有该关联的数据表就是主表。例如 posts 表定义了一个 many_to_many 关联。那么在这里(指这个关联),posts 就是主表。
  • 关联表: 在一个关联中,关联表就是除主表外的另一个表。
  • 外键: 在数据库原理中,外键的含义很复杂。但在 fleaphp 框架中的数据库关联功能中,外键泛指一个记录中用于关联另一个记录的字段。例如 profile 表中的 user_id 字段就是用于关联 users 表的字段。这个 user_id 字段就是一个外键。
  • 中间表: 在 many_to_many 关联中,除了主表和关联表,还需要另一个表来保存这两个表的记录之间的互相关联关系。这个表称为中间表。

理解这几个术语后,我们再来看每一种关联的详细解释。



has_one 一对一关联

has_one 是一种非常简单的关联关系。表示一个记录拥有另一个记录。这两个记录分别位于两个数据表中。

示例

在一个信息管理系统中,users 表用于存储用户帐户的基本信息,例如用户名、密码等。而 profiles 表则用于存储用户的个人信息,例如家庭住址、邮政编码等。

由于每一个用户(一条 users 表中的记录)都有一份对应的个人信息(一条 profiles 表中的记录)。因此,我们就可以为 users 表定义一个 has_one 关联。

很明显,users 表的记录拥有一条 profiles 表的记录。因此,当 users 表中的一条记录被删除时,被删除记录所拥有的 profiles 表中的关联记录也会被自动删除。

表定义

在 has_one 关联中,要求外键放置在关联表中。

上述例子的表定义简化版如下:

users 表:

  • user_id 主键字段
  • username

profiles 表:

  • profile_id 主键字段
  • address
  • postcode
  • user_id 外键字段

对应的 mysql 代码如下:

create table `users` (    `user_id` int not null auto_increment ,    `username` varchar( 32 ) not null ,    primary key ( `user_id` ));create table `profiles` (    `profile_id` int not null auto_increment ,    `address` varchar( 128 ) not null ,    `postcode` varchar( 8 ) not null ,    `user_id` int not null ,    primary key ( `profile_id` ));

对应的 flea_db_tabledatagateway 继承类的定义如下:

<?phpload_class('flea_db_tabledatagateway');class users extends flea_db_tabledatagateway{    var $tablename = 'users';    var $primarykey = 'user_id';    var $hasone = array(        'tableclass' => 'profiles',        'foreignkey' => 'user_id',        'mappingname' => 'profile',    );}class profiles extends flea_db_tabledatagateway{    var $tablename = 'profiles';    var $primarykey = 'profile_id';}?>

演示代码

<?php// 首先插入一条 users 记录$modelusers =& new users();$newuserid = $modelusers->create(    array('username' => 'dualface'));// 接下来,再插入一条 profiles 记录$modelprofiles =& new profiles();$modelprofiles->create(    array(        'address' => 'sichuan zigong',        'postcode' => '643000',        'user_id' => $newuserid    ));// ok,我们现在尝试读取一条 users 记录,看看会得到什么结果$user = $modelusers->find($newuserid);dump($user);?>

结果很有趣,多出来的 ‘profile’ 字段正好是我们刚刚插入 profiles 表的记录内容:

array(    [user_id] => 1    [username] => dualface    [ref___id] => 1    [profile] => array        (            [profile_id] => 1            [address] => sichuan zigong            [postcode] => 643000            [user_id] => 1            [ref___id] => 1        ))

说明

在上面的例子中,users 类中有一个 $hasone 成员变量。该变量为一个数组:

var $hasone = array(    'tableclass' => 'profiles',    'foreignkey' => 'user_id',    'mappingname' => 'profile',);

$hasone 成员变量用于为一个表数据库入口指定 has_one 关联。

在关联的定义中,tableclass 指定关联表的表数据入口类名称,foreignkey 指定外键字段名,而 mappingname 则指定在主表的查询结果中用什么字段映射关联表的数据。






发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表