1
  • 我有一个原始字符串,它可以是 12 位字符串,也可以是 19 位客户 ID,它也只包含数字。这两种情况都是独一无二的。
  • 为了这个问题,让我们假设我们正在处理一个 12 位的字符串,如果您提出解决方案,那么它也必须适合 19 位的情况。
  • 我们的客户希望我们创建一个长度必须为 11 个字符的 SECOND 唯一字符串(无论原始字符串是 12 位还是 19 位);此唯一字符串/ID 应包含字母数字字符,其中“alpha”表示所有字母都可以是其中的一部分;即“参与”字符仅是数字+从“a”到“e”的小写字母的解决方案是不够的,因为它给我们提供的选项比使用所有小写和大写字母时更少。
  • 请注意,我不需要加密,但需要 UNIQUE 功能。
  • 我将不胜感激任何帮助,首选示例或示例链接。
  • 如果有人知道解决这种情况的第三方包,那么请留下一个链接。提前致谢!
4

2 回答 2

1

您可以尝试一种方法。

将您的 12 位数字转换为二进制表示,应该可以用 5 个字节表示。使用base64编码,应该可以用9个字母数字字符表示。(好的...... base64 确实包含几个非字母数字字符......:P)

(如果您难以将其设为 5 字节表示,则将 12 位数字分解为 3 组 4 位数字,每组 2 字节表示也应该可以)

搜索base64并了解一下,然后你可以用类似的方式实现自己的编码方法。


添加一些代码:(未经测试,只是给你一个想法)

String originalId= "123456789012";

String resultString = new String(Base64.encodeBase64(new BigInteger(originalId).toByteArray());
于 2013-02-28T07:51:17.430 回答
0

最初的答案建议使用基数为 36 的 BigInteger,但这对于 19 位数字来说是不够的。

我不知道是否有库可以转换为 base 62,但下面的人为示例让您了解如何做到这一点。输出是:

originalId = 999999999999999999
newId = bUI6zOLZTrh
retrieveOriginalId = 999999999999999999

使用 base 62 的基本原理如下:

  • 如果原始编号是唯一的,那么新编号也将是唯一的,因为它们实际上是相同的编号(即存在一对一的关系)
  • 您可以N ^ 11以 N 为底表示 11 个字符的数字(例如,以 10 为底,一个 11 位数字可以介于 0 和 10 ^ 11 或 1000 亿之间)
  • 最大的 19 位数字(以 10 为底)是 10 ^ 19 - 1
  • 当 N = 62 时,您有 62 ^ 11 = 5 * 10 ^ 19 种可能性,它大于 10 ^ 19,因此可以表示任何 19 位数字。实际上使用基数 54 就足够了。

示例代码(受 BigInteger 和 Long 类启发的算法 - 将添加异常处理):

class Base62 {

    private static final BigInteger RADIX = BigInteger.valueOf(62);
    private static final char[] DIGITS = {
        '0', '1', '2', '3', '4', '5',
        '6', '7', '8', '9', 'a', 'b',
        'c', 'd', 'e', 'f', 'g', 'h',
        'i', 'j', 'k', 'l', 'm', 'n',
        'o', 'p', 'q', 'r', 's', 't',
        'u', 'v', 'w', 'x', 'y', 'z',
        'A', 'B', 'C', 'D', 'E', 'F',
        'G', 'H', 'I', 'J', 'K', 'L',
        'M', 'N', 'O', 'P', 'Q', 'R',
        'S', 'T', 'U', 'V', 'W', 'X',
        'Y', 'Z'
    };

    public static void main(String[] args) throws IOException {
        String originalId = "999999999999999999";
        System.out.println("originalId = " + originalId);

        String newId = getBase62From10(originalId);
        System.out.println("newId = " + newId);

        String retrieveOriginalId = getBase10From62(newId);
        System.out.println("retrieveOriginalId = " + retrieveOriginalId);
    }

    /**
     *
     * @param number a positive number in base 10
     *
     * @return the same number, in base 62
     */
    public static String getBase62From10(String number) {
        char[] buf = new char[number.length()];
        int charPos = number.length() - 1;

        BigInteger i = new BigInteger(number);
        BigInteger radix = BigInteger.valueOf(62);

        while (i.compareTo(radix) >= 0) {
            buf[charPos--] = DIGITS[i.mod(radix).intValue()];
            i = i.divide(radix);
        }
        buf[charPos] = DIGITS[i.intValue()];

        return new String(buf, charPos, (number.length() - charPos));
    }

    /**
     *
     * @param number a positive number in base 62
     *
     * @return the same number, in base 10
     */
    public static String getBase10From62(String number) {
        BigInteger value = BigInteger.ZERO;
        for (char c : number.toCharArray()) {
            value = value.multiply(RADIX);
            if ('0' <= c && c <= '9') {
                value = value.add(BigInteger.valueOf(c - '0'));
            }
            if ('a' <= c && c <= 'z') {
                value = value.add(BigInteger.valueOf(c - 'a' + 10));
            }
            if ('A' <= c && c <= 'Z') {
                value = value.add(BigInteger.valueOf(c - 'A' + 36));
            }
        }
        return value.toString();
    }
}
于 2013-02-28T08:31:45.040 回答