我将从一个例子开始;给定n = 1且m = 100且列表为[1,2,3]
生成包含1位和2位数的所有数字,依此类推但在这种情况下它们需要小于100.
生成包含1位和2位数的所有数字,依此类推但在这种情况下它们需要小于100.
输出:
– 1,3,11,12,13,21,22,23,31,32,33
然后我们停下来,因为下一个数字将超过100,例如:
– 111,112,113,121,122,123,131,132,133,21 ..,. 22 ……,23 ……,33
正如你所注意到我将1,4附加到之前创建的数字上,为了做到这一点,我使用了一个递归函数,它在我的列表中的每个数字的for循环中启动,它们运行直到生成数字大于我的限制.
def x(str,finish,d,c) return if d >= finish [1,4].each do |e| x(str,end,d*c+e) end # do something if d >= str end
如果我需要从1开始,这可以正常工作,但如果我的起始数字要大很多,我仍然需要开始创建这个序列.
有人可以帮我一个产生相同序列的解决方案,但是从任何起点而不是1,所以如果例如起点是100和结束200那么输出将是:
111,114,124,132 […]
任何编程语言的解决方案都不错,但请不要内置核心库.
解决方法
您可以从“右”(最低有效数字)到“左”(最高有效数字),继续跟踪两个值:
> min,与您到目前为止处理的数字位数相同的最小有效整数.因此,例如,如果您处理了尾随27,则最小有效两位数整数为11.
> max,与您到目前为止处理的数字位数相同的最小有效整数,大于或等于您目前处理的数字.因此,那么大于或等于27的最小有效两位数整数是31.
>请注意,max不会总是存在.例如,没有大于或等于70的有效两位数整数.
您需要min的原因是,如果您遇到的数字不在您允许的数字列表中,那么新的最大值将包含前一个min而不是之前的最大值. (例如,大于或等于02的最小有效两位数整数是11,而不是12.)
最后,如果存在,则返回max;否则,你返回min,但在你的列表前面加上最少的数字.
例如,如果this.allowedDigits是允许数字的集合,那么我们可以编写(在Java中):
private Integer getAllowedDigitGreaterThanOrEqualTo(final int digit) { for (int result = digit; result < 10; ++result) { if (this.allowedDigits.contains(result)) { return result; } } // digit is bigger than anything in the list: return null; } private int getAllowedNumberGreaterThanOrEqualTo(int n) { int minResult = 0; Integer maxResult = 0; int powerOfTen = 1; while (n > 0) { final int digit = n % 10; n /= 10; minResult = getAllowedDigitGreaterThanOrEqualTo(0) * powerOfTen + minResult; if (maxResult != null && this.allowedDigits.contains(digit)) { maxResult = digit * powerOfTen + maxResult; } else { final Integer newDigit = getAllowedDigitGreaterThanOrEqualTo(digit + 1); if (newDigit == null) { maxResult = null; } else { maxResult = newDigit * powerOfTen + minResult; } } powerOfTen *= 10; } if (maxResult == null) { return getAllowedDigitGreaterThanOrEqualTo(1) * powerOfTen + minResult; } else { return maxResult; } }