QOJ.ac

QOJ

IDProblemSubmitterResultTimeMemoryLanguageFile sizeSubmit timeJudge time
#661187#8941. Even or Odd Spanning TreeDBsoleil#RE 0ms26144kbC++233.1kb2024-10-20 15:06:142024-10-20 15:06:14

Judging History

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

  • [2024-10-20 15:06:14]
  • 评测
  • 测评结果:RE
  • 用时:0ms
  • 内存:26144kb
  • [2024-10-20 15:06:14]
  • 提交

answer


#include <bits/stdc++.h>
using namespace std;
static constexpr int Maxn = 2e5 + 5, Maxm = 5e5 + 5, LogN = 18;
static constexpr int64_t inf = 0x3f3f3f3f3f3f3f3f;
int n, m, fa[Maxn], lowbit[Maxn];
int fnd(int x) { return x == fa[x] ? x : fa[x] = fnd(fa[x]); }
vector<pair<int, int64_t>> g[Maxn];
int64_t ans[2], cur;
struct edge {
  int u, v, w;
  edge() = default;
  friend bool operator < (const edge &lhs, const edge &rhs) {
    return lhs.w < rhs.w;
  }
} e[Maxm];
bool has[Maxm];
int top[Maxn], par[Maxn], dep[Maxn], hson[Maxn], siz[Maxn], dfn[Maxn], idfn[Maxn], dn;
int64_t s[2][LogN][Maxn], a[2][Maxn], b[2][Maxn];
void dfs1(int u, int fa, int depth) {
  par[u] = fa, siz[u] = 1, dep[u] = depth;
  for (const auto &[v, w]: g[u]) if (v != fa) {
    a[0][v] = (w % 2 == 1 ? -inf : w);
    a[1][v] = (w % 2 == 0 ? -inf : w);
    dfs1(v, u, depth + 1), siz[u] += siz[v];
    if (siz[v] > siz[hson[u]]) hson[u] = v;
  }
} // dfs
void dfs2(int u, int topu) {
  top[u] = topu;
  dfn[u] = ++dn; idfn[dn] = u;
  if (hson[u]) dfs2(hson[u], topu);
  for (const auto &[v, w]: g[u])
    if (v != par[u] && v != hson[u])
      dfs2(v, v);
} // dfs2
int64_t query(int k, int l, int r) {
  int j = lowbit[r - l + 1];
  return max(s[k][j][l], s[k][j][r - (1 << k) + 1]);
} // query
int64_t ask(int k, int u, int v) {
  int64_t ret = 0;
  while (top[u] != top[v]) {
    if (dep[top[u]] > dep[top[v]]) swap(u, v);
    ret = max(ret, query(k, dfn[top[v]], dfn[v]));
    v = par[top[v]];
  }
  if (dep[u] > dep[v]) swap(u, v);
  ret = max(ret, query(k, dfn[u], dfn[v]));
  return ret;
} // ask
int main(void) {
  cin.tie(nullptr)->sync_with_stdio(false);
  int tests; cin >> tests;
  while (tests--) {
    cin >> n >> m;
    ans[0] = ans[1] = inf; cur = 0; dn = 0;
    for (int i = 1; i <= n; ++i) g[i].clear();
    for (int i = 1; i <= m; ++i) has[i] = false;
    for (int i = 2; i <= n; ++i) lowbit[i] = lowbit[i >> 1] + 1;
    for (int i = 1; i <= m; ++i)
      cin >> e[i].u >> e[i].v >> e[i].w;
    sort(e + 1, e + m + 1);
    for (int i = 1; i <= n; ++i) fa[i] = i;
    for (int i = 1; i <= m; ++i) {
      int u = e[i].u, v = e[i].v, w = e[i].w;
      int fu = fnd(u), fv = fnd(v);
      if (fu == fv) continue; has[i] = true;
      fa[fu] = fv;
      g[u].emplace_back(v, w);
      g[v].emplace_back(u, w);
      cur += w;
    } ans[cur & 1] = cur;
    bool uncon = false;
    for (int i = 1; i <= n; ++i) if (fnd(i) != fnd(1)) uncon = true;
    if (uncon) { cout << "-1 -1\n"; continue; }
    dfs1(1, 0, 1), dfs2(1, 1);
    for (int k = 0; k < 2; ++k) {
      for (int i = 1; i <= n; ++i) s[k][0][i] = b[k][i] = a[k][idfn[i]];
      for (int j = 1; j < LogN; ++j)
        for (int i = 1; i + (1 << j) - 1 <= n; ++i)
          s[k][j][i] = max(s[k][j - 1][i], s[k][j - 1][i + (1 << j - 1)]);
    }
    int64_t ruc = inf;
    for (int i = 1; i <= m; ++i) if (!has[i]) {
      int u = e[i].u, v = e[i].v, w = e[i].w;
      ruc = min(ruc, w - ask(w & 1 ^ 1, u, v));
    }
    ans[cur & 1 ^ 1] = cur + ruc;
    cout << (ans[0] >= inf ? -1 : ans[0]) << ' ' << (ans[1] >= inf ? -1 : ans[1]) << endl;
  }
  return 0;
} // main

Details

Tip: Click on the bar to expand more detailed information

Test #1:

score: 100
Accepted
time: 0ms
memory: 26144kb

input:

3
2 1
1 2 5
3 1
1 3 1
4 4
1 2 1
1 3 1
1 4 1
2 4 2

output:

-1 5
-1 -1
4 3

result:

ok 6 numbers

Test #2:

score: -100
Runtime Error

input:

10000
16 50
1 2 649251632
2 3 605963017
3 4 897721528
4 5 82446151
5 6 917109168
6 7 79647183
7 8 278566178
7 9 573504723
3 10 679859251
8 11 563113083
1 12 843328546
10 13 594049160
11 14 997882839
7 15 569431750
2 16 244494799
16 7 960172373
13 4 317888322
13 3 446883598
9 3 678142319
5 8 43208692...

output:

3140067932 3038805477

result: