あなたの問題はプログラミングの問題ではなく、数学の問題です。
Let m = 1+E(log10(N))
、つまり の 10 進数表記の桁数N
(対数を使用するよりも桁数を数えて計算する方がおそらく高速です)。
の数mK
を と7
しN
ます。
を出力N'
数とします。
私は4つのケースを見ます:
K >= m
: 次にN' = 7..7
(K
桁)。
K == mK
:その後N' = N
。
K > mK and K < m
: 次に、最下位桁から順に、すべての非7
数字をに置き換えます。7
例: N = 1 357 975 , K = 4 => N' = 1 357 777
. 警告 : がある場合は特殊なケースがあります。8
例: N = 80, N' = 79
. この場合、共通の接頭辞を使用してから、すべての7
接尾辞を生成します (特殊なケース: 接頭辞からもう 1 つ削除して を追加します7 9 7 7 ... 7
)。special case
コードで参照してください。
K < mK
: 可能な数は 2 つあります。
分解しましょうN
: N = a1 a2 ... ap 7 b1 b2 ... bq
, where
a1 ... ap
とはp
数字です[0..9]
b1 ... bq
のq
数字です[0..9] \ {7}
and (または の後の数字)A = a1 ... ap 6 9 ... 9
とします。次に、. 両方の数値が等しく近い場合、選択はあなた次第です。B = a1 ... ap 8 0 ... 0
q
6
8
N' = closestToN(A,B)
数学の書式設定が悪くてすみません。
コードをより簡単に記述できるようになりました。ここに私の実装があります:
#include <iostream>
unsigned long long getClosestWith7(unsigned long long n, unsigned int k)
{
// Count number of digits
unsigned long long tmp = n;
unsigned int m = 0, mK = 0;
while(tmp > 0)
{
if(tmp % 10 == 7) mK++;
tmp /= 10;
m++;
}
// Distinct cases
if(k == mK && n != 0)
return n;
else if(k >= m || n == 0) // implicit: k != mK
{
unsigned long long r = 0;
while(k > 0)
{
r = 10 * r + 7;
k--;
}
return r;
}
else if(k > mK) // implicit: k != mK, k < m
{
unsigned long long r = n;
unsigned long long s = 0;
m = 0;
while(mK < k)
{
if(r % 10 != 7) mK++;
r /= 10;
m++;
}
if(r % 10 == 8) // special case
s = 79 + 100 * (r / 10);
while(m > 0)
{
r = 10 * r + 7;
if(s != 0 && m > 1) // special case
s = 10 * s + 7;
m--;
}
return (r < n && n - r < n - s) || (r >= n && r - n < n - s) ? r : s;
}
else // implicit : k < mK
{
// Generate a and b
unsigned long long a = n;
unsigned long long b = 0;
m = 0;
while(mK > k)
{
if(a % 10 == 7) mK--;
a /= 10;
m++;
}
b = 10 * a + 8;
a = 10 * a + 6;
m--;
while(m > 0)
{
a = 10 * a + 9;
b = 10 * b + 0;
m--;
}
// Compare (return lowest if equal)
return n - a <= b - n ? a : b;
}
}
#define CLOSEST7( N , K ) \
std::cout << "N = " << N << ", K = " << K << " => N' = " << getClosestWith7(N,K) << "\n"
int main()
{
CLOSEST7(1773,3);
CLOSEST7(83,1);
CLOSEST7(17273,3);
CLOSEST7(1273679750,6);
CLOSEST7(1773,1);
CLOSEST7(83,5);
CLOSEST7(0,2);
CLOSEST7(0,0);
}
についての質問についてlong long
は、コンパイラに依存します。多くの場合、この型のサイズは 64 ビットであるため、0 から 2^64 - 1 (符号なし) までの数値を格納できます。これは 18 446 744 073 709 551 615 であり、ほとんどの実装ではデータ範囲で問題ありません。 .