传送门

有上下界最小费用可行流,行列建边变形。

行列建边相信大家都做过,没做过的可以看一下这个题:bzoj3698XWW的难题,对应的我写的题解题解

这个题需要变形一下,不只是单纯的对行列进行连边,首先对于一个空格,我们知道它最多影响2个线索:1个横向的和1个纵向的

所以我们可以对于每个线索划分出一个块,这个块包含所有的能影响它的空格以及它自己,显然这个做法和行列连边是同性质的,正确性亦显然

所以我们就得出了一个位置最多属于两个块,这个时候如果不考虑修改已经可以判断当前局面是否可行了

但是需要考虑修改,发现一个位置修改后的权值范围是\([1,+\infty]\),上下界预定了。

对于第\(i\)行第\(j\)列的这个点来说,如果它不是既不是空格也不是线索。

假设它纵向属于第x块,横向属于第y块(假如x为0就将x改为源点,y为0就将y改为汇点,自己思考,很显然的)

以下的连边都是以这样的格式\(add(x,y,down,up,cost)\)(x连y,下界,上界,费用)

第一种情况:这是一个空格

显然需要先建出\(add(x,y,val,val,0)\)

假如这个点可以修改,建出\(add(x,y,0.inf,cost),add(y,x,0,val-1,cost)\)

这两条边是分别代表增加权值和减少权值,看不懂自己思考,很显然的

第二种情况:这个位置有左下角的线索(有右上角也没关系)

先建出\(add(s,x,val,val,0)\)

假如这个点可以修改,建出\(add(s,x,0.inf,cost),add(x,s,0,val-1,cost)\)

边的含义同第一种情况

第三种情况:这个位置有右上角的线索(有左下角也没关系)

先建出\(add(y,t,val,val,0)\)

假如这个点可以修改,建出\(add(y,t,0.inf,cost),add(t,y,0,val-1,cost)\)

边的含义同前两种情况


P.S:

1、上下界网络流的细节我就不赘言了,来写的肯定都会吧

2、本人阐述能力有点差,不懂只能看代码了

3、本人代码费用流部分写的zkw费用流,其他费用流算法也是可以的

代码:

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
void read(int &x) {
char ch; bool ok;
for(ok=0,ch=getchar(); !isdigit(ch); ch=getchar()) if(ch=='-') ok=1;
for(x=0; isdigit(ch); x=x*10+ch-'0',ch=getchar()); if(ok) x=-x;
}
#define ll long long
#define rg register
const int maxn=1e5+10;const ll inf=1e18;
int s,t,ss,tt,n,cnt=1,m,id,in[maxn],mp[31][31],val[70][70][3],va[70][70],vb[70][70];
int pre[maxn*2],nxt[maxn*2],h[maxn],w[maxn*2];
int cost[70][70][3];ll sum,ans,flow,v[maxn*2],tot,dis[maxn],sla[maxn];
bool vis[maxn];
void add(int x,int y,ll z,int d)
{
if(!z)return ;
pre[++cnt]=y,nxt[cnt]=h[x],h[x]=cnt,v[cnt]=z,w[cnt]=d;
pre[++cnt]=x,nxt[cnt]=h[y],h[y]=cnt,v[cnt]=0,w[cnt]=-d;
}
ll dfs(int x,ll flow)
{
if(x==tt||!flow){ans+=dis[tt]*flow,tot+=flow;return flow;}
ll f=flow;vis[x]=1;
for(rg int i=h[x];i;i=nxt[i])
if(!vis[pre[i]]&&v[i])
{
if(dis[x]+w[i]==dis[pre[i]])
{
ll y=dfs(pre[i],min(f,v[i]));
f-=y,v[i]-=y,v[i^1]+=y;
if(!f)return flow;
}
else sla[pre[i]]=min(sla[pre[i]],dis[x]+w[i]-dis[pre[i]]);
}
return flow-f;
}
bool aug()
{
ll mx=inf;
for(rg int i=s;i<=tt;i++)if(!vis[i])mx=min(mx,sla[i]),sla[i]=inf;
if(mx==inf)return 1;
for(rg int i=s;i<=tt;i++)if(!vis[i])dis[i]+=mx;
return 0;
}
int main()
{
read(n),read(m);
for(rg int i=1;i<=n;i++)for(rg int j=1;j<=m;j++)read(mp[i][j]);
for(rg int i=1;i<=n;i++)
for(rg int j=1;j<=m;j++)
{
if(mp[i][j]==1)read(val[i][j][0]);
if(mp[i][j]==2)read(val[i][j][1]);
if(mp[i][j]==3)read(val[i][j][0]),read(val[i][j][1]);
if(mp[i][j]==4)read(val[i][j][2]);
if(mp[i][j]==1||mp[i][j]==3)
{
int now=++id,k=i;va[i][j]=now;k++;
while(mp[k][j]==4)va[k][j]=now,k++;
}
if(mp[i][j]==2||mp[i][j]==3)
{
int now=++id,k=j;vb[i][j]=now;k++;
while(mp[i][k]==4)vb[i][k]=now,k++;
}
}
s=0,t=++id;add(t,s,inf,0);ss=++id,tt=++id;
for(rg int i=1;i<=n;i++)
for(rg int j=1;j<=m;j++)
{
if(!mp[i][j])continue;
if(mp[i][j]==1)read(cost[i][j][0]);
if(mp[i][j]==2)read(cost[i][j][1]);
if(mp[i][j]==3)read(cost[i][j][0]),read(cost[i][j][1]);
if(mp[i][j]==4)read(cost[i][j][2]);
int x=va[i][j]?va[i][j]:s,y=vb[i][j]?vb[i][j]:t;
if(mp[i][j]==4)
{
add(x,y,0,0),in[x]-=val[i][j][2],in[y]+=val[i][j][2];
if(cost[i][j][2]!=-1)
add(x,y,inf,cost[i][j][2]),add(y,x,val[i][j][2]-1,cost[i][j][2]);
}
if(mp[i][j]==1||mp[i][j]==3)
{
add(s,x,0,0),in[s]-=val[i][j][0],in[x]+=val[i][j][0];
if(cost[i][j][0]!=-1)
add(s,x,inf,cost[i][j][0]),add(x,s,val[i][j][0]-1,cost[i][j][0]);
}
if(mp[i][j]==2||mp[i][j]==3)
{
add(y,t,0,0),in[y]-=val[i][j][1],in[t]+=val[i][j][1];
if(cost[i][j][1]!=-1)
add(y,t,inf,cost[i][j][1]),add(t,y,val[i][j][1]-1,cost[i][j][1]);
}
}
for(rg int i=s;i<=tt;i++)
{
if(in[i]>0)sum+=in[i],add(ss,i,in[i],0);
if(in[i]<0)add(i,tt,-in[i],0);
}
memset(sla,127,sizeof sla);
while(1)
{
while(1){memset(vis,0,sizeof vis);if(!dfs(ss,inf))break;}
if(aug())break;
}
printf("%lld\n",tot==sum?ans:-1);
}

