【学术篇】SDOI2008 山贼集团
今天一月一号..
突然想安利一波我的中二的2017总结...
传送门1:codevs
传送门2:luogu
时限5s和1s的区别(你没看我传送门都给的大牛分站了)
现在不仅线筛.. 有负数的快读都打不对了..
来比较一下他们的区别?
inline int gn(int a=0,char c=0,int f=1){
for(;(c<48||c>57)&&c!='-';c=getchar());if(c=='-')f=-1,c=getchar();
for(;c>47&&c<58;c=getchar()) a=a*10+c-'0'; return a*f;
}
inline int gn(int a=0,char c=0,bool f=1){
for(;(c<48||c>57)&&c!='-';c=getchar());if(c=='-')f=-1,c=getchar();
for(;c>47&&c<58;c=getchar()) a=a*10+c-'0'; return a*f;
}
inline int gn(int a=0,char c=0,bool f=1){
for(;(c<48||c>57)&&c!='-';c=getchar());if(c=='-')f=-1;
for(;c>47&&c<58;c=getchar()) a=a*10+c-'0'; return a*f;
}
Emmmm 于是就愉快的残掉了..
好吧回到正题.
这个题网上的题解好少啊OvO
毕竟bzoj有10道sdoi2008, 这题就属于被忽略的题目之一...
不知道为什么...
数据范围\(p<=12\)一眼状压...
我们可以用12位二进制表示一个集合...
这样定义状态\(f_{x,s}\)为第\(i\)个节点上安排集合\(s\)的状态...
这样的话\(s\)就是每个儿子和安排在该点的集合们的并集...
但是很多个儿子差集就不好取了, 考虑多叉树转二叉树...
然而似乎传统的左儿子右兄弟是不行的... 我们考虑另一种转化方式..
比如我们有一棵这样的树:
转成一棵抉择方案等价的树是这样的:
这里我们对于有多个子树的节点, 建立虚拟节点(注意:虚拟节点是不能安排集合的)
如果有好多子树就继续递归下去(比如如果1有2 3 4 5四个子树, 那么就在10的右儿子挂一个12, 然后把4 5分别放在12的左右儿子.
由于有些点不能放集合, 我们定义\(g_{x,s}\)表示在以\(x\)为根的子树中不在\(x\)节点安排集合时的最大价值.
这样我们就可以根据二叉树写出状态转移方程:
max\{g_{x,k}-cost_{s-k}\}+val_s (k\subseteq s),\; x<=n\\
\\
g_{x,s},\; x>n
\end{matrix}\right.\\
g_{x,s}=max\{f_{l,k}+f_{r,s-k}\}(k\subseteq s)
\]
根据这个状态转移方程推就行了...
不过好像是有些卡时间的...
我们可以预处理出某个集合的费用\(cost\)和价值\(val\)
然后枚举子集是有技巧的:
for(int k=s;k;k=(k-1)&s){
}
这样会快一点... 大约能把复杂度从\(4^n\)降到\(3^n\)左右...
不过要记得特殊处理空集(因为这样枚举的\(k\)不会到0)
就做完了...
代码(压常数版):
由于压了常数变得非常丑(本来写的也没多好看OvO)...
#include <cstdio>
#include <cstdio>
#include <cstring>
#define ri register int
const int N=204,P=4100,I=-1061109568;
int f[N][P],g[N][P],du[N>>1];
int w[N][13],val[P],Val[P],cost[N][P];
int ch[2][N],n,nn,p,t; bool vis[N];
int a,b,s;
inline int gi(int a=0,char c=0){
for(;c<48||c>57;c=getchar());
for(;c>47&&c<58;c=getchar())a=a*10+c-48;return a;
}
inline int gn(int a=0,char c=0,int f=1){
for(;(c<48||c>57)&&c!='-';c=getchar());if(c=='-')f=-1,c=getchar();
for(;c>47&&c<58;c=getchar()) a=a*10+c-'0'; return a*f;
}
inline int max(const int &a,const int &b){return a>b?a:b;}
inline int min(const int &a,const int &b){return a<b?a:b;}
struct edge{
int to,next;
}e[N]; int v[N>>1],tot;
inline void buildedge(const int &x,const int &y){
e[++tot].to=y; e[tot].next=v[x]; v[x]=tot; ++du[x];
e[++tot].to=x; e[tot].next=v[y]; v[y]=tot; ++du[y];
}
void dfs1(int x){
int now=x; vis[x]=1;
for(ri i=v[x];i;i=e[i].next){int y=e[i].to;
if(!vis[y]){
if(!ch[0][now]) ch[0][now]=y;
else if(du[x]==1) ch[1][now]=y;
else ch[1][now]=++nn,now=nn,ch[0][now]=y;
--du[y];--du[x]; dfs1(y);
}
}
}
int G(int x,int zt);
int F(int x,int zt){
if(f[x][zt]>I) return f[x][zt];
if(x>n) return G(x,zt);
f[x][zt]=Val[zt]-cost[x][zt];
for(ri z=zt;z;z=(z-1)&zt)
f[x][zt]=max(f[x][zt],G(x,z)-cost[x][zt^z]+Val[zt]);
return f[x][zt];
}
int G(int x,int zt){
if(!ch[0][x]&&!ch[1][x]) return I;
if(g[x][zt]>I) return g[x][zt];
if(!ch[1][x])
g[x][zt]=F(ch[0][x],zt);
else if(!ch[0][x])
g[x][zt]=F(ch[1][x],zt);
else{
for(ri z=zt;z;z=(z-1)&zt)
g[x][zt]=max(g[x][zt],F(ch[0][x],z)+F(ch[1][x],zt^z));
g[x][zt]=max(g[x][zt],F(ch[1][x],0)+F(ch[1][x],zt));
}
return g[x][zt];
}
int main(){
nn=n=gi(); p=gi();
memset(f,192,sizeof(f));
memset(g,192,sizeof(g));
for(ri i=1;i<n;++i){
a=gi(),b=gi();
buildedge(a,b);
}
for(ri i=1;i<=n;++i)
for(ri j=1;j<=p;++j)
w[i][j]=gi();
t=gn();
for(ri i=1;i<=t;++i){
a=gn(),b=gi(),s=0;
for(ri j=0;j<b;++j)
s|=(1<<(gn()-1));
val[s]+=a;
} dfs1(1);
for(ri i=0;i<1<<p;++i){
for(ri j=i;j;j=(j-1)&i)
Val[i]+=val[j];
for(ri j=1;j<=n;++j)
for(ri k=1;k<=p;++k)
if(i&(1<<(k-1)))
cost[j][i]+=w[j][k];
}
printf("%d\n",F(1,(1<<p)-1));
}
【学术篇】SDOI2008 山贼集团的更多相关文章
- P2465 [SDOI2008]山贼集团 dp
这个题是一道树形dp+状压dp二合一,先预处理每种组合会有什么额外的费用,然后在树上dp就行了. 题干: 题目描述 某山贼集团在绿荫村拥有强大的势力,整个绿荫村由N个连通的小村落组成,并且保证对于每两 ...
- [SDOI2008]山贼集团
题目描述 某山贼集团在绿荫村拥有强大的势力,整个绿荫村由\(N\)个连通的小村落组成,并且保证对于每两个小村落有且仅有一条简单路径相连. 小村落用阿拉伯数字编号为\(1,2,3,4, \dots ,n ...
- 洛咕 P2465 [SDOI2008]山贼集团
裸的状压dp. 设f[i][j]表示在i字数内放j集合的分部,直接sb转移. // luogu-judger-enable-o2 #include<bits/stdc++.h> #defi ...
- 【[SDOI2008]山贼集团】
非常好的一道题 树上的状压\(dp\) 根据数据范围我们就能知道这是一道需要状压的题目 所以状态就是\(dp[i][S]\)表示在以\(i\)为根的子树里,选择的状态为\(S\)的最大收益 这个收益只 ...
- 山贼集团 (group)
山贼集团 (group) 题目描述 某山贼集团在绿荫村拥有强大的势力,整个绿荫村由N个连通的小村落组成,并且保证对于每两个小村落有且仅有一条简单路径相连.小村落用阿拉伯数字编号为1,2,3,4,-,n ...
- 【学术篇】SDOI2008 仪仗队
Part1:传送门&吐槽 水题... 然而由于线筛里面的\(j\)打成了\(i\)然后就不能1A了OvO Part2:题目分析 这个正方形是对称的... 而且很显然对角线上只有一个点会被看到. ...
- 【学术篇】SDOI2008 沙拉公主的困惑
传送门! 题目在这里... 题目大意? 难道不是说的很清楚了么OvO 求n!中与m!互质的数的个数.. 题目分析. 显然的数论... 所以就是化式子呗.. 一个很显然的性质就是如果\(gcd(a,b) ...
- 【Luogu】P2465山贼集团(树形状压DP)
题目链接 写了个70分暴力还挂了,第一遍提交只拿了十分……海星 首先建虚拟节点多叉树转成二叉,然后子集枚举DP 设g[x][i]是以x为根的子树内山贼集合i,x啥都不选也没贡献的时候的最大价值 f[x ...
- 【学术篇】51nod 1238 最小公倍数之和
这是一道杜教筛的入(du)门(liu)题目... 题目大意 求 \[ \sum_{i=1}^n\sum_{j=1}^nlcm(i,j) \] 一看就是辣鸡反演一类的题目, 那就化式子呗.. \[ \s ...
随机推荐
- Pandas之read_excel()和to_excel()函数解析
read_excel() 加载函数为read_excel(),其具体参数如下. read_excel(io, sheetname=0, header=0, skiprows=None, skip_fo ...
- 洛谷 P2652 同花顺(离散化)
洛谷 P2652 同花顺(题面) 手动模拟了一下,其实离散化排序可以起很大作用题目要求花色相同,数字连续,那么我们要做的就是找一种花色,并提取出其中一串数字留下那些舍弃的牌换成相应花色,并和之前留下的 ...
- String,StringBuffer,StringBuilder
String是字符串常量:StringBuffer,StringBuilder是字符串变量 StringBuffer是线程安全的且效率低:StringBuilder是线程不安全的故效率高 操作少量字符 ...
- Vue之获取用户当前所在省市
今天小编给大家带来的是使用Vue获取用户所在城市,Vue是很强大的,给大家准备好现成的插件供大家调用,下面的Demo小编使用的是百度API. 首先我们从百度平台申请百度地图的秘钥,申请成功后我们将&l ...
- Mariadb 10.2.8版本GTID主从环境搭建以及切换
1.首先搭建主从 主环境:192.168.1.117 从环境:192.168.1.123 a.首先以二进制包的形式安装好MariaDB (忽略不计) b.配置环境的变量 通配 [mysqld] bin ...
- python学习笔记(十)——正则表达式和re模块
#正则表达式和re模块 # match(pattern, string,[flag]) #在字符串开始时进行匹配 # pattern 正则表达式 # string 要匹配的字符串 # [flag] 可 ...
- ElementUI的Loading组件 —— 想实现在请求后台数据之前开启Loading组件,请求成功或失败之后,关闭Loading组件
我在实际项目开发中,遇到了这个需求,记录一下~~~~~~ 在ElementUI官网上有几种实现Loading的方法,但官网上是在一个方法里写了开启与关闭组件,所以可以根据官网的实现方法进行一个封装,便 ...
- CSIC_716_20191129【面向对象高级----反射、类的内置方法(魔法方法)、单例模式】
反射 反射是通过'字符串'对 对象的属性进行操作,反射有四个内置的方法. hasattr 通过字符串 判断对象的属性或者方法是否存在 getattr 通过字符串 获取对象的属性或者方法 ...
- CSS选择器,层叠
CSS选择器 .class .intro 选择 class="intro" 的所有元素. 1 #id #firstname 选择 id="firstname" ...
- leetcode-158周赛-5225-最大相等频率
题目描述: 方法: class Solution(object): def maxEqualFreq(self, A): count = collections.Counter() freqs = c ...