不使用[0..9]符号而是使用[0..9A..Z]符号
使用base-64系统
而不是使用base-10系统我想在这个例子中创建一个函数:
next('ABC')返回'ACA' - 这是下一个包含3个单位的字符串
就像我们有从0到9的数字,函数返回下一个数字
next2(135)返回136 - 这是下一个3位数的数字
我们使用基数为10的数字系统,我想使用数字表来表示基数为36的系统,并得到下一个所谓的数字
答案 0 :(得分:2)
这是一个在base-3字母数字系统中产生下一个值的函数:
function nextval($input, $pad = 1) {
$map = array(0 => 'A', 1 => 'B', 2 => 'C');
//convert letters to numbers
$num = '';
for ($i = 0; $i < strlen($input); $i++) {
$num .= array_search($input{$i}, $map);
}
//convert the number to base 10, then add 1 to it
$base10 = base_convert($num, 3, 10);
$base10++;
//convert back to base 3
$base3 = base_convert($base10, 10, 3);
//swap the digits back to letters
$num = '';
for ($i = 0; $i < strlen($base3); $i++) {
$num .= $map[$base3{$i}];
}
//pad with leading A's
while (strlen($num) < $pad) {
$num = 'A' . $num;
}
return $num;
}
echo nextval('ABC', 3); //ACA
请注意,结果为“CA”,因为“ACA”与在base-10中写入“06”相同...我们通常不会写前导零,因此您不会写出前导“A”。
因此我添加了一个pad参数,可让您指定要填充的位数。使用$pad=3
,您可以从“ABC”获得“ACA”。
答案 1 :(得分:1)
像这样的东西
<?php
function toNext($input) {
$conv = strtr(strtolower($input), array(
'a' => '0',
'b' => '1',
'c' => '2' ));
$conv = base_convert($conv, 3, 10);
$conv++;
$output = base_convert($conv, 10, 3);
$output = sprintf("%03d", $output);
$output = strtr((string) $output, array(
'0' => 'a',
'1' => 'b',
'2' => 'c' ));
return strtoupper($output);
}
var_dump(toNext('ABC'));
var_dump(toNext('ABA'));
答案 2 :(得分:0)
我现在想到的一种方法是将字符转换为base26,然后将数字加1并将其转换回来,我希望你能得到这个想法。 Next2应该做同样的但是使用base10这是默认的,所以我只是数字本身的+1。 期待看到其他实现。
编辑:没注意到你最后有A.傻到我这样做。那将是base3而不是26。
答案 3 :(得分:0)
这些手动基本转换功能不会受到内置转换功能的不准确性的影响。
<?php
function next($str)
{
$baseDec = array('0', '1', '2', '3', '4', '5', '6', '7', '8', '9');
$baseAln = array('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');
return base_conv((string) ((int) base_conv($str, $baseAln, $baseDec) + 1), $baseDec, $baseAln);
}
function base_conv_str($valStr, $baseToStr, $baseFromStr)
{
$baseTo = str_split($baseToStr);
$baseFrom = str_split($baseFromStr);
return base_arr_to_str(base_conv_arr(base_str_to_arr((string) $valStr, $baseFrom), count($baseTo), count($baseFrom)), $baseTo);
}
function base_conv($valStr, &$baseTo, &$baseFrom)
{
return base_arr_to_str(base_conv_arr(base_str_to_arr((string) $valStr, $baseFrom), count($baseTo), count($baseFrom)), $baseTo);
}
function base_conv_arr($val, $baseToDigits, $baseFromDigits)
{
$valDigits = count($val);
$result = array();
do
{
$divide = 0;
$newlen = 0;
for ($i = 0; $i < $valDigits; ++$i)
{
$divide = $divide * $baseFromDigits + $val[$i];
if ($divide >= $baseToDigits)
{
$val[$newlen ++] = (int) ($divide / $baseToDigits);
$divide = $divide % $baseToDigits;
}
else if ($newlen > 0)
{
$val[$newlen ++] = 0;
}
}
$valDigits = $newlen;
array_unshift($result, $divide);
}
while ($newlen != 0);
return $result;
}
function base_arr_to_str($arr, &$base)
{
$str = '';
foreach ($arr as $digit)
{
$str .= $base[$digit];
}
return $str;
}
function base_str_to_arr($str, &$base)
{
$arr = array();
while ($str === '0' || !empty($str))
{
foreach ($base as $index => $digit)
{
if (mb_substr($str, 0, $digitLen = mb_strlen($digit)) === $digit)
{
$arr[] = $index;
$str = mb_substr($str, $digitLen);
continue 2;
}
}
throw new Exception();
}
return $arr;
}
?>
答案 4 :(得分:0)
根据您对Xavier Barbosa的回答的评论,如果您想使用所有字母from a to z
,您可以这样做:
$str = 'ajz';
echo ++$str,"\n";
这将打印:aka
答案 5 :(得分:-2)
要获取base36中的下一个号码,请使用:
function next_base36($n) {
$n = base_convert($n, 36, 10);
return base_convert($n + 1, 10, 36);
}