题目链接:http://codeforces.com/contest/832/problem/D

题意:给定一棵n个点的树,然后给你q个询问,每个询问为三元组(a,b,c),问你从这三个点中选取一个作为终点,一个作为Misha的起点,一个作为Grisha的起点。然后每天早上Misha从起点到终点所经过的点都是标记为1, 傍晚Grisha从起点到终点所经过的点中带有标记的点的数目最多是多少?

思路:对于每个询问,我们枚举终点(共3种情况),其余两个点作为一个作为M的起点一个作为G的起点,然后问题就是M的起点到终点这条路径的点赋值1,统计G的起点到终点这条路径的1的个数,然后3种情况取个最大值即可。 然后就是经典的树链剖分题目,树剖之后就是区间覆盖+区间查询问题了。 起初用的是线段树,然后终测TLE掉了(可能我写的线段树不够优美,被卡常了),后来换成树状数组来维护区间覆盖,区间查询就AC掉了。

#define _CRT_SECURE_NO_DEPRECATE
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<string>
#include<queue>
#include<vector>
#include<time.h>
#include<cmath>
using namespace std;
typedef long long int LL;
const int MAXN = 1e5 + ;
const int INF = 0x3f3f3f3f;
const int mod = 1e9 + ;
int fa[MAXN],top[MAXN],deep[MAXN],num[MAXN],p[MAXN],fp[MAXN],son[MAXN];
int pos,cp[MAXN],n,q;
LL bit0[MAXN],bit1[MAXN];
vector<int>edge[MAXN];
void init(){
pos = ;
memset(bit0,,sizeof(bit0));
memset(bit1,,sizeof(bit1));
memset(son, -, sizeof(son));
}
void dfs1(int u, int pre, int d){
deep[u] = d; fa[u] = pre; num[u] = ;
for (int i = ; i < edge[u].size(); i++){
int v = edge[u][i];
if (v != pre){
dfs1(v, u, d + );
num[u] += num[v];
if (son[u] == - || num[v] > num[son[u]]
){
son[u] = v;
}
}
}
}
void dfs2(int u, int sp){
top[u] = sp; p[u] = pos++; fp[p[u]] = u;
if (son[u] == -){
return;
}
dfs2(son[u], sp);
for (int i = ; i < edge[u].size(); i++){
int v = edge[u][i];
if (v != fa[u] && v != son[u]){
dfs2(v, v);
}
}
} //BIT
void Add(LL *b,int i,LL val){
while (i<=n){
b[i]+=val; i+=i&-i;
}
}
LL Sum(LL *b,int i){
LL s=;
while (i>){
s+=b[i]; i-=i&-i;
}
return s;
}
void Modify(int l,int r,int val){ //区间[l,r] + val
//printf("M:%d %d %d\n",l,r,val);
Add(bit0,l,-val*(l-));
Add(bit1,l,val);
Add(bit0,r+,val*r);
Add(bit1,r+,-val);
}
int Query(int l,int r){ //区间[l,r] 1 的个数
//printf("Q:%d %d\n",l,r);
LL res=;
res+=Sum(bit0,r)+1LL*Sum(bit1,r)*r;
res-=Sum(bit0,l-)+1LL*Sum(bit1,l-)*(l-);
return res;
}
void solveC(int u, int v,int val){ //修改链
int f1 = top[u], f2 = top[v];
while (f1!=f2){
if (deep[f1] < deep[f2]){
swap(f1, f2); swap(u, v);
}
Modify(p[f1], p[u], val);
u = fa[f1];
f1 = top[u];
}
if (deep[u] > deep[v]){
swap(u, v);
}
Modify(p[u], p[v], val);
}
int solveQ(int u, int v){ //查询链
int f1 = top[u], f2 = top[v];
int tmp = ;
while (f1 != f2){
if (deep[f1] < deep[f2]){
swap(f1, f2); swap(u, v);
}
tmp+=Query(p[f1], p[u]);
u = fa[f1]; f1 = top[u];
}
if (deep[u] > deep[v]){
swap(u, v);
}
tmp+=Query(p[u], p[v]);
return tmp;
}
int solve(int s, int t, int f){
solveC(s, f, );
int tmp = solveQ(t, f);
solveC(s, f, -);
return tmp;
}
int main(){
#ifdef kirito
freopen("in.txt", "r", stdin);
freopen("out.txt", "w", stdout);
#endif
while (~scanf("%d%d",&n,&q)){
init();
for (int i = ; i <= n; i++){
edge[i].clear();
}
for (int i = ; i <= n; i++){
scanf("%d", &cp[i]);
edge[cp[i]].push_back(i);
edge[i].push_back(cp[i]);
}
dfs1(, , );
dfs2(, );
for (int i = ; i <= q; i++){
int a, b, c,res=;
scanf("%d%d%d", &a, &b, &c);
res = max(res, solve(a, b, c));
res = max(res, solve(a, c, b));
res = max(res, solve(b, c, a));
printf("%d\n", res);
}
}
return ;
}

