【试题来源】

2011中国国家集训队命题答辩

【问题描述】

高一一班的座位表是个n*m的矩阵,经过一个学期的相处,每个同学和前后左右相邻的同学互相成为了好朋友。这学期要分文理科了,每个同学对于选择文科与理科有着自己的喜悦值,而一对好朋友如果能同时选文科或者理科,那么他们又将收获一些喜悦值。作为计算机竞赛教练的scp大老板,想知道如何分配可以使得全班的喜悦值总和最大。

【输入格式】

第一行两个正整数n,m。
接下来是六个矩阵
第一个矩阵为n行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学选择文科获得的喜悦值。
第二个矩阵为n行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学选择理科获得的喜悦值。
第三个矩阵为n-1行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i+1行第j列的同学同时选择文科获得的额外喜悦值。
第四个矩阵为n-1行m列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i+1行第j列的同学同时选择理科获得的额外喜悦值。
第五个矩阵为n行m-1列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i行第j+1列的同学同时选择文科获得的额外喜悦值。
第六个矩阵为n行m-1列 此矩阵的第i行第j列的数字表示座位在第i行第j列的同学与第i行第j+1列的同学同时选择理科获得的额外喜悦值。

【输出格式】

输出一个整数,表示喜悦值总和的最大值

【样例输入】

1 2
1 1
100 110
1
1000

【样例输出】

1210

【样例说明】

两人都选理,则获得100+110+1000的喜悦值。

【数据规模和约定】

对于10%以内的数据,n,m<=4
对于30%以内的数据,n,m<=8
对于100%以内的数据,n,m<=100 数据保证答案在2^30以内
对于100%的数据,时间限制为0.5s。
 
【题解】
    这道题的难点在于确定边权。最小割问题,割去的就是我们失去的部分;两点之间有关系,总是通过建边来实现的。对于这道题来说,每种情况我们都失去了什么?以源点代表文科,汇点代表理科。都选一科(三角环),失去了共同选另一科和分别选另一科的喜悦值。分别选两科(二字形),失去了两个共同喜悦值和两个单独选另一科的喜悦值。(可以证明可能出现的情况只有这两种,否则都不会是最小割)相同位置的边权构成一定相同,因此用数学方法推出每个人到源点或汇点的边权为个人喜悦值+1/2共同喜悦值,两点之间边权为1/2都选文+1/2都选理。注意共同边要双向建边,因为两点之间是完全等效的;每个人向源点和汇点的边应该在边权全部处理完之后再统一添加。
    可以发现边权会出现实型,结果却一定是整型。对于这种情况,可以把边权全部*2,最后结果再/2来避免double的麻烦。dfs函数中有一个语句:if(!f) break;原先从来没打过,这道题不加这个却会超时,加了之后直接上榜,确实是一个非常有理有据的优化。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
