1

到目前为止,我所看到的一切都是关于自动删除数据库中的重复条目。我想在开始时强调数据库中没有重复的数据。我还将从以下事实开始,即我仍在学习 RDBMS 设计、规范化、关系,以及最重要的是 SQL!

我有一个客户表,带有一个 clientid (PK) 和一个 client_name。我有一个角色表,其中包含一个角色 ID (PK) 和一个角色名称。任何客户端都可以有多个与之关联的角色。所以我创建了一个client_role_link 表,其中clientid 和roleid 作为两个字段。然后我运行这个:

SELECT client.client_name, role.role_name FROM client 
  LEFT JOIN client_role_link ON client_role_link.clientid=client.clientid 
  LEFT JOIN role ON client_role_link.roleid=role.roleid 
  WHERE (role.roleid='1' OR role.roleid='2')

所以假设我有一个客户端,它有两个与之关联的角色(角色“1”和“2”)。此查询返回两行,每个角色一个。当我得到这些结果时,我正在使用一个while循环来循环遍历结果并将它们输出到一个<select>列表中。然后它会导致两个<option>'s 列出相同的客户端。

我理解为什么我的查询返回两行,这是有道理的。所以这里有两个问题:

  1. 我应该使用更好的数据库/表设计,还是更优化的查询?
  2. 或者这是我应该在 PHP 中处理的事情?如果是这样,是否有更优雅的解决方案将所有结果添加到数组中,然后循环返回数组并删除重复项?

想法?

4

2 回答 2

3

如果您想同时显示这两个角色,那么您的查询是OK.

MySQL不支持数组数据类型,因此您应该PHP使用具有重复客户端名称的结果集填充一个关联数组。

如果您只需要显示具有任一角色的客户端,请使用以下查询:

SELECT  c.*
FROM    client c
WHERE   c.clientid IN
        (
        SELECT  roleid
        FROM    client_role_link crl
        WHERE   crl.roleid IN (1, 2)
        )

这将为每个客户返回一条记录,但不会显示任何角色。

第三种方法将在服务器端内爆角色名称:

SELECT  c.*, GROUP_CONCAT(role_name SEPARATOR ';') AS roles
FROM    client c
LEFT JOIN
        client_role_link crl
ON      crl.clientid = c.clientid
        AND crl.roleid IN (1, 2)
LEFT JOIN
        role r
ON      r.roleid = crl.roleid
GROUP BY
        c.id

并将它们放在PHP一边,但要确保角色名称不会与分隔符混合。

于 2010-02-04T17:04:25.170 回答
0

您可以使用mysql_fetch_assoc()以数组形式将它们取回。然后你可以有一些未经测试的代码,但可能会工作

$sql = "SELECT client.id, client.client_name, role.role_name FROM client LEFT JOIN client_role_link ON client_role_link.clientid=client.clientid LEFT JOIN role ON client_role_link.roleid=role.roleid WHERE (role.roleid='1' OR role.roleid='2')";
$result = mysql_query($sql);
$res = array();
while ($row = mysql_fetch_assoc($result)) {
    $res[$row['id']]['roles'][] = $row['role_name'];
    $res[$row['id']]['client_name'] = $row['client_name']; //you'll be overwriting each iteration probably a better way
}
于 2010-02-04T17:16:55.947 回答