結果

問題 No.386 貪欲な領主
ユーザー しらっ亭
提出日時 2016-07-02 00:07:48
言語 C++14
(gcc 8.2.0)
結果
AC  
実行時間 175 ms
コード長 3,063 Byte
コンパイル時間 1,598 ms
使用メモリ 22,464 KB
最終ジャッジ日時 2019-08-22 19:14:51

テストケース

テストケース表示
入力 結果 実行時間
使用メモリ
test1.txt AC 3 ms
1,572 KB
test2.txt AC 3 ms
1,560 KB
test3.txt AC 3 ms
1,568 KB
test4.txt AC 2 ms
1,568 KB
test5.txt AC 175 ms
22,464 KB
test6.txt AC 137 ms
15,284 KB
test7.txt AC 137 ms
15,280 KB
test8.txt AC 4 ms
1,592 KB
test9.txt AC 18 ms
2,608 KB
test10.txt AC 5 ms
1,628 KB
test11.txt AC 3 ms
1,568 KB
test12.txt AC 2 ms
1,568 KB
test13.txt AC 3 ms
1,576 KB
test14.txt AC 7 ms
1,868 KB
test15.txt AC 138 ms
15,284 KB
test16.txt AC 152 ms
22,460 KB
テストケース一括ダウンロード

ソースコード

diff #
#include <bits/stdc++.h>
using namespace std;

#define _p(...) (void)printf(__VA_ARGS__)
#define forr(x,arr) for(auto&& x:arr)
#define _overload3(_1,_2,_3,name,...) name
#define _rep2(i,n) _rep3(i,0,n)
#define _rep3(i,a,b) for(int i=int(a);i<int(b);++i)
#define rep(...) _overload3(__VA_ARGS__,_rep3,_rep2,)(__VA_ARGS__)
#define _rrep2(i,n) _rrep3(i,0,n)
#define _rrep3(i,a,b) for(int i=int(b)-1;i>=int(a);i--)
#define rrep(...) _overload3(__VA_ARGS__,_rrep3,_rrep2,)(__VA_ARGS__)
#define ALL(x) (x).begin(), (x).end()
#define BIT(n) (1LL<<(n))
#define SZ(x) ((int)(x).size())
#define fst first
#define snd second
typedef vector<int> vi;typedef vector<vi> vvi;typedef pair<int,int> pii;typedef vector<pii> vpii;
typedef long long ll;

class LCA {
  public:
    int H, W;
    vector<vector<int> > parent;
    vector<int> depth;

    // LCA(木の頂点数)
    LCA(int V, vector<vector<int> >& E) : W(V), depth(V) {
      H = 1;
      while (1 << H < W) H++;
      parent.resize(H, vector<int>(W));
      // 適当な頂点からdfsして親を求めておく
      dfs(E, 0, -1, 0);
      // 親をダブリングする
      for (int k = 1; k < H; k++) {
        for (int v = 0; v < W; v++) {
          parent[k][v] = parent[k-1][v] == -1 ? -1 : parent[k-1][parent[k-1][v]];
        }
      }
    }

    void dfs(vector<vector<int> >& E, int cur, int par, int dep) {
      parent[0][cur] = par;
      depth[cur] = dep;
      for (int i = 0; i < (int) E[cur].size(); i++) {
        if (E[cur][i] != par) dfs(E, E[cur][i], cur, dep + 1);
      }
    }

    // u と v の LCA を求める
    int query(int u, int v) {
      // 深い方を v にしておく
      if (depth[u] > depth[v]) swap(u, v);
      // 根からの深さが u と同じになるまで v の親をたどる
      for (int k = 0; k < H; k++) {
        if ((depth[v]-depth[u]) >> k & 1) {
          v = parent[k][v];
        }
      }
      if (u == v) return u;
      // 二分探索でLCAを求める
      for (int k = H - 1; k >= 0; k--) {
        if (parent[k][u] != parent[k][v]) {
          u = parent[k][u];
          v = parent[k][v];
        }
      }
      return parent[0][u];
    }
};

vi U;
vvi E;
vi tax;

void dfs(int cur, int pre, int ct) {
  tax[cur] = ct + U[cur];
  forr(nex, E[cur]) {
    if (nex != pre) {
      dfs(nex, cur, ct + U[cur]);
    }
  }
}

void Main() {
  int N;
  scanf("%d", &N);
  E.resize(N);
  tax.resize(N);
  rep(i,N-1) {
    int a, b;
    scanf("%d%d", &a, &b);
    E[a].push_back(b);
    E[b].push_back(a);
  }

  U.resize(N);
  rep(i, N) scanf("%d", &U[i]);

  LCA lca(N, E);

  tax.resize(N);
  dfs(0, -1, 0);

  //rep(i,SZ(tax)) cout<<"tax["<<i<<"]: "<<tax[i]<<endl;

  ll ans = 0;

  int M;
  scanf("%d", &M);
  rep(i,M) {
    int a,b,c;
    scanf("%d%d%d", &a, &b, &c);
    int x = lca.query(a, b);

    ll tt = tax[a] + tax[b] - tax[x] * 2 + U[x];
    //_p("(%d,%d):%d, %lld(%d,%d,%d)\n", a,b,x,tt,tax[a],tax[b],tax[x]);

    ans += tt * c;
  }
  _p("%lld\n", ans);
}
int main() { cin.tie(0); ios::sync_with_stdio(false); Main(); return 0; }
0