BZOJ3531: [Sdoi2014]旅行

3531: [Sdoi2014]旅行

Time Limit: 20 Sec  Memory Limit: 512 MB
Submit: 323  Solved: 192
[Submit][Status]

Description

 S國有N個城市,編號從1到N。城市間用N-1條雙向道路鏈接,知足
從一個城市出發能夠到達其它全部城市。每一個城市信仰不一樣的宗教,如飛天麪條神教、隱形獨角獸教、絕地教都是常見的信仰。爲了方便,咱們用不一樣的正整數表明 各類宗教,  S國的居民經常旅行。旅行時他們總會走最短路,而且爲了不麻煩,只在信仰和他們相同的城市留宿。固然旅程的終點也是信仰與他相同的城市。S國政府爲每一個 城市標定了不一樣的旅行評級,旅行者們常會記下途中(包括起點和終點)留宿過的城市的評級總和或最大值。
    在S國的歷史上常會發生如下幾種事件:
」CC x c」:城市x的居民全體改信了c教;
」CW x w」:城市x的評級調整爲w;
」QS x y」:一位旅行者從城市x出發,到城市y,並記下了途中留宿過的城市的評級總和;
」QM x y」:一位旅行者從城市x出發,到城市y,並記下了途中留宿過
的城市的評級最大值。
    因爲年代久遠,旅行者記下的數字已經遺失了,但記錄開始以前每座城市的信仰與評級,還有事件記錄自己是無缺的。請根據這些信息,還原旅行者記下的數字。    爲了方便,咱們認爲事件之間的間隔足夠長,以至在任意一次旅行中,全部城市的評級和信仰保持不變。
php

Input

    輸入的第一行包含整數N,Q依次表示城市數和事件數。
    接下來N行,第i+l行兩個整數Wi,Ci依次表示記錄開始以前,城市i的
評級和信仰。
    接下來N-1行每行兩個整數x,y表示一條雙向道路。
    接下來Q行,每行一個操做,格式如上所述。
node

Output

    對每一個QS和QM事件,輸出一行,表示旅行者記下的數字。
ui

Sample Input

5 6
3 1
2 3
1 2
3 3
5 1
1 2
1 3
3 4
3 5
QS 1 5
CC 3 1
QS 1 5
CW 3 3
QS 1 5
QM 2 4

Sample Output

8
9
11
3

HINT

N,Q < =10^5    , C < =10^5 spa


 數據保證對全部QS和QM事件,起點和終點城市的信仰相同;在任意時

刻,城市的評級老是不大於10^4的正整數,且宗教值不大於C。
blog

Source

Round 1 Day 1事件

 

這題題意是每一個點有a,b兩個值, 詢問路徑x,y中a值跟x相同的b值的最大值或者和ip

樹鏈剖分,給每一個a值建一顆線段樹,而後動態開點,複雜度log^2(n)get

沒啥trickstring

/*
Author:wuhuajun
*/
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <algorithm>
#define lson l, mid, seg[rt].l
#define rson mid+1, r, seg[rt].r
using namespace std;
 
typedef long long ll;
typedef double dd;
const int maxn=200010;
const int maxq = 20000010;
 
int edge, n, fa[maxn], sz[maxn], son[maxn], dep[maxn], hash[maxn], top[maxn];
int h[maxn], num , x, y, tx, ty, Q, a[maxn], b[maxn], node, rel;
char s[22];
 
struct Edge
{
    int to, ne;
} e[maxn * 2];
 
struct Seg
{
    int maxx, sum, l, r;
    void clear()
    {
        maxx = -1000000000;
        sum = 0;
    }
} seg[maxq], ret, ans;
 
 
void close()
{
    exit(0);
}
 
void addedge(int x,int y)
{
    e[edge].to = y;
    e[edge].ne = h[x];
    h[x] = edge++;
}
 
void dfs(int k,int from)
{
    sz[k] = 1;
    son[k] = 0;
    dep[k] = dep[from] + 1;
    for (int p=h[k];p!=-1;p=e[p].ne)
    {
        int to = e[p].to;
        if (from == to) continue;
        fa[to] = k;
        dfs(to, k);
        sz[k] += sz[to];
        if (sz[to] > sz[son[k]]) son[k] = to;
    }
}
 
