【题目链接】

http://www.tsinsen.com/A1219

【题意】

给定一棵树,a[u][i]代表u结点分配i人的收益,可以随时改变a[u],查询(u,v)代表在u子树的所有节点,在u->v(不含u)路径上的节点分配人数的最优收益。

【思路】

树链剖分:构造重链时先访问重儿子,因此一个重链的区间连续,同时一个子树的区间连续。

查询分为两部分:构造在u子树内分配人数i的最大收益ans1[i],以及构造在u->v路径上一个结点分配人数i的最大收益ans2[i]。则ans=max{ ans1[i]+ans2[m-i] }。

考虑链剖:一棵线段树维护一条重链上的两类信息,c[i]为任意分配的最优,g[i]为在其中一个分配的最优。则ans1可以通过询问u子树的连续区间得,ans2可以询问u->v路径上的重链得。

需要注意的是线段树中c的递推应该逆序枚举,否则覆盖原值。

【代码】

 #include<set>
#include<cmath>
#include<queue>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define trav(u,i) for(int i=front[u];i;i=e[i].nxt)
#define FOR(a,b,c) for(int a=(b);a<=(c);a++)
using namespace std; typedef long long ll;
const int N = 4e4+;
const int M = ;
const ll X = <<;
const ll Y = ; ll read() {
char c=getchar();
ll f=,x=;
while(!isdigit(c)) {
if(c=='-') f=-; c=getchar();
}
while(isdigit(c))
x=x*+c-'',c=getchar();
return x*f;
} struct Edge {
int v,nxt;
}e[N<<];
int en=,front[N];
void adde(int u,int v)
{
e[++en]=(Edge){v,front[u]}; front[u]=en;
} int n,m,C,a[N][M],dfs_list[N],SZ,dfn; ll A,B,Q;
int dep[N],son[N],top[N],siz[N],fa[N],pl[N],L[N],R[N]; struct Tnode {
int u,l,r,c[M],g[M];
void maintain() ;
} T[N<<];
void Tnode::maintain() {
if(T[u].l==T[u].r) return ;
memset(c,,sizeof(c));
FOR(i,,m) FOR(j,,m-i)
c[i+j]=max(c[i+j],T[u<<].c[i]+T[u<<|].c[j]);
FOR(i,,m)
g[i]=max(T[u<<].g[i],T[u<<|].g[i]);
} void build(int u,int l,int r)
{
T[u].u=u,T[u].l=l,T[u].r=r;
if(l==r) {
memcpy(T[u].c,a[dfs_list[l]],sizeof(T[u].c));
memcpy(T[u].g,a[dfs_list[l]],sizeof(T[u].g));
return ;
}
int mid=l+r>>;
build(u<<,l,mid);
build(u<<|,mid+,r);
T[u].maintain();
}
void update(int u,int x,int* A)
{
if(T[u].l==T[u].r) {
memcpy(T[u].c,A,sizeof(T[u].c));
memcpy(T[u].g,A,sizeof(T[u].g));
} else {
int mid=T[u].l+T[u].r>>;
if(x<=mid) update(u<<,x,A);
else update(u<<|,x,A);
T[u].maintain();
}
}
void query1(int u,int L,int R,int* ans)
{
if(L<=T[u].l&&T[u].r<=R) {
for(int i=m;i;i--) for(int j=i;j;j--) //逆序枚举i 避免覆盖
ans[i]=max(ans[i],ans[i-j]+T[u].c[j]);
} else {
int mid=T[u].l+T[u].r>>;
if(L<=mid) query1(u<<,L,R,ans);
if(mid<R) query1(u<<|,L,R,ans);
}
}
void query2(int u,int L,int R,int* ans)
{
if(L<=T[u].l&&T[u].r<=R) {
FOR(i,,m) ans[i]=max(ans[i],T[u].g[i]);
} else {
int mid=T[u].l+T[u].r>>;
if(L<=mid) query2(u<<,L,R,ans);
if(mid<R) query2(u<<|,L,R,ans);
}
} void dfs1(int u)
{
siz[u]=; son[u]=;
trav(u,i) if(e[i].v!=fa[u]) {
int v=e[i].v;
fa[v]=u;
dep[v]=dep[u]+;
dfs1(v);
siz[u]+=siz[v];
if(siz[v]>siz[son[u]]) son[u]=v;
}
}
void dfs2(int u,int tp)
{
top[u]=tp; pl[u]=L[u]=++SZ; dfs_list[SZ]=u;
if(son[u]) dfs2(son[u],tp);
trav(u,i) if(e[i].v!=fa[u]&&e[i].v!=son[u])
dfs2(e[i].v,e[i].v);
R[u]=SZ;
}
void query2(int u,int v,int*ans)
{
while(top[u]!=top[v]) {
if(dep[top[u]]<dep[top[v]]) swap(u,v);
query2(,pl[top[u]],pl[u],ans);
u=fa[top[u]];
}
if(dep[u]>dep[v]) swap(u,v);
query2(,pl[u],pl[v],ans);
} int get_int()
{
A=((A^B)+(B/X)+(ll)(B*X))&Y;
B=((A^B)+(A/X)+(ll)(A*X))&Y;
return (int)(A^B)%Q;
} int ans1[M],ans2[M]; int main()
{
// freopen("in.in","r",stdin);
// freopen("out.out","w",stdout);
n=read(),m=read(),A=read(),B=read(),Q=read();
int op,u,v;
FOR(i,,n) {
u=read(); adde(u,i);
}
FOR(i,,n) {
FOR(j,,m) a[i][j]=get_int();
sort(a[i]+,a[i]+m+);
}
dfs1(); dfs2(,);
build(,,SZ);
C=read();
FOR(i,,C) {
op=read(),u=read();
if(op==) {
FOR(j,,m) a[u][j]=get_int();
sort(a[u]+,a[u]+m+);
update(,L[u],a[u]);
} else {
v=read();
memset(ans1,,sizeof(ans1));
memset(ans2,,sizeof(ans2));
query1(,L[u],R[u],ans1);
if(u!=v) query2(fa[u],v,ans2);
int ans=;
FOR(i,,m)
ans=max(ans,ans1[i]+ans2[m-i]);
printf("%d\n",ans);
}
}
return ;
}

