【题目链接】 http://acm.hdu.edu.cn/showproblem.php?pid=5735

【题目大意】

  给出一棵树,树上每个节点都有一个权值w,w不超过216,树的根为1,从一个点往根的方向走,可以得到他的祖先序列,现在需要从v1点的祖先序列中挑选出一定数量的点,组成数列v1,v2,v3……vm,要求vi是vi-1的祖先,求dp[v1]=max(dp[vi]+(w[v1] opt w[vi])),opt是一种运算,在题目中可为xor,or或者and,最后求出ans=sum_{i=1}^{n}(i*(w[i]+dp[i]))

【题解】

  对于这道题,我们首先考虑它的简化版本,在一个一维数组上求dp[i]=max(dp[j]+(w[i] opt w[j])) (j<i),显然枚举前缀的O(n2)的用脚趾头都能想出来的算法,出题人是不会给过的。那么我们观察一下题目,发现一个很奇巧的东西,w的值不超过216,难道说每次计算以w结尾的dp最大值,然后枚举二进制?一次6w多的计算量,明显也没有产生太大的优化,顺着这个思路下去,这道题采用了一种拆值DP的神奇的方式,

  dp[i]=max(dp[j]+([w[i]前八位]opt[w[j]前八位])<<8+[w[i]后八位]opt[w[j]后八位])

  记dp[A][B]=以前八位为A结尾,后八位以B结尾的dp值,于是就可以发现:

dp[A][B]=max(dp[i][B]+([w[i]前八位]opt[w[A]前八位])<<8)

  那么,在知道了后八位的情况下,前八位就能轻松dp,既然这样,那我们就在计算完每个节点之后,预处理后八位的dp值:

dp[A][i]=max(dp[A][j]+([w[i]后八位]opt[w[j]后八位]))

  这样子每次转移所需要的复杂度就只有28,可以接受。顺利完成。

  而这道题所处理的却是树上的问题,那么在每条链上DP的过程中预处理祖先节点dp数组,按照上述方法计算子节点的dp值即可,而对于不同的子节点,dp数组备份,然后回溯即可。

【代码】

#include <cstdio>
#include <vector>
#include <algorithm>
using namespace std;
typedef unsigned int UI;
const int N=65540,mod=1e9+7;
UI T,n,i,w[N],nxt[N],x,f[256][256],tmp[N][256],v[256],ans;
vector<UI>g[N];
char op[5];
UI opt(UI a,UI b){
if(op[0]=='A')return a&b;
if(op[0]=='O')return a|b;
if(op[0]=='X')return a^b;
}
void dfs(UI x){
UI dp=0,A=w[x]>>8,B=w[x]&255;
for(int i=0;i<256;i++)if(v[i])dp=max(dp,f[i][B]+(opt(A,i)<<8));
ans=(1LL*x*(dp+w[x])+ans)%mod;
for(v[A]++,i=0;i<256;i++)tmp[x][i]=f[A][i],f[A][i]=max(f[A][i],opt(B,i)+dp);
for(int i=0;i<g[x].size();i++)dfs(g[x][i]);
for(v[A]--,i=0;i<256;i++)f[A][i]=tmp[x][i];
}
int main(){
scanf("%d",&T);
while(T--){
scanf("%d %s",&n,op);
for(int i=1;i<=n;i++)scanf("%d",&w[i]),g[i].clear();
for(int i=2;i<=n;i++)scanf("%d",&x),g[x].push_back(i);
ans=0; dfs(1);
printf("%d\n",ans);
}return 0;
}

  