Codeforces Round #425 (Div. 2) - D的更多相关文章

  1. Codeforces Round #425 (Div. 2)C

    题目连接:http://codeforces.com/contest/832/problem/C C. Strange Radiation time limit per test 3 seconds ...

  2. Codeforces Round #425 (Div. 2)

    A 题意:给你n根棍子,两个人每次拿m根你,你先拿,如果该谁拿的时候棍子数<m,这人就输,对手就赢,问你第一个拿的人能赢吗 代码: #include<stdio.h>#define ...

  3. Codeforces Round #425 (Div. 2) Problem D Misha, Grisha and Underground (Codeforces 832D) - 树链剖分 - 树状数组

    Misha and Grisha are funny boys, so they like to use new underground. The underground has n stations ...

  4. Codeforces Round #425 (Div. 2) Problem C Strange Radiation (Codeforces 832C) - 二分答案 - 数论

    n people are standing on a coordinate axis in points with positive integer coordinates strictly less ...

  5. Codeforces Round #425 (Div. 2) Problem B Petya and Exam (Codeforces 832B) - 暴力

    It's hard times now. Today Petya needs to score 100 points on Informatics exam. The tasks seem easy ...

  6. Codeforces Round #425 (Div. 2) Problem A Sasha and Sticks (Codeforces 832A)

    It's one more school day now. Sasha doesn't like classes and is always bored at them. So, each day h ...

  7. Codeforces Round #425 (Div. 2) B. Petya and Exam(字符串模拟 水)

    题目链接:http://codeforces.com/contest/832/problem/B B. Petya and Exam time limit per test 2 seconds mem ...

  8. Codeforces Round #425 (Div. 2))——A题&&B题&&D题

    A. Sasha and Sticks 题目链接:http://codeforces.com/contest/832/problem/A 题目意思:n个棍,双方每次取k个,取得多次数的人获胜,Sash ...

  9. Codeforces Round #425 (Div. 2) B - Petya and Exam

    地址:http://codeforces.com/contest/832/problem/B 题目: B. Petya and Exam time limit per test 2 seconds m ...

  10. Codeforces Round #425 (Div. 2) C - Strange Radiation

    地址:http://codeforces.com/contest/832/problem/C 题目: C. Strange Radiation time limit per test 3 second ...

随机推荐

  1. 转载--C++的反思

    转载自http://blog.csdn.net/yapian8/article/details/46983319 最近两年 C++又有很多人出来追捧,并且追捧者充满了各种优越感,似乎不写 C++你就一 ...

  2. vue路由实例

    router.js: import Vue from 'vue' import Router from 'vue-router' import Home from '@/components/Home ...

  3. mysql分组,行转列

    aaarticlea/jpeg;base64,/9j/4QEsRXhpZgAASUkqAAgAAAAPAJqCCgABAAAAwgAAABABAgAQAAAAygAAAAABAwABAAAAQBAAA

  4. rich-text 副文本组件 text文本组件

    rich-text 副文本组件 要知道我们小程序常用的标签是view 但是我们想使用div   span  h1 i 标签等等,这种带特性的标签,怎么办的,我们就可以使用我们的 rich-text组件 ...

  5. [转]玩转Google开源C++单元测试框架Google Test系列(gtest)(总)

    文章转载自CoderZh的技术博客 地址:https://www.cnblogs.com/coderzh/archive/2009/04/06/1426755.html 前段时间学习和了解了下Goog ...

  6. ORACLE内存管理之ASMM AMM

    ORACLE ASMM ORACLE AMM ASMM转换至AMM AMM转换至ASMM

  7. 阶段1 语言基础+高级_1-3-Java语言高级_1-常用API_1_第1节 Scanner类_1-API概述和使用步骤

    官方翻译的中文版本

  8. 测开之路一百零三:jquery元素和标签的插入与删除

    标签内插入 标签外插入 给标签加标签 !DOCTYPE html><html lang="en"><head> <meta charset=&q ...

  9. Week 5 - 529.Minesweeper

    529.Minesweeper Let's play the minesweeper game (Wikipedia, online game)! You are given a 2D char ma ...

  10. idea创建maven中的 jar、war、 pom项目

    我用的是:2019.1版本的idea 创建maven项目时候的选项: 1: 2: 3:之后的步骤都一样按照自己的来就行