0

[0..9] 記号を使用する代わりに、[0..9A..Z] 記号を使用します。

base-10 システムを使用する代わりに、base-64 システムを使用します。

この例のような関数を作りたい:

next('ABC') return 'ACA' - これは 3 単位の次の文字列です

0 から 9 までの数字があり、関数が次の数字を返すようなものです。

next2(135) return 136 - これは 3 桁の次の番号です

数字には10進法のシステムを使用し、36進法のシステムを意味するnumberlettersを使用して、次のいわゆる番号を取得します

4

6 に答える 6

2

次の関数は、基数 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

"ACA" は base-10 で "06" を書くのと同じなので、結果は "CA" になることに注意してください。

そのため、パディングする桁数を指定できる pad パラメータを追加しました。の$pad=3場合、「ABC」の次は「ACA」になります。

于 2011-02-05T09:24:44.153 に答える
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'));
于 2011-02-05T09:30:28.383 に答える
0

私が今考えることができる 1 つの方法は、文字を base26 に変換してから、数値に 1 を加えて元に戻すことです。Next2 も同じことを行う必要がありますが、base10 がデフォルトであるため、数値自体に +1 を加えるだけです。これに関する他の実装を楽しみにしています。

編集:最後に A があることに気づきませんでした。それをするのはばかげています。それは26ではなくbase3になります。

于 2011-02-05T09:19:07.963 に答える
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;
    }
?>
于 2011-02-05T09:32:14.693 に答える
0

Xavier Barbosa's answer へのコメントによると、すべての文字を使用したい場合は、次のfrom a to zことができます。

$str = 'ajz';
echo ++$str,"\n";

これは印刷されます:aka

于 2011-02-05T10:14:06.267 に答える
-2

base36 で次の番号を取得するには、次のようにします。

function next_base36($n) {
    $n = base_convert($n, 36, 10);
    return base_convert($n + 1, 10, 36);
}
于 2011-02-05T10:44:32.407 に答える