void build(int k,int from)
{
    hash[k] = ++num;
    top[k] = from;
    if (son[k]) build(son[k], from);
    for (int p=h[k];p!=-1;p=e[p].ne)
    {
        int to = e[p].to;
        if (to != fa[k] && to != son[k])
            build(to, to);
    }
}
 
//{{{Segment部分
void new_node(int rt)
{
    if (seg[rt].l == 0) seg[rt].l = ++node;
    if (seg[rt].r == 0) seg[rt].r = ++node;
}
 
Seg update(Seg a,Seg b)
{
    Seg c;
    c.sum = a.sum + b.sum;
    c.maxx = max(a.maxx, b.maxx);
    return c;
}
 
void pushup(int rt)
{
    seg[rt].maxx = max(seg[seg[rt].l].maxx, seg[seg[rt].r].maxx);
    seg[rt].sum = seg[seg[rt].l].sum + seg[seg[rt].r].sum;
}
 
void change(int L,int R,int val,int l,int r,int rt)
{
    new_node(rt);
    if (L <= l && r <= R)
    {
        seg[rt].sum = seg[rt].maxx = val;
        return;
    }
    int mid = (l + r) >> 1;
    if (L <= mid)
        change(L,R,val,lson);
    if (mid + 1 <= R)
        change(L,R,val,rson);
    pushup(rt);
}
 
Seg query(int L,int R,int l,int r,int rt)
{
    new_node(rt);
    if (L <= l && r <= R)
    {
        return seg[rt];
    }
    int mid = (l + r) >> 1;
    Seg ans;
    ans.clear();
    if (L <= mid)
        ans = update(ans, query(L,R,lson));
    if (mid + 1 <= R)
        ans = update(ans, query(L,R,rson));
    return ans;
}
 
//}}}
 
Seg get_ans()
{
    tx = top[x];
    ty = top[y];
    ans.clear();
    while (tx != ty)
    {
        if (dep[tx] < dep[ty])
        {
            swap(tx, ty);
            swap(x, y);
        }
        ans = update(ans, query(hash[tx], hash[x], 1, n, rel));
        x = fa[tx];
        tx = top[x];
    }
    if (dep[x] > dep[y]) swap(x, y);
    return update(ans, query(hash[x], hash[y], 1, n, rel));
}
 
void init()
{
    scanf("%d %d",&n, &Q);
    memset(h,-1,sizeof(h));
    for (int i=1;i<=n;i++)
    {
        scanf("%d %d",&b[i], &a[i]);
    }
    for (int i=1;i<=n-1;i++)
    {
        scanf("%d %d",&x, &y);
        addedge(x, y);
        addedge(y, x);
    }
    for (int i=1;i<=n;i++)
        scanf("%d", &a[i]);
    dfs(1, 0);
    build(1, 1);
    node = n;
    /*
       for (int i=1;i<=n;i++)
       {
       printf("i:%d top:%d hash:%d\n",i, top[i], hash[i]);
       }
       */
    for (int i=1;i<=n;i++)
        change(hash[i], hash[i], b[i], 1, n, a[i]); // a -> 宗教 b -> 評級
    while (Q--)
    {
        scanf("%s %d %d", s, &x, &y);
        if (s[1] == 'C')
        {
            change(hash[x], hash[x], 0, 1, n, a[x]); //把本屬於宗教a[i]的x點評級改爲0;
            a[x] = y;
            change(hash[x], hash[x], b[x], 1, n, a[x]); //把本屬於宗教a[i]的x點評級改爲現有的
        }
        if (s[1] == 'W')
        {
            b[x] = y;
            change(hash[x], hash[x], y, 1, n, a[x]);
        }
        if (s[1] == 'S')
        {
            rel = a[x];
            ret = get_ans();
            printf("%d\n", ret.sum);
        }
        if (s[1] == 'M')
        {
            rel = a[x];
            ret = get_ans();
            printf("%d\n", ret.maxx);
        }
    }
}
 
int main ()
{
    init();
    close();
    return 0;
}
相關文章
相關標籤/搜索