首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

使用业务实体的自然id作为主键

基础概念: 在数据库设计中,主键(Primary Key)是用于唯一标识表中每一条记录的一个或一组字段。自然ID(Natural ID)是指业务上已经存在的、具有唯一性的标识符,如身份证号、手机号、员工编号等。

相关优势

  1. 业务意义明确:自然ID通常在业务场景中有明确的含义,易于理解和使用。
  2. 唯一性:由于自然ID在业务上是唯一的,因此可以确保数据的唯一性。
  3. 减少数据冗余:使用自然ID作为主键可以避免生成额外的ID字段,减少数据冗余。

类型

  • 单一自然ID:如身份证号、手机号等单个字段作为主键。
  • 复合自然ID:由多个字段组合而成的唯一标识,如(公司ID,员工编号)。

应用场景

  • 用户管理系统:使用手机号或邮箱作为主键。
  • 订单系统:使用订单号作为主键。
  • 库存管理系统:使用商品SKU作为主键。

可能遇到的问题及原因

  1. 性能问题:自然ID可能较长或不规则,影响索引效率。
    • 原因:长字符串或不规则的数据类型会导致索引变大,降低查询效率。
    • 解决方法:可以考虑使用哈希函数生成固定长度的主键,或使用UUID作为辅助主键。
  • 数据变更问题:自然ID可能会在业务过程中发生变化。
    • 原因:如用户更换手机号、员工调动等。
    • 解决方法:尽量避免自然ID的变更,或在变更时同步更新所有相关联的数据。
  • 隐私泄露问题:某些自然ID可能包含敏感信息。
    • 原因:如直接使用身份证号作为主键可能导致隐私泄露。
    • 解决方法:对敏感信息进行脱敏处理,或使用代理键(Surrogate Key)作为主键。

示例代码: 假设我们有一个用户表,使用手机号作为自然ID作为主键:

代码语言:txt
复制
CREATE TABLE users (
    phone_number VARCHAR(15) PRIMARY KEY,
    name VARCHAR(100),
    email VARCHAR(100)
);

如果担心手机号变更带来的问题,可以考虑使用代理键:

代码语言:txt
复制
CREATE TABLE users (
    user_id INT AUTO_INCREMENT PRIMARY KEY,
    phone_number VARCHAR(15) UNIQUE NOT NULL,
    name VARCHAR(100),
    email VARCHAR(100)
);

在这种情况下,user_id 是一个自增的代理键,而 phone_number 作为唯一索引保留其业务意义。

通过这种方式,既保证了数据的唯一性和业务意义,又避免了直接使用自然ID可能带来的问题。

页面内容是否对你有帮助?
有帮助
没帮助

相关·内容

InnoDB引擎为什么推荐使用自增ID作为主键?

如果主键是非自增 id,为了确保索引有序,MySQL 就需要将每次插入的数据都放到合适的位置上。...这就造成了页分裂,这个大量移动数据的过程是会严重影响插入效率的。 自增id 可以保证每次插入时B+索引是从右边扩展的,可以避免B+树频繁合并和分裂(对比使用UUID而言)。...因此,只要可以,请尽量在InnoDB上采用自增字段做主键。 ◆ 二、尽量使用更小的主键 在满足业务需求的情况下,尽量使用占空间更小的主键。...普通索引的叶子节点上保存的是主键 id 的值,如果主键 id 占空间较大的话,那将会成倍增加 MySQL 空间占用大小。 ◆ 三、什么时候不需用自增主键?...(2)KV场景 在全表只有一个唯一索引(Key-Value场景),且读多写少的前提下,应尽量避免查询时回表(也就是搜索两颗索引树),这种情况可以考虑用业务字段做主键。

3.7K30

为什么MySQL不推荐使用uuid或者雪花id作为主键?

