共享租户对象
Shared tenant objects
我有一个带有单个数据库的多租户应用程序。我有一个 "entity" table 存储所有对象的地方。 "sahred_entity" table 用于存储租户X共享给租户Y的对象。例如"Tenant 2"可以共享"Entity with ID 4"到"Tenant 1"。
在下面的示例中,"Entity with ID 4" 共享给 "Tenant 1" 和 "Tenant 3"
+--------+--------------------------------------------------
| Table | Create Table
+--------+--------------------------------------------------
| entity | CREATE TABLE `entity` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`tenant_id` int(10) unsigned NOT NULL,
`added_at` timestamp NOT NULL,
`color` varchar(20) NOT NULL,
`size` varchar(5) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=latin1 |
+--------+--------------------------------------------------
+---------------+---------------------------------------
| Table | Create Table
+---------------+---------------------------------------
| shared_entity | CREATE TABLE `shared_entity` (
`tenant_to` int(10) unsigned NOT NULL,
`tenant_from` int(10) unsigned NOT NULL,
`entity_id` int(10) unsigned NOT NULL
) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
+---------------+---------------------------------------
样本数据为
select * from entity;
+----+-----------+---------------------+--------+------+
| id | tenant_id | added_at | color | size |
+----+-----------+---------------------+--------+------+
| 1 | 1 | 2019-03-07 00:00:00 | red | m |
| 2 | 1 | 2019-03-07 00:00:00 | green | xl |
| 3 | 2 | 2019-03-07 00:00:00 | green | xl |
| 4 | 2 | 2019-03-07 00:00:00 | red | m |
| 5 | 3 | 2019-03-07 00:00:00 | yellow | l |
+----+-----------+---------------------+--------+------+
select * from shared_entity;
+-----------+-------------+-----------+
| tenant_to | tenant_from | entity_id |
+-----------+-------------+-----------+
| 1 | 2 | 4 |
| 3 | 2 | 4 |
+-----------+-------------+-----------+
现在我需要创建一个简单的搜索查询。现在我找到了两种方法。第一个是通过自我加入
SELECT e.* FROM `entity` as e
LEFT JOIN entity as e1 ON (e.id = e1.id AND e1.tenant_id = 1)
LEFT JOIN entity as e2 ON (e.id = e2.id AND e2.id IN (4))
WHERE (e1.id IS NOT NULL OR e2.id IS NOT NULL) AND e.`color` = 'red';
第二种是通过子查询和联合
SELECT * FROM
(
SELECT * FROM entity as e1 WHERE e1.tenant_id = 1
UNION
SELECT * FROM entity as e2 WHERE e2.id IN(4)
) as entity
WHERE color = 'red';
两个查询return 预期结果
+----+-----------+---------------------+-------+------+
| id | tenant_id | added_at | color | size |
+----+-----------+---------------------+-------+------+
| 1 | 1 | 2019-03-07 00:00:00 | red | m |
| 4 | 2 | 2019-03-07 00:00:00 | red | m |
+----+-----------+---------------------+-------+------+
但是哪种方法更适合大型 tables?如何创建正确的索引?或者也许有更好的解决方案?
您也可以使用以下查询获得相同的结果
SELECT *
FROM entity
WHERE (tenant_id = 1 or id = 4) AND color = 'red'
我不清楚为什么需要所有连接
每个 table 应该有一个 PRIMARY KEY
。 shared_entity
需要 PRIMARY KEY(tenant_from, tenant_to, entity_id)
;任何顺序都可能就足够了。
至于性能,hogan 的建议,加上 INDEX(color)
,小 table:
SELECT *
FROM entity
WHERE (tenant_id = 1 OR id = 4)
AND color = 'red'
但是 OR
阻止了大多数形式的优化。如果color
的选择性足够,那么这不是问题;它将简单地扫描所有 "red" 个项目,检查每个项目是否 tenent_id
和 id
.
如果有数千个红色项目,这会运行更快:
( SELECT *
FROM entity
WHERE tenant_id = 1
AND color = 'red' )
UNION DISTINCT
( SELECT *
FROM entity
WHERE id = 4
AND color = 'red' )
连同
INDEX(color, tenant_id) -- in either order
-- PRIMARY KEY(id) -- already exists and is unique
如果您知道 tenant-1 和 id-4 不引用同一行,UNION DISTINCT
可以加速到 UNION ALL
。
我有一个带有单个数据库的多租户应用程序。我有一个 "entity" table 存储所有对象的地方。 "sahred_entity" table 用于存储租户X共享给租户Y的对象。例如"Tenant 2"可以共享"Entity with ID 4"到"Tenant 1"。
在下面的示例中,"Entity with ID 4" 共享给 "Tenant 1" 和 "Tenant 3"
+--------+--------------------------------------------------
| Table | Create Table
+--------+--------------------------------------------------
| entity | CREATE TABLE `entity` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`tenant_id` int(10) unsigned NOT NULL,
`added_at` timestamp NOT NULL,
`color` varchar(20) NOT NULL,
`size` varchar(5) NOT NULL,
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=latin1 |
+--------+--------------------------------------------------
+---------------+---------------------------------------
| Table | Create Table
+---------------+---------------------------------------
| shared_entity | CREATE TABLE `shared_entity` (
`tenant_to` int(10) unsigned NOT NULL,
`tenant_from` int(10) unsigned NOT NULL,
`entity_id` int(10) unsigned NOT NULL
) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
+---------------+---------------------------------------
样本数据为
select * from entity;
+----+-----------+---------------------+--------+------+
| id | tenant_id | added_at | color | size |
+----+-----------+---------------------+--------+------+
| 1 | 1 | 2019-03-07 00:00:00 | red | m |
| 2 | 1 | 2019-03-07 00:00:00 | green | xl |
| 3 | 2 | 2019-03-07 00:00:00 | green | xl |
| 4 | 2 | 2019-03-07 00:00:00 | red | m |
| 5 | 3 | 2019-03-07 00:00:00 | yellow | l |
+----+-----------+---------------------+--------+------+
select * from shared_entity;
+-----------+-------------+-----------+
| tenant_to | tenant_from | entity_id |
+-----------+-------------+-----------+
| 1 | 2 | 4 |
| 3 | 2 | 4 |
+-----------+-------------+-----------+
现在我需要创建一个简单的搜索查询。现在我找到了两种方法。第一个是通过自我加入
SELECT e.* FROM `entity` as e
LEFT JOIN entity as e1 ON (e.id = e1.id AND e1.tenant_id = 1)
LEFT JOIN entity as e2 ON (e.id = e2.id AND e2.id IN (4))
WHERE (e1.id IS NOT NULL OR e2.id IS NOT NULL) AND e.`color` = 'red';
第二种是通过子查询和联合
SELECT * FROM
(
SELECT * FROM entity as e1 WHERE e1.tenant_id = 1
UNION
SELECT * FROM entity as e2 WHERE e2.id IN(4)
) as entity
WHERE color = 'red';
两个查询return 预期结果
+----+-----------+---------------------+-------+------+
| id | tenant_id | added_at | color | size |
+----+-----------+---------------------+-------+------+
| 1 | 1 | 2019-03-07 00:00:00 | red | m |
| 4 | 2 | 2019-03-07 00:00:00 | red | m |
+----+-----------+---------------------+-------+------+
但是哪种方法更适合大型 tables?如何创建正确的索引?或者也许有更好的解决方案?
您也可以使用以下查询获得相同的结果
SELECT *
FROM entity
WHERE (tenant_id = 1 or id = 4) AND color = 'red'
我不清楚为什么需要所有连接
每个 table 应该有一个 PRIMARY KEY
。 shared_entity
需要 PRIMARY KEY(tenant_from, tenant_to, entity_id)
;任何顺序都可能就足够了。
至于性能,hogan 的建议,加上 INDEX(color)
,小 table:
SELECT *
FROM entity
WHERE (tenant_id = 1 OR id = 4)
AND color = 'red'
但是 OR
阻止了大多数形式的优化。如果color
的选择性足够,那么这不是问题;它将简单地扫描所有 "red" 个项目,检查每个项目是否 tenent_id
和 id
.
如果有数千个红色项目,这会运行更快:
( SELECT *
FROM entity
WHERE tenant_id = 1
AND color = 'red' )
UNION DISTINCT
( SELECT *
FROM entity
WHERE id = 4
AND color = 'red' )
连同
INDEX(color, tenant_id) -- in either order
-- PRIMARY KEY(id) -- already exists and is unique
如果您知道 tenant-1 和 id-4 不引用同一行,UNION DISTINCT
可以加速到 UNION ALL
。