QOJ.ac

QOJ

ID题目提交者结果用时内存语言文件大小提交时间测评时间
#405483#7613. Inverse Problemucup-team3215TL 1ms3572kbC++202.1kb2024-05-06 00:41:282024-05-06 00:41:30

Judging History

你现在查看的是最新测评结果

  • [2024-05-06 00:41:30]
  • 评测
  • 测评结果:TL
  • 用时:1ms
  • 内存:3572kb
  • [2024-05-06 00:41:28]
  • 提交

answer

#include <algorithm>
#include <cstdint>
#include <iostream>
#include <numeric>
#include <unordered_map>
#include <vector>

using namespace std;

constexpr int mod = 1e9 + 7, cut = 6;

auto& mul(auto&& a, auto b) { return a = a * (uint64_t) b % mod; }
int inv(int a) { for (int r = 1, b = mod - 2; ; b /= 2, mul(a, a)) if (!b) return r; else if (b % 2) mul(r, a); }

using namespace std;

vector<unordered_map<int, vector<int>>> high(1);
vector<int> inv_(1);

void gen_high(int n0, int n, int c, int p, auto&& cur) {
  high[n0 - n].emplace(p, cur);
  for (; c > cut; --c) if (c <= n) {
    int np = p;
    for (int i = 0; i < c; ++i) mul(np, inv_[n0 - i]);
    cur.push_back(c);
    gen_high(n0, n - c, c, np, cur);
    cur.pop_back();
  }
}

vector<int> qs;
vector<vector<int>> ans;

void solve(int n0, int n, int c, int p, auto&& cur) {
  for (int i = 0; i < qs.size(); ++i) if (ans[i].empty() && high[n].count(mul(+p, qs[i]))) {
    ans[i] = cur;
    auto& t = high[n][mul(+p, qs[i])];
    ans[i].insert(ans[i].end(), t.begin(), t.end());
  }
  for (; c > 0; --c) if (c <= n) {
    int np = p;
    for (int i = 0; i < c; ++i) mul(np, n0 - i);
    cur.push_back(c);
    solve(n0, n - c, c, np, cur);
    cur.pop_back();
  }
}

int main() {
  int tc; cin >> tc;
  qs.resize(tc);
  ans.resize(tc);
  for (int i = 0; i < tc; ++i) {
    cin >> qs[i];
    if (qs[i] == 1) ans[i] = {-1};
    else if (qs[i] == 2) ans[i] = {-1};
    qs[i] = inv(qs[i]);
  }
  for (int n = 1, done = 0; !done++; ++n) {
    inv_.push_back(inv(n));
    high.push_back({});
    for (auto& x: high) x.clear(), x.max_load_factor(.2);
    gen_high(n, n, n, 1, vector<int>{});
    solve(n, n, cut, mul(n + 1, n + 2), vector<int>{});
    for (int i = 0; i < tc; ++i) done &= !ans[i].empty();
  }
  for (int i = 0; i < tc; ++i) {
    if (qs[i] == 1) cout << "1\n";
    else {
      if (ans[i][0] == -1) ans[i] = {};
      cout << accumulate(ans[i].begin(), ans[i].end(), 0) + 2 << '\n' << "1 2\n";
      for (int j = 0, k = 2; j < ans[i].size(); ++j)
      while (ans[i][j]--) cout << j + 1 << ' ' << ++k << '\n';
    }
  }
}

详细

Test #1:

score: 100
Accepted
time: 1ms
memory: 3572kb

input:

4
2
360
1
509949433

output:

2
1 2
5
1 2
1 3
1 4
2 5
1
10
1 2
1 3
2 4
3 5
4 6
5 7
6 8
7 9
8 10

result:

ok OK (4 test cases)

Test #2:

score: -100
Time Limit Exceeded

input:

9
185396120
468170792
837583517
696626231
338497514
762842660
800028852
928391161
733524004

output:

14
1 2
1 3
1 4
2 5
2 6
3 7
4 8
5 9
6 10
7 11
8 12
9 13
10 14
122
1 2
1 3
1 4
1 5
1 6
1 7
1 8
2 9
2 10
2 11
3 12
3 13
3 14
4 15
4 16
4 17
5 18
5 19
5 20
6 21
6 22
6 23
7 24
7 25
8 26
8 27
9 28
9 29
10 30
11 31
12 32
13 33
14 34
15 35
16 36
17 37
18 38
19 39
20 40
21 41
22 42
23 43
23 44
23 45
23 46
2...

result: