BZOJ_1495_[NOI2006]网络收费_树形DP
BZOJ_1495_[NOI2006]网络收费_树形DP
Description
Input
Output
你的程序只需要向输出文件输出一个整数,表示NS中学支付给网络公司的最小总费用。(单位:元)
Sample Input
1 0 1 0
2 2 10 9
10 1 2
2 1
3
Sample Output
8
直接模拟退火有80分。
代码:
// luogu-judger-enable-o2
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <cstdlib>
using namespace std;
typedef double f2;
#define N 1050
int f[N][N],C[N],n,m,cnt,ls[N<<2],rs[N<<2],a[N],fa[N<<2],L[N][N],R[N][N];
int c1[N],c2[N],b[N];
struct A {
int l,r;
}t[N<<2];
void build(int l,int r,int &p) {
if(l==r) {p=l; return ;}
else p=++cnt;
t[p]=(A){l,r};
int mid=(l+r)>>1;
build(l,mid,ls[p]); build(mid+1,r,rs[p]);
fa[ls[p]]=p; fa[rs[p]]=p;
}
A lca(int x,int y) {
while(x!=y) {
x=fa[x]; y=fa[y];
}
return t[x];
}
struct Q {
int b[N];
}ans;
int mn=1<<30;
int get_cost(const Q tmp) {
int i,re=0,j;
for(i=1;i<=m;i++) b[i]=tmp.b[i];
for(i=1;i<=m;i++) {
if(a[i]!=b[i]) re+=C[i];
if(!b[i]) c1[i]=c1[i-1]+1,c2[i]=c2[i-1];
else c1[i]=c1[i-1],c2[i]=c2[i-1]+1;
}
for(i=1;i<=m;i++) {
for(j=i+1;j<=m;j++) {
int l=L[i][j],r=R[i][j],na=c1[r]-c1[l-1],nb=c2[r]-c2[l-1];
if(!b[i]&&!b[j]) {
if(na<nb) re+=2*f[i][j];
}else if(!b[i]&&b[j]) {
re+=f[i][j];
}else if(b[i]&&!b[j]) {
re+=f[i][j];
}else {
if(na>=nb) re+=2*f[i][j];
}
}
}
if(mn>re) {
mn=re; ans=tmp;
}
return re;
}
f2 Rand() {
return 1.0*rand()/RAND_MAX;
}
void orz(f2 BG,f2 ED,f2 d) {
f2 B=BG;
int i,tmn;
Q nowp;
for(i=1;i<=m;i++) nowp.b[i]=a[i];
tmn=get_cost(nowp);
for(i=1;i<=m;i++) b[i]=a[i];
for(;B>ED;B*=d) {
Q tmp=nowp;
for(i=1;i<=B;i++) {
int k=rand()%m+1;
tmp.b[k]^=1;
}
int tans=get_cost(tmp);
if(tans<tmn||Rand()<exp(1.0*(tmn-tans)/B)) {
tmn=tans; nowp=tmp;
}
}
for(i=1;i<=1000;i++) {
Q tmp=ans;
int k=rand()%m+1;
tmp.b[k]^=1;
get_cost(tmp);
}
}
int main() {
// freopen("network.in","r",stdin);
// freopen("network.out","w",stdout);
srand(19260817); rand();
scanf("%d",&n);
m=1<<n; cnt=1<<n;
int i,j;
for(i=1;i<=m;i++) scanf("%d",&a[i]);
for(i=1;i<=m;i++) scanf("%d",&C[i]);
for(i=1;i<=m;i++) for(j=i+1;j<=m;j++) scanf("%d",&f[i][j]);
int root=n;
build(1,m,root);
for(i=1;i<=m;i++) {
for(j=i+1;j<=m;j++) {
A tmp=lca(i,j); L[i][j]=tmp.l; R[i][j]=tmp.r;
}
}
orz(m,1.5,0.99);
printf("%d\n",mn);
}
/*
2
1 0 1 0
2 2 10 9
10 1 2
2 1
3
*/
观察计算贡献的方式,相当于选哪边的少就计算哪边的贡献,此时i的贡献不止有小于i的这部分,还有大于i的这部分。
这样,只需要知道每个点对应子树内选哪边的少就可以确定贡献的计算方法。
设s[i][j]表示第i个点对应的深度为j的祖先上有多少贡献,这步在求lca的时候处理。
然后dfs整棵树。对于每个节点,枚举选a多还是选b多,在叶子节点统计答案。
时间复杂度:O(2^n *n)
代码:
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
#define N 2050
#define ls p<<1
#define rs p<<1|1
#define _min(x,y) ((x)<(y)?(x):(y))
__attribute__((optimize("-O3")))inline char nc() {
static char buf[100000],*p1,*p2;
return p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++;
}
__attribute__((optimize("-O3")))int rd() {
int x=0; char s=nc();
while(s<'0'||s>'9') s=nc();
while(s>='0'&&s<='9') x=(x<<3)+(x<<1)+s-'0',s=nc();
return x;
}
int f[N][1050],n,C[N],s[N][15],m,sta[N],a[N],g[4][N];
struct A {
int l,r,dep;
}t[N];
__attribute__((optimize("-O3")))void build(int l,int r,int p) {
t[p].l=l; t[p].r=r;
if(l==r) return ;
int mid=(l+r)>>1;
t[ls].dep=t[rs].dep=t[p].dep+1;
build(l,mid,ls); build(mid+1,r,rs); g[0][ls]=g[0][rs]=p;
}
__attribute__((optimize("-O3")))int lca(int x,int y) {
int i;
if(g[3][x]!=g[3][y]) x=g[3][x],y=g[3][y];
if(g[2][x]!=g[2][y]) x=g[2][x],y=g[2][y];
if(g[1][x]!=g[1][y]) x=g[1][x],y=g[1][y];
if(g[0][x]!=g[0][y]) x=g[0][x],y=g[0][y];
return g[0][x];
}
__attribute__((optimize("-O3")))void dfs(int p) {
int d=t[p].dep,i,j;
if(d==n) {
if(a[p-m+1]) f[p][1]=0,f[p][0]=C[p-m+1];
else f[p][0]=0,f[p][1]=C[p-m+1];
for(i=0;i<n;i++) f[p][!sta[i]]+=s[p-m+1][i];
return ;
}
int y=1<<(n-d-1);
for(i=0;i<=y<<1;i++) f[p][i]=1<<30;
sta[d]=0;
dfs(ls); dfs(rs);
for(i=0;i<=y;i++) for(j=0;i+j<=y;j++) f[p][i+j]=_min(f[p][i+j],f[ls][i]+f[rs][j]);
sta[d]=1;
dfs(ls); dfs(rs);
for(i=1;i<=y;i++) for(j=y-i+1;j<=y;j++) f[p][i+j]=_min(f[p][i+j],f[ls][i]+f[rs][j]);
}
__attribute__((optimize("-O3")))int main() {
n=rd();
m=1<<n;
build(1,m,1);
int i,j,x;
int k=m<<1;
for(i=1;i<=3;i++) for(j=1;j<=k;j++) g[i][j]=g[i-1][g[i-1][j]];
for(i=1;i<=m;i++) a[i]=rd();
for(i=1;i<=m;i++) C[i]=rd();
for(i=1;i<=m;i++) {
for(j=i+1;j<=m;j++) {
int l=lca(i+m-1,j+m-1);
x=rd();
s[i][t[l].dep]+=x;
s[j][t[l].dep]+=x;
}
}
// puts("FUCK");
dfs(1);
int ans=1<<30;
for(i=0;i<=m;i++) ans=_min(ans,f[1][i]);
printf("%d\n",ans);
}
BZOJ_1495_[NOI2006]网络收费_树形DP的更多相关文章
- BZOJ1495 [NOI2006]网络收费 【树形dp + 状压dp】
题目链接 BZOJ1495 题解 观察表格,实际上就是分\(A\)多和\(B\)两种情况,分别对应每个点选\(A\)权值或者\(B\)权值,所以成对的权值可以分到每个点上 所以每个非叶节点实际对应一个 ...
- 【bzoj1495】[NOI2006]网络收费 暴力+树形背包dp
题目描述 给出一个有 $2^n$ 个叶子节点的完全二叉树.每个叶子节点可以选择黑白两种颜色. 对于每个非叶子节点左子树中的叶子节点 $i$ 和右子树中的叶子节点 $j$ :如果 $i$ 和 $j$ 的 ...
- BZOJ 1495 [NOI2006]网络收费(暴力DP)
题意 给定一棵满二叉树,每个叶节点有一个状态0/10/10/1,对于每两个叶节点i,ji,ji,j,如果这两个叶节点状态相同但他们的LCALCALCA所管辖的子树中的与他们状态相同的叶节点个数较少(少 ...
- 【BZOJ1495】[NOI2006]网络收费 暴力+DP
[BZOJ1495][NOI2006]网络收费 Description 网络已经成为当今世界不可或缺的一部分.每天都有数以亿计的人使用网络进行学习.科研.娱乐等活动.然而,不可忽视的一点就是网络本身有 ...
- 洛谷 P4297 [NOI2006]网络收费
P4297 [NOI2006]网络收费 题目背景 noi2006 day1t1 题目描述 网络已经成为当今世界不可或缺的一部分.每天都有数以亿计的人使用网络进行学习.科研.娱乐等活动.然而,不可忽视的 ...
- 并不对劲的[noi2006]网络收费
题目略长,就从大视野上复制了. 听上去好像费用流,然而…… ***************************表示略长的题目的分界线************************ 1495: [ ...
- 【简】题解 P4297 [NOI2006]网络收费
传送门:P4297 [NOI2006]网络收费 题目大意: 给定一棵满二叉树,每个叶节点有一个状态(0,1),任选两个叶节点,如果这两个叶节点状态相同但他们的LCA所管辖的子树中的与他们状态相同的叶节 ...
- BZOJ_1864_[Zjoi2006]三色二叉树_树形DP
BZOJ_1864_[Zjoi2006]三色二叉树_树形DP 题意: 分析:递归建树,然后DP,从子节点转移. 注意到红色和蓝色没有区别,因为我们可以将红蓝互换而方案是相同的.这样的话我们只需要知道当 ...
- BZOJ_3573_[Hnoi2014]米特运输_树形DP+hash
BZOJ_3573_[Hnoi2014]米特运输_树形DP+hash 题意: 给你一棵树每个点有一个权值,要求修改最少的权值,使得每个节点的权值等于其儿子的权值和且儿子的权值都相等. 分析: 首先我们 ...
随机推荐
- OpenSceneGraph FAQ 【转】
1.地球背面的一个点,计算它在屏幕上的坐标,能得到吗? 不是被挡住了吗? 答:计算一个空间点的屏幕坐标,使用osgAPEx::GetScreenPosition函数.当空间点处于相机视空间内(不管它是 ...
- 转帖:HttpStatusCode状态说明C#版
Continue 等效于 HTTP 状态 100.Continue 指示客户端可能继续其请求. SwitchingProtocols 等效于 HTTP 状态 101.SwitchingProtocol ...
- 理解MySql事务隔离机制、锁以及各种锁协议
一直以来对数据库的事务隔离机制的理解总是停留在表面,其内容也是看一遍忘一边.这两天决定从原理上理解它,整理成自己的知识.查阅资料的过程中发现好多零碎的概念假设串起来足够写一本书,所以在这里给自己梳理一 ...
- 「零秒思考」是个神话,不过这款笔记术你值得拥有zz
今天读完了赤羽雄二的<零秒思考>,作者是一位在麦肯锡公司工作了 14 年的资深顾问.依照作者的说法,「零秒思考」指的是: 瞬间便能认清现状, 瞬间便能整理问题, 瞬间便能考虑出解决办法, ...
- C++中字符数组和字符串string
字符数组 C++中字符数组用char str[]能够用来表示一个字符串. (1) 数组的大小和字符串的长度. 数组的大小一定要大于字符串的长度,由于系统会自己主动补上一个'\0'作为字符串的结束标 ...
- (转)MongoDB在mongo控制台下的基本使用命令
成功启动MongoDB后,再打开一个命令行窗口输入mongo,就可以进行数据库的一些操作. 输入help可以看到基本操作命令: show dbs:显示数据库列表 show collections:显示 ...
- C# wince 实现软件忙鼠标状态改变
eg: Cursor.Current = Cursors.WaitCursor; dosomething(); Cursor.Current = Cursors.Default; Cursor.Cur ...
- 【BZOJ1845】[Cqoi2005] 三角形面积并 几何+扫描线
[BZOJ1845][Cqoi2005] 三角形面积并 Description 给出n个三角形,求它们并的面积. Input 第一行为n(N < = 100), 即三角形的个数 以下n行,每行6 ...
- 用EasyDarwin进行IPTV rtsp mpeg-ts smil流的转发和分发直播服务
对RTSP/RTP的转发和分发一直都是EasyDarwin的基础功能,尤其是在安防行业中,EasyDarwin非常贴合安防监控的需求,但一直未尝试用EasyDarwin进行IPTV的RTSP流进行转发 ...
- XML 解析错误:找不到根元素
大家在开发web项目的过程中,可能会遇到“XML 解析错误:找不到根元素”这么一个问题,引起这个问题的原因可能有很多种,在这儿我只是跟大家分享一下我遇到一种情况. 1.项目背景描述 extjs 结合a ...