多属性、多分类MySQL模式设计[通俗易懂]

多属性、多分类MySQL模式设计[通俗易懂]一、导读 这是来自B乎的一个问答。 当数据同时具备多个属性/分类时,改如何设计表结构和查询? 二、需求描述 我偶尔也会逛逛B乎,看到一些感兴趣的话题也会回复下。 有一次,看到这样的一个话题: 链接:…

一、导读
这是来自B乎的一个问答。 当数据同时具备多个属性/分类时,改如何设计表结构和查询?

二、需求描述
我偶尔也会逛逛B乎,看到一些感兴趣的话题也会回复下。 有一次,看到这样的一个话题:

链接:https://www.zhihu.com/questio…

[mysql] 当数据同时属于多个分类时,该怎么查询?

分类cate字段为[1,2,3,4,5] ,假如要查询满足分类’2’和’5′ 的数据该怎么查询? 我尝试过用 cate like ‘%2%’ AND cate like ‘%5%’去查。 想问有没有更好的办法,我这样写数据少了还好,多了根本没法查,效率太低了。

恰好我以前做过类似的业务需求设计,所以就回复了这个问题。

三、模式设计思路
这个需求可以有几种不同的解决思路,我们分别展开说一下。

(一)用bit数据类型
大概思路如下:
1、物品属性列c1 用bit数据类型 来表示,也就是只有0、1两种取值
2、当物品属性具备某个分类属性时,其值为1,否则为0
3、假如共有5个分类,当物品拥有全部分类属性时,则其值为11111,若其不具备第3个分类属性,则其值为11011,在数据库中转成十进制存储
4、上述两种情况下,将二进制转换成十进制表示,即分别是31和27

[root@yejr.me] [zhishutang]> select conv(11111, 2, 10), conv(11011, 2, 10);
+--------------------+--------------------+
| conv(11111, 2, 10) | conv(11011, 2, 10) |
+--------------------+--------------------+
| 31                 | 27                 |
+--------------------+--------------------+

代码100分

5、然后,只需要对该列用十进制值进行查询比对就行 6、现在如果想判断是否同时具备2、5两个分类属性时,其二进制表示为01001,转成十进制为9,只需要用条件 where c1=9 即可

我们来演示一下:

代码100分[root@yejr.me] [zhishutang]>show create table t_bitG

*************************** 1. row ***************************
       Table: t_bit
Create Table: CREATE TABLE `t_bit` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
  `c1` int(10) unsigned NOT NULL DEFAULT "0",
  `c2` varchar(10) NOT NULL DEFAULT "",
  PRIMARY KEY (`id`),
  KEY `c1` (`c1`)
) ENGINE=InnoDB;

insert into t_bit select 0,conv(00001, 2, 10), "item1";
insert into t_bit select 0,conv(00011, 2, 10), "item2";
insert into t_bit select 0,conv(00111, 2, 10), "item3";
insert into t_bit select 0,conv(01111, 2, 10), "item4";
insert into t_bit select 0,conv(11111, 2, 10), "item5";
insert into t_bit select 0,conv(10111, 2, 10), "item6";
insert into t_bit select 0,conv(11011, 2, 10), "item7";
insert into t_bit select 0,conv(11101, 2, 10), "item8";
insert into t_bit select 0,conv(11110, 2, 10), "item9";

[root@yejr.me] [zhishutang]>select * from t_bit;
+----+----+-------+
| id | c1 | c2    |
+----+----+-------+
|  1 |  1 | item1 |
|  2 |  3 | item2 |
|  3 |  7 | item3 |
|  4 | 15 | item4 |
|  5 | 31 | item5 |
|  6 | 23 | item6 |
|  7 | 27 | item7 |
|  8 | 29 | item8 |
|  9 | 30 | item9 |
+----+----+-------+

[root@yejr.me] [zhishutang]>select * from t_bit where c1 = conv(11011,2,10);
+----+----+-------+
| id | c1 | c2    |
+----+----+-------+
|  7 | 27 | item7 |
+----+----+-------+