p=5090 前言 在mysql中设计表的时候,mysql官方推荐不要使用uuid或者不连续不重复的雪花id(long形且唯一,单机递增),而是推荐连续自增的主键id,官方的推荐是auto_increment...key作为主键,其它我们完全保持不变....带着疑问,我们来探讨一下这个问题: 二、使用uuid和自增id的索引结构对比 2.1.使用自增id的内部结构 ? 自增的主键的值是顺序的,所以Innodb把每一条记录都存储在一条记录的后面。...结论:使用innodb应该尽可能的按主键的自增顺序插入,并且尽可能使用单调的增加的聚簇键的值来插入新行。 2.3.使用自增id的缺点 那么使用自增的id就完全没有坏处了吗?...并不是,自增id也会存在以下几点问题: ①别人一旦爬取你的数据库,就可以根据数据库的自增id获取到你的业务增长信息,很容易分析出你的经营情况 ②对于高并发的负载,innodb在按主键进行插入的时候会造成明显的锁争用

4K20
  • 使用雪花 id 或 uuid 作为 MySQL 主键,被老板怼了一顿!

    , uuid 作为主键, 随机 key 作为主键, 其它我们完全保持不变....根据控制变量法, 我们只把每个表的主键使用不同的策略生成, 而其他的字段完全一样,然后测试一下表的插入速度和查询速度: 注:这里的随机 key 其实是指用雪花算法算出来的前后不连续不重复**无规律**...带着疑问, 我们来探讨一下这个问题: ### 二、使用 uuid 和自增 id 的索引结构对比 **2.1 使用自增 id 的内部结构** !...结论:使用 innodb 应该尽可能的按主键的自增顺序插入,并且尽可能使用单调的增加的聚簇键的值来插入新行 **2.3 使用自增 id 的缺点** 那么使用自增的 id 就完全没有坏处了吗?...并不是,自增 id 也会存在以下几点问题: ①:别人一旦爬取你的数据库, 就可以根据数据库的自增 id 获取到你的业务增长信息,很容易分析出你的经营情况 ②:对于高并发的负载,innodb 在按主键进行插入的时候会造成明显的锁争用

    2.9K00

    使用雪花id或uuid作为Mysql主键,被老板怼了一顿!

    前言: 在mysql中设计表的时候,mysql官方推荐不要使用uuid或者不连续不重复的雪花id(long形且唯一),而是推荐连续自增的主键id,官方的推荐是auto_increment,那么为什么不建议采用...作为主键,其它我们完全保持不变.根据控制变量法,我们只把每个表的主键使用不同的策略生成,而其他的字段完全一样,然后测试一下表的插入速度和查询速度: 注:这里的随机key其实是指用雪花算法算出来的前后不连续不重复无规律的...带着疑问,我们来探讨一下这个问题: # 使用uuid和自增id的索引结构对比 1.使用自增id的内部结构 自增的主键的值是顺序的,所以Innodb把每一条记录都存储在一条记录的后面。...结论:使用innodb应该尽可能的按主键的自增顺序插入,并且尽可能使用单调的增加的聚簇键的值来插入新行 3.使用自增id的缺点 那么使用自增的id就完全没有坏处了吗?...并不是,自增id也会存在以下几点问题: ①:别人一旦爬取你的数据库,就可以根据数据库的自增id获取到你的业务增长信息,很容易分析出你的经营情况 ②:对于高并发的负载,innodb在按主键进行插入的时候会造成明显的锁争用

    1.6K10

    使用雪花id或uuid作为MySQL主键,被老板怼了一顿!

    磊哥,前几天在做项目demo的时候,使用雪花id或uuid作为Mysql主键,被老板怼了一顿!...key作为主键,其它我们完全保持不变.根据控制变量法,我们只把每个表的主键使用不同的策略生成,而其他的字段完全一样,然后测试一下表的插入速度和查询速度: 注:这里的随机key其实是指用雪花算法算出来的前后不连续不重复无规律的...带着疑问,我们来探讨一下这个问题: 二、使用uuid和自增id的索引结构对比 2.1 使用自增id的内部结构 ? 自增的主键的值是顺序的,所以Innodb把每一条记录都存储在一条记录的后面。...结论:使用innodb应该尽可能的按主键的自增顺序插入,并且尽可能使用单调的增加的聚簇键的值来插入新行 2.3 使用自增id的缺点 那么使用自增的id就完全没有坏处了吗?...并不是,自增id也会存在以下几点问题: ①:别人一旦爬取你的数据库,就可以根据数据库的自增id获取到你的业务增长信息,很容易分析出你的经营情况 ②:对于高并发的负载,innodb在按主键进行插入的时候会造成明显的锁争用

    8.9K32

    使用雪花id或uuid作为Mysql主键,被老板怼了一顿!

    ,而是推荐连续自增的主键id,官方的推荐是auto_increment,那么为什么不建议采用uuid,使用uuid究竟有什么坏处?...key作为主键,其它我们完全保持不变。...带着疑问,我们来探讨一下这个问题: 二、使用uuid和自增id的索引结构对比 2.1 使用自增id的内部结构 自增的主键的值是顺序的,所以Innodb把每一条记录都存储在一条记录的后面。...结论:使用innodb应该尽可能的按主键的自增顺序插入,并且尽可能使用单调的增加的聚簇键的值来插入新行 2.3 使用自增id的缺点 那么使用自增的id就完全没有坏处了吗?...并不是,自增id也会存在以下几点问题: ①. 别人一旦爬取你的数据库,就可以根据数据库的自增id获取到你的业务增长信息,很容易分析出你的经营情况 ②.

    1.2K20

    使用雪花id或uuid作为Mysql主键,被老板怼了一顿!

    ---- 前言 在mysql中设计表的时候,mysql官方推荐不要使用uuid或者不连续不重复的雪花id(long形且唯一,单机递增),而是推荐连续自增的主键id,官方的推荐是auto_increment...,user_random_key,分别表示自动增长的主键,uuid作为主键,随机key作为主键,其它我们完全保持不变....带着疑问,我们来探讨一下这个问题: 二、使用uuid和自增id的索引结构对比 2.1.使用自增id的内部结构 ? 自增的主键的值是顺序的,所以Innodb把每一条记录都存储在一条记录的后面。...结论:使用innodb应该尽可能的按主键的自增顺序插入,并且尽可能使用单调的增加的聚簇键的值来插入新行 2.3.使用自增id的缺点 那么使用自增的id就完全没有坏处了吗?...并不是,自增id也会存在以下几点问题: ①别人一旦爬取你的数据库,就可以根据数据库的自增id获取到你的业务增长信息,很容易分析出你的经营情况 ②对于高并发的负载,innodb在按主键进行插入的时候会造成明显的锁争用

    2.2K10

    使用 int 和 string 作为主键的优劣

    然而,在某些场景下,使用字符串(string)作为主键也是可行的。本文将分析使用 int 和 string 作为主键的优劣,并讨论在实际应用中如何选择合适的主键类型。 首先,我们需要了解主键的概念。...一、使用 int 作为主键的优劣 优点 (1)高性能:整数类型的处理速度通常快于字符串类型,因为整数操作的计算复杂度更低。在数据库中,使用 int 作为主键可以提高查询和更新的效率。...三、如何选择合适的主键类型 根据业务需求:根据具体业务场景来选择合适的主键类型。...例如,如果业务需求中需要处理大量数值类型的数据,可以使用 int 作为主键;如果需要处理字符串类型的数据,可以使用 string 作为主键。...综上所述,使用 int 和 string 作为主键各有优劣。在实际应用中,需要根据具体业务场景和数据特点来选择合适的主键类型,以满足数据的存储、管理和处理需求。

    1.6K50

    为什么建议使用递增的业务ID

    什么是递增的业务ID 1. 什么是业务ID定义 业务ID是一个唯一标识符,用于在系统中标识一个特定的业务实体。 业务ID标识的业务实体可能是一个订单、一个账户、一个病历,或者一个课程。...业务ID是我们理解、管理和操作业务实体的关键。通过业务ID,我们可以查询、更新和删除业务实体,也可以跟踪业务实体的状态和历史。 2. 什么是递增的业务ID 递增的业务ID是一种常见的ID生成策略。...它的基本思想是,每当创建一个新的业务实体时,就在上一个ID的基础上加一(也可以是加一定的数值),生成一个新的ID。这样,我们就可以得到一个唯一且递增的ID序列,用于标识和管理业务实体。...为什么要使用递增的业务ID 1. 易于管理和跟踪 使用递增的业务ID可以使得数据管理和跟踪变得更加容易。...例如,我们可以使用二分查找算法来快速定位到特定的业务ID,或者使用基于比较的排序算法来对业务ID进行排序。 2. 有助于数据库性能优化 使用递增的业务ID还可以帮助优化数据库的性能。

    29410

    序列作为主键使用的原理、优缺点讨论

    这几天和同事一直在讨论关于表设计中主键选择的问题,用sequence作为主键究竟有什么好处,又有什么缺点,尤其是有些事务场景上下文需要用到创建的序列值,如何用?...如果一个事务中INSERT一张表后,还需要插入时的主键ID值,作为外键插入其他表,那么就需要在INSERT第一张表前使用select seq.nextval from dual提前获取可用的ID保存到一个变量中...注:最近在讨论某系统和一个外系统做全局事务的事情,本想用这个主键作为两系统传输的一部分,用于控制全局事务,且用其作为判断交易先后顺序的依据,这是不太符合要求的,因为是RAC,序列是基于实例级cache,...,当然究竟是否会产生资源争用,还是要依据实际的业务并发量,或者压力测试才能证明,这里只是说可能会这样的问题,不是一定会,否则Oracle就不会有提供这种order的创建属性,凡事不绝对。...如果一个事务中INSERT一张表后,还需要插入时的主键ID值,作为外键插入其他表,那么就需要在INSERT第一张表前使用select seq.nextval from dual提前获取可用的ID保存到一个变量中

    1.1K20

    使用 C# 9 的records作为强类型ID - 初次使用

    强类型ID 实体通常是整数,GUID或者string类型,因为数据库直接支持这些类型,但是,如果实体的ID的类型是一样的,比如都是整数的ID,这有可能会出现ID值传错的问题,看下边的示例。...幸运的是,可以定义强类型id来解决这个问题,这个想法很简单,为每个实体的ID声明一个特定的类型,现在需要这样写: // 使用强类型ID代替整数ID public void AddProductToOrder...a.Equals(b); } 上面的代码没什么难的,但是如果每个实体都需要的话,那确实有点麻烦,在C# 9 可以使用source generators来完成这些,但是C# 9还引入了另一个功能,使用起来更方便...主要区别在于:我们的手动实现是struct,即值类型,但是记录是引用类型,这意味着它们可以为null,这可能不是主要问题,尤其是在使用可为空的引用类型的情况下,但是要知道这一点。...现在为模型中的每个实体编写一个强类型的id是不是很简单,使用Record 非常方便,当然,还有其他问题需要考虑,例如JSON序列化,与Entity Framework Core一起使用等,但这是另一篇文章的故事

    54820

    高效使用lua作为业务开发语言的秘诀在这里!

    如果你选择使用蓝图,我只能说作为程序员你骨骼清奇,这玩意都能用于实际业务开发,跑跑demo,做做prototype还行,一般没有业务用蓝图作为主要开发语言用于产品,它最大的问题是不能merge,无法多人协作开发...嘿嘿嘿,slua就是在unity下广泛流行的lua开发插件,适用于在unity引擎使用lua作为开发语言开发游戏业务,而作者就是我本人,那理所当然,我有必要做一个unreal版本方便slua的用户可以快速迁移到...2)支持使用lua function作为蓝图的事件代理 在蓝图里支持代理,例如: ?...6)支持out类型的蓝图参数和引用类型的c++参数作为返回值 与c#类似,蓝图也支持out类型的参数用于返回多余的返回值,而c++这里,一般我们使用非const引用来返回多余参数(当然也可能不),slua...答案是 slua-unreal 已经集成在潘多拉智能营销解决方案,用于支持腾讯多款游戏业务,通过了DAU千万级的产品测试,上线质量稳定,大家可以放心使用。

    3.8K31

    使用 C# 9 的records作为强类型ID - 路由和查询参数

    上一篇文章,我介绍了使用 C# 9 的record类型作为强类型id,非常简洁 public record ProductId(int Value); 但是在强类型id真正可用之前,还有一些问题需要解决...路由和查询字符串参数的模型绑定 假设我们有一个这样的实体: public record ProductId(int Value); public class Product { public...ProductId,由于它不是int,是我们定义的强类型ID,并且没有关联的类型转换器。...": 0.8 } 现在是返回了,但是还有点问题,id 在json中显示了一个对象,如何在json中处理,是我们下一篇文章给大家介绍的,现在还有一点是,我上面写了一个ProductId的转换器,但是如果我们的类型足够多...; } } 到这里,我们可以直接删除之前的 ProductIdConvert, 现在有一个通用的可以使用,现在.NET Core 的路由匹配已经没有问题了,接下来的文章,我会介绍如何处理在JSON

    1.9K20

    使用Janus作为对讲服务器的后台框架和业务流程

    对讲后台业务架构图如下图: 图1、对讲后台业务架构图 通过NGINX负载Janus http服务器的API接口,通过该API接口可以获取可用Janus服务器的IP和端口;客户端拿到可用对讲服务器的IP...客户端进入会议室对讲主要流程: 客户端加入会议室后,使用两个peerconnection分别用来做publisher角色和subscriber角色类型通信,publisher主要用来讲话的通道,subscriber...则主要用来听对讲的通道;每次讲话前都需要申请TBCP讲话权限,获取成功,则开启录音,并发送数据,否则不开启录音;如果其他人在讲话,则打开播放器,开始播放声音,主要业务流程如下图: 图2、对讲客户端主要业务...详细的客户端加入流程如下图3所示: 图3、对讲加入群组主要业务流程 退出会议室时,一定要保证两个Peerconnection的连接都能同服务器断开; 断网重连流程:如果碰到断网重连时,需要重新获取该房间的...janus服务器的IP和端口,并在重连WebSocket成功后,重走图3的流程,完成重新自动进入对讲房间的工作。

    1.3K10

    基于ABP落地领域驱动设计-02.聚合和聚合根的最佳实践和原则

    聚合根/实体中的主键 一个聚合根通常有一个ID属性作为其标识符(主键,Primark Key: PK)。推荐使用 Guid 作为聚合根实体的PK。 聚合中的实体(不是聚合根)可以使用复合主键。...这并不意味着子集合实体应该总是有复合主键,只有当需要时设置;通常是单一的ID属性。 复合主键实际上是关系型数据库的一个概念,因为子集合实体有自己的表,需要一个主键。...另一方面,例如:在MongoDB中,你根本不需要为子集合实体定义主键,因为它们是作为聚合根的一部分来存储的。 聚合根/实体构造函数 构造函数是实体的生命周期开始的地方。...实体中业务逻辑需要用到外部服务 当业务逻辑只使用该实体的属性时,在实体方法中实现业务规则是很简单的。如果业务逻辑需要查询数据库或使用任何应该从依赖注入系统中获取的外部服务时,该怎么办?...请记住,实体不能注入服务。 有两个方式实现: 在实体方法上实现业务逻辑,并将外部依赖项作为方法的参数。

    3.1K30

    人在数海漂,挨了“数据粒度”这一刀

    我将(日期,卖家ID,卖家名称)作为维度组合,来计算指标,同日出现两个数据。我心想这是数据粒度更细了,数据不是重复,我应该没有责任(其实我责无旁贷)。...代理键:不具有任何业务含义,仅做维度表数据唯一性区分的属性,通常以主键形式出现,比如自增的ID。...自然键:具有业务含义,是业务实体一个实例的唯一性区分,比如卖家ID、商品ID,在维度表中不一定做表的主键。...维度属性:描述同一业务实体各种特征的维度列,比如卖家名称、商品名称等 从维表的组成部分,我们很容易的看到维表的关键粒度,是自然主键,维度的主键ID一定是唯一(属性名称通常取最新的),维表中的维度属性在不同天是更新变化的...共享维度表内容:其他表加工过程中使用到维度属性内容,直接从维度表中获取,该实体的所有属性,均以维度表中属性为准,仅在维度表中进行维护,其他事实表/维度表中使用到维度表的指定属性,仅做内容共享。

    29510

    业务系统中自带的集成模块能否作为统一集成平台iPaaS使用?

    管理者摸不清门道,但实际情况并非如此简单,作为一名在iPaaS集成平台项目中深耕多年的实施专家,我想就"业务系统中自带的集成模块能否作为统一集成平台iPaaS使用"这个问题,谈谈我的看法和经验。...稳定性隐患: 集成平台的关键作用稳定性问题也是一大隐患。集成平台作为连接各个业务系统的"中枢神经",其稳定性直接关系到整个IT架构的健康运行。...我曾参与过一个项目,客户使用某WMS平台的内置集成功能来处理核心业务流程,结果在业务高峰期频繁出现数据不一致、处理延迟等问题,严重影响了日常运营。7....专业运维的重要性运维能力是另一个关键因素。iPaaS平台作为企业IT架构的核心组件,一旦出现问题可能会影响到所有相关的业务系统。...总的来说,虽然一些业务系统自带的集成模块可以解决一些简单的集成需求,但它们远远不能替代专业的iPaaS平台,把他作为业务系统与其他系统链接的中转站或自身的开放平台是非常合适的。

    13910

    SSH框架之Hibernate第二篇

    不用自己去写代码保证非空唯一. 1.2 Hibernate 主键生成策略: 1.2.1 区分自然主键和代理主键 自然主键 : 创建表的时候,使用的是实体中的自身属性作为表的主键....例如 : 创建一张人员表,可以用自生的身份证号(唯一的不可重复)作为主键. 代理主键 : 创建表的时候,使用的不是实体中的自身的属性作为主键,创建一个不相关的字段作为主键....例如 : 创建一张人员表,不使用自生的身份证号(唯一的不可重复的)作为主键,用另外创建一个id作为主键....尽量使用代理主键.如果选用自然主键,当主键需要参与到业务逻辑中,有可能需要改变很多的设计和代码. 1.2.2 的主键生成策略 主键不应该由用户自己维护,应该统一管理,那么Hibernate...(适用于short,int,long类型的主键) 首先使用 select max(cust_id) from cust_customer;将这个最大值 + 1 作为下一条记录的主键.

    52320

    高级框架-springDate-JPA 第二天【悟空教程】

    JPA 第二天 第1章 JPA 中的主键生成策略 通过annotation(注解)来映射实体类和数据库表的对应关系,基于annotation的主键标识为@Id注解, 其生成规则由@GeneratedValue...1.2.1 hibernate 中提供的主键生成规则 在讲解 Hibernate 的主键生成策略之前,先来了解两个概念,即自然主键和代理主键,具体如下: 自然主键: 把具有业务含义的字段作为主键...,称之为自然主键。...尽管这也是可行的,但是不能满足不断变化的业务需求,一旦出现了允许客户重名的业务需求,就必须修改数据模型,重新定义表的主键,这给数据库的维护增加了难度。...代理主键: 把不具备业务含义的字段作为主键,称之为代理主键。该字段一般取名为“ID”,通常为整数类型,因为整数类型比字符串类型要节省更多的数据库空间。在上面例子中,显然更合理的方式是使用代理主键。

    2.5K10
    领券