m 桁目を n より大きくして残りは0で埋める。下の桁で digit1, digit2 の帳尻を合わせる。
もう少しスッキリした実装ができても良かった気がする。
vector<lli> l2v(lli n)
{
vector<lli> v;
while (n) {
v.push_back(n % 10LL);
n /= 10LL;
}
reverse(v.begin(), v.end());
return v;
}
lli v2l(vector<lli> v)
{
lli n = 0;
for (int i = 0; i < (int)v.size(); ++i) {
n = n * 10LL + v[i];
}
return n;
}
map<lli, int> count_digit(vector<lli> v)
{
map<lli, int> cnt;
bool flg = false;
for (int i = 0; i < (int)v.size(); ++i) {
if (v[i]) flg = true;
if (flg) ++cnt[v[i]];
}
return cnt;
}
lli f(lli N, int d1, int c1, int d2, int c2)
{
vector<lli> v = l2v(N);
map<lli, int> cnt = count_digit(v);
if (c1 <= cnt[d1] && c2 <= cnt[d2]) return N;
return 1LL << 60;
}
lli g(lli N, int d1, int c1, int d2, int c2)
{
lli ret = 1LL << 60;
vector<lli> v = l2v(N);
while (v.size() < 16) v.insert(v.begin(), 0);
const int pref = v.size();
for (int i = 0; i < pref; ++i) {
for (int j = v[i]; j <= 9; ++j) {
vector<lli> u(v.size(), 0);
copy(v.begin(), v.begin() + i, u.begin());
u[i] = j;
map<lli, int> cnt = count_digit(u);
reverse(u.begin(), u.end());
int k = 0, l = 0;
for (; k < c2 - cnt[d2]; ++k) {
u[k] = d2;
}
for (; l < c1 - cnt[d1]; ++l) {
u[k + l] = d1;
}
reverse(u.begin(), u.end());
cnt = count_digit(u);
if (c1 <= cnt[d1] && c2 <= cnt[d2]) {
lli n = v2l(u);
if (N <= n) ret = min(ret, v2l(u));
}
}
}
return ret;
}
class FavouriteDigits {
public:
long long findNext(long long N, int d1, int c1, int d2, int c2)
{
if (d1 < d2) ; else {
swap(d1, d2);
swap(c1, c2);
}
lli a = f(N, d1, c1, d2, c2);
lli b = g(N, d1, c1, d2, c2);
return min(a, b);
}
};