bzoj5148:[BeiJing2018]Kakuro的更多相关文章

  1. HDU 3338 Kakuro Extension (网络流,最大流)

    HDU 3338 Kakuro Extension (网络流,最大流) Description If you solved problem like this, forget it.Because y ...

  2. hdu3338 Kakuro Extension 最大流

    If you solved problem like this, forget it.Because you need to use a completely different algorithm ...

  3. HDU3338:Kakuro Extension(最大流)

    Kakuro Extension Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  4. HDU3338 Kakuro Extension —— 最大流、方格填数类似数独

    题目链接:https://vjudge.net/problem/HDU-3338 Kakuro Extension Time Limit: 2000/1000 MS (Java/Others)     ...

  5. 使用GAC加速 解决CSP问题 Kakuro - Cross Sums

    Kakuro - Cross Sums 问题如下 一个简单的例子 可以看出限制条件是某行或某列的某几个空白格子求和等于某个值,且每一个限制中的格子所填的数必须为1-9且互异. 直接暴力搜索,空白格子太 ...

  6. Kakuro Extension HDU - 3338 (Dinic)

    Kakuro puzzle is played on a grid of "black" and "white" cells. Apart from the t ...

  7. [BJWC2018] Kakuro

    一.题目 点此看题 二.解法 我一开始一直想不出来,直接刚这个题实在是太复杂了,因为一开始就是不合法的. 下次遇到复杂的题一定要想 调整法 ,我再不往这个方向想我吔屎 好了言归正传,我们先找一组可行的 ...

  8. L - Kakuro Extension - HDU 3338 - (最大流)

    题意:有一个填数字的游戏,需要你为白色的块内填一些值,不过不能随意填的,是有一些规则的(废话),在空白的上方和作方给出一些值,如果左下角有值说明下面列的和等于这个值,右上角的值等于这行后面的数的和,如 ...

  9. 【最大流】【HDU3338】【Kakuro Extension】

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3338 题目大意:填数字,使白色区域的值得和等于有值得黑色区域的相对应的值,用网络流来做 题目思路:增加 ...

随机推荐

  1. java之快速排序

    //基本思想是:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据 ...

  2. PO 审批及生成xml文件

    *********************************************************************** * Report : YTST_RAINY_MM2 * ...

  3. C++ xml 解析器

    C++的xml解析器有很多,这个知乎回答里有一个列表:https://www.zhihu.com/question/32046606 下面使用其中的RapidXml试试. 官方地址: https:// ...

  4. target!

    工作到现在也有8个月了,从学生时代想从事嵌入是开发,到工作中从事android开发,跨度还是比较大的:曾经想从事这些消费类电子产品的开发,想从一个用户变成一个生产者,但是真正进入到这一行之后,才知道在 ...

  5. Mapper3中Example的高级使用方法

    http://zhuangxiaoda.leanote.com/post/Mapper3%E4%B8%ADExample%E7%9A%84%E4%BD%BF%E7%94%A8%E6%96%B9%E6% ...

  6. Windows程序设计(1)——Win32运行原理(三)

    进程控制 1 获得系统进程 2 终止当前进程 3 终止其他进程 4 进程控制 4.1 获得系统进程 使用toolhelp模块可以实现获取系统中当前运行当中的进程列表. 思路如下,使用CreateToo ...

  7. this.triggerEvent()用法

    在对组件进行封装时 在当前页面想要获取组件中的某一状态,需要使用到this.triggerEvent(' ',{},{}),第一个参数是自定义事件名称,这个名称是在页面调用组件时bind的名称,第二个 ...

  8. 重学DSP:对于卷积的理解

    最近,我发现自己对于一个事情,如果不给自己一个说服自己的理由,就会出现不能理解,不能记住,以至于不会使用或者“盲目”应用的情况. 但是,我学的这个学科就是应当建立在对信号作用过程的理解上面的. 下面, ...

  9. ES6 对Math对象的扩展

    Math 对象的扩展 Math.trunc() Math.trunc(4.1) // 4 Math.trunc(4.9) // 4 Math.trunc(-4.1) // -4 Math.trunc( ...

  10. bzoj2132【圈地计划】

    题面 思路: 一开始以为和为了博多一样,两边连一样的,后来发现中间连负边的话根本不会割,即割断两块收益为负,所以WA的起飞…… 正解是先黑白染色,每个点和它周围的点连边方式不同.对于黑点A,S--&g ...