const int sj=;
int n,m,sx[sj][sj],e,h[sj*sj],s,t,dep[sj*sj];
int w[sj][sj],l[sj][sj],g[sj][sj],z[sj][sj],a1,ans;
struct B
{
int ne,v,w;
}b[sj*sj*];
queue<int> q;
void add(int x,int y,int z)
{
b[e].v=y;
b[e].w=z;
b[e].ne=h[x];
h[x]=e++;
}
void init()
{
scanf("%d%d",&n,&m);
t=n*m+;
memset(h,-,sizeof(h));
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
{
scanf("%d",&w[i][j]);
sx[i][j]=(i-)*m+j;
ans+=w[i][j];
w[i][j]*=;
}
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
{
scanf("%d",&l[i][j]);
ans+=l[i][j];
l[i][j]*=;
}
for(int i=;i<n;i++)
for(int j=;j<=m;j++)
{
scanf("%d",&a1);
ans+=a1;
g[i][j]=a1;
w[i][j]+=a1;
w[i+][j]+=a1;
}
for(int i=;i<n;i++)
for(int j=;j<=m;j++)
{
scanf("%d",&a1);
ans+=a1;
g[i][j]+=a1;
l[i][j]+=a1;
l[i+][j]+=a1;
add(sx[i][j],sx[i+][j],g[i][j]);
add(sx[i+][j],sx[i][j],g[i][j]);
}
for(int i=;i<=n;i++)
for(int j=;j<m;j++)
{
scanf("%d",&a1);
ans+=a1;
z[i][j]=a1;
w[i][j]+=a1;
w[i][j+]+=a1;
add(s,sx[i][j],w[i][j]);
add(sx[i][j],s,);
}
for(int i=;i<=n;i++)
{
add(s,sx[i][m],w[i][m]);
add(sx[i][m],s,);
}
for(int i=;i<=n;i++)
for(int j=;j<m;j++)
{
scanf("%d",&a1);
ans+=a1;
z[i][j]+=a1;
l[i][j]+=a1;
l[i][j+]+=a1;
add(t,sx[i][j],);
add(sx[i][j],t,l[i][j]);
add(sx[i][j],sx[i][j+],z[i][j]);
add(sx[i][j+],sx[i][j],z[i][j]);
}
for(int i=;i<=n;i++)
{
add(sx[i][m],t,l[i][m]);
add(t,sx[i][m],);
}
ans*=;
}
bool bfs(int x)
{
while(!q.empty()) q.pop();
memset(dep,,sizeof(dep));
dep[x]=;
q.push(x);
while(!q.empty())
{
x=q.front();
q.pop();
for(int i=h[x];i!=-;i=b[i].ne)
if(!dep[b[i].v]&&b[i].w)
{
dep[b[i].v]=dep[x]+;
if(b[i].v==t) return ;
q.push(b[i].v);
}
}
return ;
}
int bj(int x,int y)
{
return x<y?x:y;
}
int dfs(int x,int f)
{
if(x==t) return f;
int ans=,d;
for(int i=h[x];i!=-;i=b[i].ne)
if(dep[b[i].v]==dep[x]+&&b[i].w)
{
d=dfs(b[i].v,bj(f,b[i].w));
f-=d;
ans+=d;
b[i].w-=d;
b[i^].w+=d;
if(!f) break;
}
if(!ans) dep[x]=-;
return ans;
}
int main()
{
init();
while(bfs(s)) ans-=dfs(s,0x7fffffff);
printf("%d",ans/);
return ;
}

happiness

 

