- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
将任意长度和大小的数字列表压缩或编码为单个字母数字字符串的最佳方法是什么?
目标是能够将类似 1,5,8,3,20,212,42 的内容转换为类似 a8D1jN 的内容以在 URL 中使用,然后再转换回 1,5,8,3,20,212,42 .
对于生成的字符串,我可以使用任何数字和任何 ASCII 字母,小写和大写,所以:0-9a-zA-Z。我不希望有任何标点符号。
最佳答案
如果您将列表视为一个字符串,那么您有 11 个不同的字符需要编码(0-9 和逗号)。这可以用 4 位表示。如果您愿意添加,请说 $ 和 !到您的可接受字符列表,那么您将有 64 个不同的输出字符,因此能够对每个字符编码 6 位。
这意味着您可以将字符串映射到比原始字符串短约 30% 的编码字符串,并且相当模糊且看起来随机。
这样您就可以将数字系列 [1,5,8,3,20,212,42] 转码为字符串“gLQfoIcIeQqq”。
更新:我受到启发并为此解决方案编写了一个 python 解决方案(速度不快但功能足够......)
ZERO = ord('0')
OUTPUT_CHARACTERS = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789$!"
def encode(numberlist):
# convert to string -> '1,5,8,3,20,212,42'
s = str(numberlist).replace(' ','')[1:-1]
# convert to four bit values -> ['0010', '1011', '0110', ... ]
# (add 1 to avoid the '0000' series used for padding later)
four_bit_ints = [0 <= (ord(ch) - ZERO) <= 9 and (ord(ch) - ZERO) + 1 or 11 for ch in s]
four_bits = [bin(x).lstrip('-0b').zfill(4) for x in four_bit_ints]
# make binary string and pad with 0 to align to 6 -> '00101011011010111001101101...'
bin_str = "".join(four_bits)
bin_str = bin_str + '0' * (6 - len(bin_str) % 6)
# split to 6bit blocks and map those to ints
six_bits = [bin_str[x * 6 : x * 6 + 6] for x in range(0, len(bin_str) / 6)]
six_bit_ints = [int(x, 2) for x in six_bits]
# map the 6bit integers to characters
output = "".join([OUTPUT_CHARACTERS[x] for x in six_bit_ints])
return output
def decode(input_str):
# map the input string from characters to 6bit integers, and convert those to bitstrings
six_bit_ints = [OUTPUT_CHARACTERS.index(x) for x in input_str]
six_bits = [bin(x).lstrip('-0b').zfill(6) for x in six_bit_ints]
# join to a single binarystring
bin_str = "".join(six_bits)
# split to four bits groups, and convert those to integers
four_bits = [bin_str[x * 4 : x * 4 + 4] for x in range(0, len(bin_str) / 4)]
four_bit_ints = [int(x, 2) for x in four_bits]
# filter out 0 values (padding)
four_bit_ints = [x for x in four_bit_ints if x > 0]
# convert back to the original characters -> '1',',','5',',','8',',','3',',','2','0',',','2','1','2',',','4','2'
chars = [x < 11 and str(x - 1) or ',' for x in four_bit_ints]
# join, split on ',' convert to int
output = [int(x) for x in "".join(chars).split(',') if x]
return output
if __name__ == "__main__":
# test
for i in range(100):
numbers = range(i)
out = decode(encode(numbers))
assert out == numbers
# test with original series
numbers = [1,5,8,3,20,212,42]
encoded = encode(numbers)
print encoded # prints 'k2UBsZgZi7uW'
print decode(encoded) # prints [1, 5, 8, 3, 20, 212, 42]
关于algorithm - 将数字列表压缩或编码为单个字母数字字符串的最佳方法是什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3858245/
如何使用 SPListCollection.Add(String, String, String, String, Int32, String, SPListTemplate.QuickLaunchO
我刚刚开始使用 C++ 并且对 C# 有一些经验,所以我有一些一般的编程经验。然而,似乎我马上就被击落了。我试过在谷歌上寻找,以免浪费任何人的时间,但没有结果。 int main(int argc,
这个问题已经有答案了: In Java 8 how do I transform a Map to another Map using a lambda? (8 个回答) Convert a Map>
我正在使用 node + typescript 和集成的 swagger 进行 API 调用。我 Swagger 提出以下要求 http://localhost:3033/employees/sear
我是 C++ 容器模板的新手。我收集了一些记录。每条记录都有一个唯一的名称,以及一个字段/值对列表。将按名称访问记录。字段/值对的顺序很重要。因此我设计如下: typedef string
我需要这两种方法,但j2me没有,我找到了一个replaceall();但这是 replaceall(string,string,string); 第二个方法是SringBuffer但在j2me中它没
If string is an alias of String in the .net framework为什么会发生这种情况,我应该如何解释它: type JustAString = string
我有两个列表(或字符串):一个大,另一个小。 我想检查较大的(A)是否包含小的(B)。 我的期望如下: 案例 1. B 是 A 的子集 A = [1,2,3] B = [1,2] contains(A
我有一个似乎无法解决的小问题。 这里...我有一个像这样创建的输入... var input = $(''); 如果我这样做......一切都很好 $(this).append(input); 如果我
我有以下代码片段 string[] lines = objects.Split(new string[] { "\r\n", "\n" }, StringSplitOptions.No
这可能真的很简单,但我已经坚持了一段时间了。 我正在尝试输出一个字符串,然后输出一个带有两位小数的 double ,后跟另一个字符串,这是我的代码。 System.out.printf("成本:%.2
以下是 Cloud Firestore 列表查询中的示例之一 citiesRef.where("state", ">=", "CA").where("state", "= 字符串,我们在Stack O
我正在尝试检查一个字符串是否包含在另一个字符串中。后面的代码非常简单。我怎样才能在 jquery 中做到这一点? function deleteRow(locName, locID) { if
这个问题在这里已经有了答案: How to implement big int in C++ (14 个答案) 关闭 9 年前。 我有 2 个字符串,都只包含数字。这些数字大于 uint64_t 的
我有一个带有自定义转换器的 Dozer 映射: com.xyz.Customer com.xyz.CustomerDAO customerName
这个问题在这里已经有了答案: How do I compare strings in Java? (23 个回答) 关闭 6 年前。 我想了解字符串池的工作原理以及一个字符串等于另一个字符串的规则是
我已阅读 this问题和其他一些问题。但它们与我的问题有些无关 对于 UILabel 如果你不指定 ? 或 ! 你会得到这样的错误: @IBOutlet property has non-option
这两种方法中哪一种在理论上更快,为什么? (指向字符串的指针必须是常量。) destination[count] 和 *destination++ 之间的确切区别是什么? destination[co
This question already has answers here: Closed 11 years ago. Possible Duplicates: Is String.Format a
我有一个Stream一个文件的,现在我想将相同的单词组合成 Map这很重要,这个词在 Stream 中出现的频率. 我知道我必须使用 collect(Collectors.groupingBy(..)
我是一名优秀的程序员,十分优秀!