字符串生成伪唯一十六进制整数

本文介绍了一种解决广告EXCHANGE中全局唯一十六进制ID生成的问题。通过将两个数据库自增主键(UDID)拼接并进行MD5哈希处理,确保ID的唯一性和避免溢出问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近对接了让人头疼的广告EXCHANGE,需要为每个广告设置一个全局唯一的十六进制ID。初步设计是讲2个UDID(BIGINT20)拼接起来,但这样遇到2个问题:

  1. 唯一性:2个UDID都是数据库自增主键,类型为BIGINT20,拼接会导致非唯一性

  2. 溢出:对方使用long类型接受ID,会导致溢出

因此,要对拼接后的字符串做特殊处理,处理的大致方向是保证唯一性以及缩短字符串长度,大致方法是先做MD5,然后对拼接字符串做哈希:

public static long generateUUID(String id){

        //MD5 id
        MessageDigest md5 = null;
        try {
            md5 = MessageDigest.getInstance("MD5");
        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        }
        //AdContentId + schemaId 做MD5
        byte[] bytes = md5.digest(id.toString().getBytes());
        StringBuffer sb = new StringBuffer();
        for(byte b : bytes){
        //缩短字符串长度:对每一位取后4位,做16进制处理
            int val = (int)b & 0xf;
            sb.append(Integer.toHexString(val));
        }

        //Hash String to Long
        return SDBMHash(sb.toString().toCharArray());
    }

    private static long SDBMHash(char[] str)
    {
        long hash = 0L;
        for(char c : str)
            hash = (c) + (hash << 6) + (hash << 16) - hash;

        return (hash & Long.MAX_VALUE);
    }
<think>我们正在讨论C语言中将字符串转换为十进制整数的方法。用户特别提到了atoi和strtol函数,并希望看到示例代码。根据引用[1]和引用[2]的内容,strtol函数是一个常用的转换函数,它可以处理不同进制的转换(包括十进制),并且具有错误检查功能。另外,atoi函数也可以将字符串转换为整数,但它没有错误处理机制,且只能处理十进制。因此,我们可以提供两种方法的示例代码:1.使用atoi函数:简单但不够安全,无法检测溢出或无效输入。2.使用strtol函数:更健壮,可以检测溢出和无效输入。注意:用户要求的是字符串转十进制整数,所以我们在使用strtol时,基数(base)设置为10。我们将分别给出两个示例代码,并解释它们的使用和区别。同时,根据系统级指令,我们需要生成相关问题。</think>### C语言字符串转十进制整数示例 在C语言中,将字符串转换为十进制整数主要有两种常用方法:`atoi`函数和`strtol`函数。以下是详细说明和示例代码: #### 1. **`atoi`函数(简单转换)** ```c #include <stdlib.h> #include <stdio.h> int main() { char str[] = "12345"; int num = atoi(str); // 字符串转十进制整数 printf("转换结果: %d\n", num); // 输出: 12345 return 0; } ``` - **特点**: - 简单易用,无需错误处理 - 遇到非数字字符时自动停止转换 - **缺点**:无法检测溢出或无效输入(如"abc"会返回0) #### 2. **`strtol`函数(安全转换)** ```c #include <stdlib.h> #include <stdio.h> #include <errno.h> #include <limits.h> int main() { char str[] = "2147483647"; // INT_MAX值 char *endptr; long num = strtol(str, &endptr, 10); // 基数10表示十进制 // 错误检查 if (errno == ERANGE) { printf("数值超出范围!\n"); } else if (*endptr != '\0') { printf("无效字符: %s\n", endptr); } else { printf("转换结果: %ld\n", num); // 输出: 2147483647 } return 0; } ``` - **特点**: - 通过`endptr`检测非法字符(如"123abc"会标记"abc") - 通过`errno`检测溢出(如超过`LONG_MAX`) - 支持任意进制(16进制用`base=16`) - **推荐**:生产环境首选方法[^1][^2] #### 关键区别: | 特性 | `atoi` | `strtol` | |--------------|-----------------|-------------------| | 错误检测 | ❌ 无 | ✅ 溢出/非法字符 | | 返回值类型 | `int` | `long` | | 进制支持 | 仅十进制 | 任意进制(2-36) | | 安全性 | 低(无校验) | 高(完整校验) | > ⚠️ **重要提示**: > 处理用户输入时**务必使用`strtol`**,`atoi`在转换失败时不会报错(如"abc"→0),可能导致逻辑错误[^3]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值