happiness[国家集训队2011(吴确)]的更多相关文章

  1. AC日记——[国家集训队2011]旅游(宋方睿) cogs 1867

    [国家集训队2011]旅游(宋方睿) 思路: 树链剖分,边权转点权: 线段树维护三个东西,sum,max,min: 当一个区间变成相反数时,sum=-sum,max=-min,min=-max: 来, ...

  2. cogs 1901. [国家集训队2011]数颜色

    Cogs 1901. [国家集训队2011]数颜色 ★★★   输入文件:nt2011_color.in   输出文件:nt2011_color.out   简单对比时间限制:0.6 s   内存限制 ...

  3. BZOJ 2150 cogs 1861 [国家集训队2011]部落战争

    题目描述 lanzerb的部落在A国的上部,他们不满天寒地冻的环境,于是准备向A国的下部征战来获得更大的领土. A国是一个M*N的矩阵,其中某些地方是城镇,某些地方是高山深涧无人居住.lanzerb把 ...

  4. COGS1882 [国家集训队2011]单选错位

    ★   输入文件:nt2011_exp.in   输出文件:nt2011_exp.out   简单对比时间限制:1 s   内存限制:512 MB [试题来源] 2011中国国家集训队命题答辩 [问题 ...

  5. 1893. [国家集训队2011]等差子序列(bitset)

    ★★   输入文件:nt2011_sequence.in   输出文件:nt2011_sequence.out   简单对比时间限制:0.3 s   内存限制:512 MB [试题来源] 2011中国 ...

  6. 【COGS 1873】 [国家集训队2011]happiness(吴确) 最小割

    这是一种最小割模型,就是对称三角,中间双向边,我们必须满足其最小割就是满足题目条件的互斥关系的最小舍弃,在这道题里面我们S表示文T表示理,中间一排点是每个人,每个人向两边连其选文或者选理的价值,中间每 ...

  7. [国家集训队2011]happiness

    Description 高一一班的座位表是个n*m的矩阵,经过一个学期的相处,每个同学和前后左右相邻的同学互相成为了好朋友.这学期要分文理科了,每个同学对于选择文科与理科有着自己的喜悦值,而一对好朋友 ...

  8. bzoj2144 【国家集训队2011】跳跳棋

    Description 跳跳棋是在一条数轴上进行的.棋子只能摆在整点上.每个点不能摆超过一个棋子.我们用跳跳棋来做一个简单的游戏:棋盘上有3颗棋子,分别在a,b,c这三个位置.我们要通过最少的跳动把他 ...

  9. COGS.1901.[模板][国家集训队2011]数颜色(带修改莫队)

    题目链接 COGS BZOJ2120 洛谷P1903 /* Add和Subd函数中的vis不能直接设为=1或=0 比如 l=1,r=0 -> l=3,r=5 时,[1,5]的vis标记全都是1 ...

随机推荐

  1. nginx常用配置系列-HTTPS配置

    接上篇,nginx配置系列 HTTPS现在已经很流行,特别是AppStore上线的应用要求使用HTTPS进行通信,出于安全考虑也应该使用HTTPS,HTTPS配置需要准备证书文件,现在也有很多免费证书 ...

  2. SOD开源框架MSF(消息服务框架)进阶篇

    复习:在上一篇我介绍了MSF的基本订阅,模式就是,客户端A,订阅服务器.客户端B,订阅服务器.通过服务器广播消息, 所有订阅过的客户端都能接到消息. 进阶:在上一篇的基础上,增加客户端A,发送信息到服 ...

  3. html页面多个a标签点击时显示不同的样式

    <!DOCTYPE HTML> <html lang="en-US"> <head> <meta charset="UTF-8& ...

  4. 淘宝tairKV分布式

    Tair是什么 Tair是由淘宝开发的key/value方案,系统默认支持基于内存和文件的存储引擎,对应于通常我们所说的缓存和持久化存储,这里可以获取更多关于tair的信息,淘宝团队介绍,Tair在淘 ...

  5. SharePoint 2016 每天预热脚本介绍

    使用SharePoint的朋友们应该知道,SharePoint每天夜里有自动回收的机制,使环境每天把占用的内存都释放出来,以确保不会累计占用过多内存导致服务器崩溃. 我们可以打开IIS,选中我们的应用 ...

  6. php7 安装yar 生成docker镜像

    Docker包含三个概念:(1)远程仓库即远程镜像库所有镜像的聚集地(不可进入操作).(2)本地镜像即从远程仓库拉取过来的镜像(3)运行起来的本地镜像叫做容器(分层的可操作)Docker使用:1.首先 ...

  7. [leetcode-543-Diameter of Binary Tree]

    Given a binary tree, you need to compute the length of the diameter of the tree. The diameter of a b ...

  8. Japanese Learning - Words and Sentences 1

    1. いらっしゃいませ.何名さまですか. 二人です. タバコをお吸いになりますか. かしこまりました.少々お待ちください. お勘定お願いします. ご一緒でよろしいでしょうか. 別々にお願いします. 2 ...

  9. 关于Android log拿不到的情况

    遇到很多开发者说crash了,log没有看到..出现类似情况的基本原因是因为现在的国产厂商如crash了会直接将进程杀掉,于是你的studio就看不到了,可以往下面几个方向去想办法找到crash的lo ...

  10. Asp.net管理信息系统中数据统计功能的实现

    数据统计是每个系统中必备的功能,在给领导汇报统计数据,工作中需要的进展数据时非常有用. 在我看来,一个统计的模块应该实现以下功能: 能够将常用的查询的统计结果显示出来: 显示的结果可以是表格形式,也可 ...