結果

問題 No.399 動的な領主
ユーザー roarisroaris
提出日時 2020-11-19 01:06:37
言語 PyPy3
(7.3.15)
結果
AC  
実行時間 1,511 ms / 2,000 ms
コード長 3,543 bytes
コンパイル時間 181 ms
コンパイル使用メモリ 82,256 KB
実行使用メモリ 105,884 KB
最終ジャッジ日時 2024-07-23 09:24:24
合計ジャッジ時間 14,298 ms
ジャッジサーバーID
(参考情報)
judge1 / judge4
このコードへのチャレンジ
(要ログイン)

テストケース

テストケース表示
入力 結果 実行時間
実行使用メモリ
testcase_00 AC 38 ms
53,416 KB
testcase_01 AC 40 ms
55,168 KB
testcase_02 AC 48 ms
61,960 KB
testcase_03 AC 49 ms
62,192 KB
testcase_04 AC 125 ms
78,564 KB
testcase_05 AC 231 ms
80,424 KB
testcase_06 AC 1,455 ms
99,036 KB
testcase_07 AC 1,445 ms
98,968 KB
testcase_08 AC 1,424 ms
99,300 KB
testcase_09 AC 1,483 ms
99,676 KB
testcase_10 AC 121 ms
78,176 KB
testcase_11 AC 185 ms
79,572 KB
testcase_12 AC 1,148 ms
100,772 KB
testcase_13 AC 1,098 ms
102,004 KB
testcase_14 AC 257 ms
105,884 KB
testcase_15 AC 291 ms
97,108 KB
testcase_16 AC 541 ms
99,852 KB
testcase_17 AC 1,511 ms
100,288 KB
testcase_18 AC 1,481 ms
99,512 KB
権限があれば一括ダウンロードができます

ソースコード

diff #

import sys
input = sys.stdin.readline
sys.setrecursionlimit(10**5+10)

class HLD:
    def __init__(self, N, G):
        self.G = G
        self.par = [-1]*N #親ノード
        self.size = [1]*N #部分木のサイズ
        self.depth = [-1]*N #深さ
        self.dfs1()
        self.in_order = [-1]*N #行きがけ順
        self.in_order_now = 0
        self.top = [-1]*N #分解後の連結成分の中で最も浅い頂点
        self.dfs2()
    
    def dfs1(self):
        stack = [(0, -1, 0)]
        
        while stack:
            v, pv, d = stack.pop()

            self.depth[v] = d
            
            for nv in self.G[v]:
                if nv==pv:
                    continue
                
                stack.append((nv, v, d+1))
                self.par[nv] = v
        
        vs = [(self.depth[i], i) for i in range(N)]
        vs.sort(reverse=True)   
        
        for _, v in vs:
            for nv in G[v]:
                if nv!=self.par[v]:
                    self.size[v] += self.size[nv]
    
    def dfs2(self):
        stack = [(0, -1, 0)]
        
        while stack:
            v, pv, top_node = stack.pop()
            self.in_order[v] = self.in_order_now
            self.in_order_now += 1
            self.top[v] = top_node
        
            if self.size[v]==1:
                continue
        
            M = 0
            heavy_node = -1
        
            for nv in self.G[v]:
                if nv==pv:
                    continue
            
                if self.size[nv]>M:
                    M = self.size[nv]
                    heavy_node = nv
            
            for nv in self.G[v]:
                if nv==pv:
                    continue
                
                if nv!=heavy_node:
                    stack.append((nv, v, nv))
                 
            stack.append((heavy_node, v, top_node))
    
    def query(self, u, v):
        res = []
        
        while self.top[u]!=self.top[v]:
            if self.depth[self.top[u]]<=self.depth[self.top[v]]:
                res.append((self.in_order[self.top[v]], self.in_order[v]))
                v = self.par[self.top[v]]
            else:
                res.append((self.in_order[self.top[u]], self.in_order[u]))
                u = self.par[self.top[u]]
        
        res.append((min(self.in_order[u], self.in_order[v]), max(self.in_order[u], self.in_order[v])))
        return res

class BIT:
    def __init__(self, n):
        self.n = n
        self.bit = [0]*(n+1)
    
    def add(self, i, x):
        i += 1
        
        while i<=self.n:
            self.bit[i] += x
            i += i&(-i)

    def acc(self, i):
        s = 0
        
        while i>0:
            s += self.bit[i]
            i -= i&(-i)
        
        return s

class RAQ_RSQ:
    def __init__(self, n):
        self.p = BIT(n)
        self.q = BIT(n)
    
    def add(self, s, t, x):
        self.p.add(s, -x*s)
        self.p.add(t, x*t)
        self.q.add(s, x)
        self.q.add(t, -x)
    
    def get_sum(self, s, t):
        return self.p.acc(t)+self.q.acc(t)*t-self.p.acc(s)-self.q.acc(s)*s

N = int(input())
G = [[] for _ in range(N)]

for _ in range(N-1):
    u, v = map(int, input().split())
    G[u-1].append(v-1)
    G[v-1].append(u-1)

hdl = HLD(N, G)
raq_rsq = RAQ_RSQ(N)
Q = int(input())
ans = 0

for _ in range(Q):
    A, B = map(int, input().split())
    
    for l, r in hdl.query(A-1, B-1):
        ans += raq_rsq.get_sum(l, r+1)+r-l+1
        raq_rsq.add(l, r+1, 1)

print(ans)
0