有一个需求是不用数字作为用户 ID,要用英文字母
我现在有三种解决方案
第一种:用户注册时候,生成一个随机英文字符串,去查用户表,如果重复再重新生成,直到唯一。
第二种:先生成一张字母 ID 表,每次用户从里面取,取了就标记为已用。
第三种:用函数把用户数字 ID 转为字母 ID,但是转出来的 ID,两个相近的用户,字母 ID 太相似了。
有什么更好的解决方案吗?
![]() | 1 Varobjs 2020-01-22 10:57:12 +08:00 via Android 就是随机生成不唯一的用户名呗 |
2 lg106 OP @Varobjs 也可以这么说,用户 ID 要显示在外面,我们不想让别人看到我们的数字 ID,所以想用英文来代替,包括我们的文章 ID 也是要弄成英文 ID |
![]() | 3 imlinhanchao 2020-01-22 11:00:48 +08:00 @lg106 uuid 可否? |
![]() | 4 whypool 2020-01-22 11:01:07 +08:00 新增数据可以用 UUID,把横杠去掉就行,就是有点长 用户名+时间戳+MD5 也能生成唯一 ID crypto 库也能生成唯一 ID |
![]() | 5 jfcherng 2020-01-22 11:01:56 +08:00 via Android hashids |
![]() | 6 jinhan13789991 2020-01-22 11:03:25 +08:00 via Android 第一个人是 AAAAAAAA,第二个人是 AAAAAAAB,以此类推~ 变相的 26 进制自增 |
![]() | 7 eojessie 2020-01-22 11:04:46 +08:00 @jinhan13789991 这个容易被枚举了。。。。 |
8 lg106 OP @imlinhanchao 可能太长了 |
![]() | 9 otakustay 2020-01-22 11:05:55 +08:00 就用自增键然后 hash 一下呢 |
10 lg106 OP @jinhan13789991 我第三种就是这种,可以用函数转化,但容易被找规律 |
15 GM 2020-01-22 11:10:45 +08:00 ![]() HashID 了解一下,数据表可以继续用自增 id 字段,对外做个转换变成 HashID 就好了。 |
![]() | 16 009694 2020-01-22 11:14:45 +08:00 via iPhone 前两种方案有什么缺点吗? |
![]() | 17 Marmot 2020-01-22 11:17:01 +08:00 ![]() hashids 靠谱 |
18 cgpiao 2020-01-22 11:17:40 +08:00 via iPhone 设备 id + 自定义格式时间 + 自增 转换为 36 进制 |
19 lg106 OP 试了下 hashids,这个完美解决,感谢大佬们 |
![]() | 20 eason1874 2020-01-22 11:25:52 +08:00 怕 ID 暴露可以用 36 进制,36 个字符打乱顺序就不好猜了,但如果获得连续 ID 还是可以破解出来。 |
![]() | 21 vanishcode 2020-01-22 11:33:42 +08:00 https://github.com/souyunku/SnowFlake 不知道是不是楼主想要的。。 |
![]() | 22 xaplux 2020-01-22 11:37:31 +08:00 @vanishcode 很明显不是,楼主想要的是全字母的,17 楼说的 hashids 靠谱 |
![]() | 23 tabris17 2020-01-22 11:50:06 +08:00 snowflake 算法生成 64 位整数 然后转换成 base53 字符串(仅包含字母和下划线) |
24 hubqin 2020-01-22 11:52:24 +08:00 取巧,Javascript:Math.random().toString(36).slice(2) |
25 wengcd 2020-01-22 11:55:45 +08:00 |
26 alaikis 2020-01-22 11:58:50 +08:00 数字 36 进制 |
![]() | 27 MrYELiex 2020-01-22 13:00:43 +08:00 snowflake |
28 Raymon111111 2020-01-22 13:10:39 +08:00 snowflake 这种方案然后把 0 到 9 映射到 a - j 上就行了. 一个简单的实现是, 当前时间(unixtime, 秒和毫秒都可以) + 机器码(比如集群是 100 个机器, 那就号码就是 00 - 99) + 三位轮询的数(每个机器启动时候就拿到一个打乱的大小是 1000 的数组, 里面的数是 000 - 999, 生成的时候从里面取数然后移除, 空了再生成一次) 这么干冲突的概率相当小(几乎不会有) |
![]() | 29 looplj 2020-01-22 14:37:27 +08:00 uuid->hash->baese64 取前 10 位 |
30 sleepm 2020-01-22 14:43:25 +08:00 有个自增主键 id,然后,先插入新纪录,然后获取 id,再根据 id 生成全字母的 ID |
![]() | 31 luopengfei14 2020-01-22 14:44:56 +08:00 via iPhone 36、62 进制都可以 |
![]() | 32 fx 2020-01-22 15:33:21 +08:00 hashids |
![]() | 33 jeremaihloo 2020-01-22 16:14:54 +08:00 |
34 qsbaq 2020-01-22 16:29:22 +08:00 把 ID 弄个 md5 肯定唯一了 |
35 songco 2020-01-22 16:44:02 +08:00 没有可读性要求可以直接 id 映射一下就行, 比如数字转换成 16 进制; 比如 0-9 映射成 10 个字幕 有可读性要求的话, 准备字典, 然后随机生成两到组拼起来, 预先生成也行; 生成后查重复也行; 直接按顺序映射也可以, 生成的结果大概类似 docker 的默认名字, 比如 adoring_lovelace 之类的 |
![]() | 37 wzwwzw 2020-01-22 18:50:25 +08:00 uuid 去掉 - 呗。 |
![]() | 38 kkkkkrua 2020-01-22 19:18:38 +08:00 via iPhone 将数字转成 58 进制 |
39 hauzi 2020-01-22 20:11:54 +08:00 via iPhone 第一种 |
![]() | 40 fireapp 2020-01-22 20:13:47 +08:00 via iPhone mongo object id 还不错可以试试 |
![]() | 41 lasuar 2020-01-22 20:18:44 +08:00 利用自增思想,A->B 等效于 1->2 |
42 fdingiit 2020-01-22 20:22:46 +08:00 有请求,再创建不是一个好的生产环境策略。 我们的生产环境上,id 是有个预资源池,初始可能是 n 位,随用随取。如果用完了,就 stop the world 并再生成一个位数为 n+1 的 id 新池子。 这个思路的来源是编译器中内存分配以及垃圾回收的几个简单算法。 |
![]() | 43 gamexg 2020-01-22 21:02:00 +08:00 hashid 挺好, 我做过类似的另一个需求, 不过原始 id 是二进制数据,直接固定 iv 的 aes 流加密来做的。 |
![]() | 44 rogwan 2020-01-22 21:07:26 +08:00 hashids + 盐。 |
![]() | 45 LancerEvo 2020-01-22 21:28:07 +08:00 记得在某个公司的代码里见过一个微服务 用的第二种实现 |
![]() | 46 zyqhi 2020-01-22 21:43:43 +08:00 via iPhone 26 进制 |
![]() | 47 ech0x 2020-01-22 21:53:22 +08:00 生成一个 UUID 不就行了,UUID3 或者 UUID5 |
![]() | 48 wd 2020-01-22 22:17:07 +08:00 via iPhone 我觉得第一种就挺好的 uuid 里面取几位用 不会那么大概率需要生成第二次 |
![]() | 49 zpvip 2020-01-22 22:42:41 +08:00 via Android 我不知道为什么 Ruby on Rails 在国内这么不受待见。这个需求一行代码就搞定了 gem 'friendly_id' |
50 lookas2001 2020-01-23 00:34:20 +08:00 via Android 8 位 id 取 5 次都有重复的可能性基本上就为 0 了 所以,第一种 |
51 dandandanerdan 2020-01-23 06:57:56 +08:00 uuid 是最安全的 |
52 polymerdg 2020-01-23 09:05:53 +08:00 我用的是 MD5(用名+戳+ 6 位) |
![]() | 53 huzb 2020-01-24 22:45:24 +08:00 用函数把用户 ID 转成字母 ID 是最好的,可以确保唯一且在前端就完成校验。相近 ID 这个可以用混淆和扩散的方式把变化打乱。我有总结过一篇文章:huzb.me/2018/03/23/简单的密码学生成唯一邀请码 / |