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 题意: 给你一棵树每个点有一个权值,要求修改最少的权值,使得每个节点的权值等于其儿子的权值和且儿子的权值都相等. 分析: 首先我们 ...
随机推荐
- [Algorithm] Breadth First JavaScript Search Algorithm for Graphs
Breadth first search is a graph search algorithm that starts at one node and visits neighboring node ...
- SwitchyOmega 代理设置
1.SwitchyOmega官网 https://www.switchyomega.com/ 2.下载插件 https://www.switchyomega.com/download.html 3.配 ...
- jvm基础(1)
1.整型数和浮点型数的表示 原码:第一位为符号位(0为正数,1为负数). 反码:符号位不动,源码取反. 正数补码:和原码相同. 负数补码:符号位不动,反码加1. 例如5的二进制表示可以是0000010 ...
- MySQL 创始人:写代码比打游戏爽,程序员应多泡开源社区
王练 发布于2017年09月04日 收藏 43 开源中国全球专享福利,云栖大会购票大返现!>>> 根据StackOverflow的最新调查,MySQL仍然是全世界最流行的数 ...
- PS 魔法棒
魔术棒工具是通过选取图像中颜色相近或大面积单色区域的像素来制作选区,魔术棒用于纯色背景中较多. 容差数值越大,选择出的选区就越大,容差越小,对颜色差别的要求也就越严格,选择出的选区也就越小 按住shi ...
- Redis 过期键的设置、获取和删除过期时间
Redis 过期键的设置.获取和删除过期时间 转自http://blog.51cto.com/littledevil/1813956 设置过期 默认情况下键是没有生存时间的,也就是永不过期,除非清空内 ...
- Python 007- python的各种函数
1.chr().unichr()和ord() chr()函数用一个范围在range(256)内的(就是0-255)整数作参数,返回一个对应的字符. unichr()跟它一样,只不过返回的是Unicod ...
- Linux学习笔记--ps命令(显示当前进程的命令)
ps:英文名process,进程的意思. 1. 命令格式: ps [选项] 2. 经常使用选项: "ps -a" 显示一个终端的全部进程.除了会话引线 "ps -e&qu ...
- php 批量删除数据
php 批量删除数据 :比如我们在看邮箱文件的时候,积攒了一段时间以后,看到有些文件没有用了 这时候我们就会想到把这些 没用的文件删除,这时候就用到了批量删除数据的功能,这里我是用了数据库原有的一个表 ...
- c# 怎么获取自己的IP地址
1.aspx页面,asp.net项目的页面 <%@ Page Language="C#" AutoEventWireup="true" CodeBehin ...