HDU 5735 Born Slippy(拆值DP+位运算)的更多相关文章

  1. hdu 5735 Born Slippy 暴力

    Born Slippy 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5735 Description Professor Zhang has a r ...

  2. HDU 5735 - Born Slippy

    题意: 一棵 n 个节点的根树,i 节点权重 wi 对每一个节点s,找到这样一个长 m 的标号序列 v : 1. vi是vi-1 的祖先 2. f[s] = w[vi] + ∑(i=2, m) (w[ ...

  3. hdu 4336 Card Collector (概率dp+位运算 求期望)

    题目链接 Card Collector Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Othe ...

  4. 求集合中选一个数与当前值进行位运算的max

    求集合中选一个数与当前值进行位运算的max 这是一个听来的神仙东西. 先确定一下值域把,大概\(2^{16}\),再大点也可以,但是这里就只是写写,所以无所谓啦. 我们先看看如果暴力求怎么做,位运算需 ...

  5. 基于DP+位运算的RMQ算法

    来源:http://blog.csdn.net/y990041769/article/details/38405063 RMQ算法,是一个快速求区间最值的离线算法,预处理时间复杂度O(n*log(n) ...

  6. HDU 1074 Doing Homework (动态规划,位运算)

    HDU 1074 Doing Homework (动态规划,位运算) Description Ignatius has just come back school from the 30th ACM/ ...

  7. HDU 3006 The Number of set(位运算 状态压缩)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3006 题目大意:给定n个集合,每个集合都是由大于等于1小于等于m的数字组成,m最大为14.由给出的集合 ...

  8. [BZOJ1151][CTSC2007]动物园zoo 解题报告|DP|位运算

    Description 最近一直在为了学习算法而做题,这道题是初一小神犇让我看的.感觉挺不错于是写了写. 这道题如果是一条线的话我们可以构造一个DP f[i,j]表示以i为起点,i,i+1...i+4 ...

  9. HDU 2276 Kiki & Little Kiki 2(矩阵位运算)

    Kiki & Little Kiki 2 转载自:点这里 [题目链接]Kiki & Little Kiki 2 [题目类型]矩阵位运算 &题意: 一排灯,开关状态已知,每过一秒 ...

随机推荐

  1. 在linux中限制用户ftp访问权限

    1.环境:redhat linux企业版4.ftp为vsftp.被限制用户名为aaa.被限制路径为/bbb.2.建用户:在root用户下,相继进行如下操作       adduser aaa      ...

  2. meta的属性详解

    引言 您的个人网站即使做得再精彩,在“浩瀚如海”的网络空间中,也如一叶扁舟不易为人发现,如何推广个人网站,人们首先想到的方法无外乎以下几种: ● 在搜索引擎中登录自己的个人网站 ● 在知名网站加入你个 ...

  3. Sencha Touch 之 Ext.fly方法的使用

    Ext.fly方法是Ext.js 4中的flyweight技术,该技术在浏览器中为使用Ext.fly方法的元素节点开辟一块内存,下一次使用Ext.fly方法的元素节点将占据同一块内存,即覆盖前一次的元 ...

  4. Distributed Machine Learning Toolkit

    http://www.dmtk.io http://www.dmtk.io/download.html

  5. html中上标、下标、删除字、小号字等

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  6. EPiServer网文

    ListItemCollection Rending: http://joelabrahamsson.com/episerver-7-and-mvc-how-to-customize-renderin ...

  7. 如何在C++中获得完整的类型名称(RTTI的typeid在不同平台下有不同的输出值表达,自建类改进了RTTI丢失的信息)

    Wrote by mutouyun. (http://darkc.at/cxx-get-the-name-of-the-given-type/)   地球人都知道C++里有一个typeid操作符可以用 ...

  8. Unix/Linux环境C编程入门教程(4) Debian Linux环境搭建

    Unix/Linux版本众多,我们推荐Unix/Linux初学者选用几款典型的Unix/Linux操作系统进行学习. 1.广义的Debian是指一个致力于创建自由操作系统的合作组织及其作品,由于Deb ...

  9. Day3_字符串操作与正则表达式

    本节课的主要内容有:字符串的格式化.连接与分割.比较.匹配和替换.使用正则表达式 字符串的格式化: 去除空格:trim() 使用html格式化:nl2br()  替换‘\n’为‘<br /> ...

  10. [原创] 小而美 | Mac上鲜为人知,但极大提升效率的小工具

    热爱收集实用又好用的软件,工具类软件不在多,发挥作用,提高效率最重要~推荐几个压箱底的藏货 一.Noizio -自然而然的白噪声,专注工作 Noizio是一款OS X 下的白噪音应用,可以让自己觉着是 ...