P.S.拿线段树DP,这道题实在是太太太太太神辣

Tsinsen A1219. 采矿(陈许旻) (树链剖分,线段树 + DP)的更多相关文章

  1. 【BZOJ-2325】道馆之战 树链剖分 + 线段树

    2325: [ZJOI2011]道馆之战 Time Limit: 40 Sec  Memory Limit: 256 MBSubmit: 1153  Solved: 421[Submit][Statu ...

  2. 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树

    [BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...

  3. BZOJ2243 (树链剖分+线段树)

    Problem 染色(BZOJ2243) 题目大意 给定一颗树,每个节点上有一种颜色. 要求支持两种操作: 操作1:将a->b上所有点染成一种颜色. 操作2:询问a->b上的颜色段数量. ...

  4. POJ3237 (树链剖分+线段树)

    Problem Tree (POJ3237) 题目大意 给定一颗树,有边权. 要求支持三种操作: 操作一:更改某条边的权值. 操作二:将某条路径上的边权取反. 操作三:询问某条路径上的最大权值. 解题 ...

  5. bzoj4034 (树链剖分+线段树)

    Problem T2 (bzoj4034 HAOI2015) 题目大意 给定一颗树,1为根节点,要求支持三种操作. 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子 ...

  6. HDU4897 (树链剖分+线段树)

    Problem Little Devil I (HDU4897) 题目大意 给定一棵树,每条边的颜色为黑或白,起始时均为白. 支持3种操作: 操作1:将a->b的路径中的所有边的颜色翻转. 操作 ...

  7. Aizu 2450 Do use segment tree 树链剖分+线段树

    Do use segment tree Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.bnuoj.com/v3/problem_show ...

  8. 【POJ3237】Tree(树链剖分+线段树)

    Description You are given a tree with N nodes. The tree’s nodes are numbered 1 through N and its edg ...

  9. HDU 2460 Network(双连通+树链剖分+线段树)

    HDU 2460 Network 题目链接 题意:给定一个无向图,问每次增加一条边,问个图中还剩多少桥 思路:先双连通缩点,然后形成一棵树,每次增加一条边,相当于询问这两点路径上有多少条边,这个用树链 ...

  10. bzoj2243[SDOI2011]染色 树链剖分+线段树

    2243: [SDOI2011]染色 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 9012  Solved: 3375[Submit][Status ...

随机推荐

  1. Sina App Engine(SAE)入门教程(1)

    此教程只针对刚接触SAE的小白用户,资深码农.高手请绕道.首先还是一个经典的实例,hello sae. 创建应用 在注册完账号之后,需要到 http://sae.sina.com.cn/?m=myap ...

  2. mysql字符串区分大小写的问题

    一.1. CREATE TABLE NAME(name VARCHAR(10)); 对这个表,缺省情况下,下面两个查询的结果是一样的: SELECT * FROM TABLE NAME WHERE n ...

  3. PHP优化杂烩

    讲 PHP 优化的文章往往都是教大家如何编写高效的代码,本文打算从另一个角度来讨论问题,教大家如何配置高效的环境,如此同样能够达到优化的目的. pool 一个让人沮丧的消息是绝大多数 PHP 程序员都 ...

  4. android4.4.2内核移植3.4.1

    01◑ 内核源码总目录下Makefile文件修改: 195行: ARCH ?= $(SUBARCH)替换成: ARCH ?= arm 02◑ arch/arm下makefile: ①差7行 # tes ...

  5. struct inode 和 struct file

    1.struct inode──字符设备驱动相关的重要结构介绍 内核中用inode结构表示具体的文件,而用file结构表示打开的文件描述符.Linux2.6.27内核中,inode结构体具体定义如下: ...

  6. 在Ubuntu上为Android系统的Application Frameworks层增加硬件访问服务(老罗学习笔记5)

    在数字科技日新月异的今天,软件和硬件的完美结合,造就了智能移动设备的流行.今天大家对iOS和Android系统的趋之若鹜,一定程度上是由于这两个系统上有着丰富多彩的各种应用软件.因此,软件和硬件的关系 ...

  7. Burnside引理和polay计数学习小记

    在组合数学中有这样一类问题,比如用红蓝两种颜色对2*2的格子染色,旋转后相同的算作一种.有多少种不同的染色方案?我们列举出,那么一共有16种.但是我们发现,3,4,5,6是同一种,7,8,9,10是用 ...

  8. LA 3135 (优先队列) Argus

    将多个有序表合并成一个有序表就是多路归并问题,可用优先队列来解决. #include <cstdio> #include <queue> using namespace std ...

  9. POJ 3211 (分组01背包) Washing Clothes

    题意: 小明有一个贤妻良母型的女朋友,他们两个一起洗衣服. 有M种颜色的N件衣服,要求洗完一种颜色的衣服才能洗另外一种颜色. 两人可以同时洗,一件衣服只能被一个人洗. 给出洗每件衣服所用的时间,求两个 ...

  10. hdu 4635 Strongly connected(强连通)

    考强连通缩点,算模板题吧,比赛的时候又想多了,大概是不自信吧,才开始认真搞图论,把题目想复杂了. 题意就是给你任意图,保证是simple directed graph,问最多加多少条边能使图仍然是si ...