#同时我们也注意到这个SQL是可以正常使用索引的
[root@yejr.me] [zhishutang]>desc select * from t_bit where c1 = conv(11011,2,10)G
*************************** 1. row ***************************
           id: 1
  select_type: SIMPLE
        table: t_bit
   partitions: NULL
         type: ref
possible_keys: c1
          key: c1
      key_len: 4
          ref: const
         rows: 1
     filtered: 100.00
        Extra: NULL

下面两种方法是B乎网友的回复,大家也可以参考下。
1、用JSON数据类型,然后利用JSON_CONTAINS()函数进行查询
2、用SET数据类型,然后利用FIND_IN_SET()函数进行查询

不过,JSON和SET这两种数据类型都不方便加索引以及利用索引扫描,即便是用了5.7的JSON+虚拟列功能,索引效率也是比较低的。而支持JSON数据类型 多值索引(multi-valued Indexes) 也要8.0.17 以上版本才支持。

四、总结
这样看来,总的来说,用二进制转十进制方式来解决本案例需求更为高效,也欢迎提出更多方案思路。


延伸阅读

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
转载请注明出处: https://daima100.com/10591.html

(0)
上一篇 2022-12-25
下一篇 2022-12-25

相关推荐

  • Python字典:高效存储和快速查找数据的利器

    Python字典:高效存储和快速查找数据的利器Python字典是一种可变数据类型,用于存储键-值对,是Python中最常用的数据类型之一,也被称为哈希表或关联数组。

    2024-01-09
    113
  • mysql事务和锁的关系_mysql事务注意什么

    mysql事务和锁的关系_mysql事务注意什么事务概述 当多个用户访问同一份数据时,一个用户在更改数据的过程中,可能有其他用户同时发起更改请求,为保证数据库记录的更新从一个一致性状态变为另外一个一致性状态,使用事务处理是非常必要的,事务具有以下四

    2023-05-14
    139
  • 如何通过Oracle实现自动发邮件功能

    如何通过Oracle实现自动发邮件功能在内网项目、外网项目中多有 需要自动监控发邮件提醒的功能,因为邮件功能最便捷、便宜、不用开接口,不用接口费用。现 提供 我在一个内网项目中的使用案例: 案例背景: 在有限的资源下,能够自动给下级单位发

    2022-12-28
    129
  • 解决SQL SERVER 2012自增ID突然断开的问题(ID突然增加1000的)

    解决SQL SERVER 2012自增ID突然断开的问题(ID突然增加1000的)最近这两年创建数据库的自增Id列总是出现一个问题,一开始自增正常,都是1、2、3递增,突然就变成1004、1005这样,一直以为程序有问题,后来多次查阅资料才在国外网站上找到问题。

    2023-02-10
    145
  • 使用Numpy.append实现数组拼接

    使用Numpy.append实现数组拼接在Python编程中,我们经常需要对数组进行拼接。在NumPy中,提供了append函数来实现数组拼接。NumPy的append()函数在原数组末尾添加新的数组,将其元素添加到已有数组的末尾。

    2024-06-17
    47
  • 使用Seaborn进行Python数据可视化

    使用Seaborn进行Python数据可视化a href=”https://www.python100.com/a/sm.html”font color=”red”免责声明/font/a a href=”https://beian.miit.gov.cn/”苏ICP备2023018380号-1/a Copyright www.python100.com .Some Rights Reserved.

    2024-09-21
    11
  • varchar int 查询 到底什么情况下走索引?

    varchar int 查询 到底什么情况下走索引?一个字符类型的、一个int类型的,查询的时候到底会不会走索引,其实很多工作了几年的开发人员有时也会晕,下面就用具体事例来测试一下。 1. 准备工作 先准备2张表,以备后续测试使用。 表1:创建表tes

    2023-02-13
    147
  • Python字典嵌套: 使用内部字典来组织数据

    Python字典嵌套: 使用内部字典来组织数据Python中有一种非常常用的数据类型——字典(Dictionary),它可以在程序中用来存储和组织数据。字典是由键值对组成的集合,其中每个键都对应一个值。但是有些时候,我们需要在一个字典中存储另外一个字典。这就是字典嵌套。本文将介绍如何使用内部字典来组织数据。

    2024-01-25
    109

发表回复

您的电子邮箱地址不会被公开。 必填项已用*标注