题意:给一个方格,每行每列都有对白色格子中的数之和的要求。每个格子中的数范围在[1,9]中。现在给出了这些要求,求满足条件的解。

分析:本题读入和建图比较恶心...

用网络流求解。建立源点S和汇点T,设列之和的点为A,行之和的点为B,白色格点为C。

由{A}向{C}建边,因为每个白色格点的容量不超过9且不能为0,所以在建边的时候统一减去1,最后输出解的时候再+1,则其余的弧也要相应地减去对应的白色点个数;

由{C}向{B}建边,容量为8;

由S向{A}建边,容量为 |A的列和要求 - 其下方白色点的个数|;

由{B}向T建边,容量为 |B的行和要求 - 其右方白色点的个数|。

但,这样做可能会出现某个黑色点既对列和有要求,又对行和有要求,这样建图后,这个点和源点和汇点都连接,跑出来的结果有问题。所以,将这类点拆成两个点,一个作为列之和的要求与源点相连,另一个与汇点相连。

跑出最大流后,残余网上对应弧的流量+1即为所求的解。

#include <bits/stdc++.h>
using namespace std;
const int INF = 0x3f3f3f3f;
const int MAXN=100010;//点数的最大值
const int MAXM=400010;//边数的最大值
#define captype int
struct EDGE{
int to,next;
captype cap;
};
struct SAP_MaxFlow{
EDGE edg[MAXM];
int eid,head[MAXN];
int gap[MAXN];
int dis[MAXN];
int cur[MAXN];
int pre[MAXN]; void init(){
eid=0;
memset(head,-1,sizeof(head));
}
void AddEdge(int u,int v,captype c,captype rc=0){
edg[eid].to=v; edg[eid].next=head[u];
edg[eid].cap=c; head[u]=eid++;
edg[eid].to=u; edg[eid].next=head[v];
edg[eid].cap=rc; head[v]=eid++;
}
captype maxFlow_sap(int sNode,int eNode, int n){//n是包括源点和汇点的总点个数,这个一定要注意
memset(gap,0,sizeof(gap));
memset(dis,0,sizeof(dis));
memcpy(cur,head,sizeof(head));
pre[sNode] = -1;
gap[0]=n;
captype ans=0;
int u=sNode;
while(dis[sNode]<n){
if(u==eNode){
captype Min=INF ;
int inser;
for(int i=pre[u]; i!=-1; i=pre[edg[i^1].to])
if(Min>edg[i].cap){
Min=edg[i].cap;
inser=i;
}
for(int i=pre[u]; i!=-1; i=pre[edg[i^1].to]){
edg[i].cap-=Min;
edg[i^1].cap+=Min;
}
ans+=Min;
u=edg[inser^1].to;
continue;
}
bool flag = false;
int v;
for(int i=cur[u]; i!=-1; i=edg[i].next){
v=edg[i].to;
if(edg[i].cap>0 && dis[u]==dis[v]+1){
flag=true;
cur[u]=pre[v]=i;
break;
}
}
if(flag){
u=v;
continue;
}
int Mind= n;
for(int i=head[u]; i!=-1; i=edg[i].next)
if(edg[i].cap>0 && Mind>dis[edg[i].to]){
Mind=dis[edg[i].to];
cur[u]=i;
}
gap[dis[u]]--;
if(gap[dis[u]]==0) return ans;
dis[u]=Mind+1;
gap[dis[u]]++;
if(u!=sNode) u=edg[pre[u]^1].to; //退一条边
}
return ans;
}
}F; pair<int,int> G[105][105];
char str[20];
int N,M,s,t;
int res[105][105]; void debug(){
for(int i=1;i<=N;++i){
for(int j=1;j<=M;++j){
cout<<G[i][j].first<<"/"<<G[i][j].second<<" ";
}
cout<<endl;
}
} void down(int x,int y)
{
int u = (x-1)*M+y;
int j = y;
int cnt = 0;
for(int i=x+1;i<=N;++i){
if(G[i][j].first!=-1) break;
int v = (i-1)*M+j;
F.AddEdge(u,v,8);
cnt++;
}
F.AddEdge(s,u,G[x][y].first-cnt);
} void right(int x,int y)
{
int v = (x-1)*M+y;
int i = x;
int cnt = 0;
for(int j = y+1;j<=M;++j){
if(G[i][j].first!=-1) break;
int u = (i-1)*M+j;
F.AddEdge(u,v,8);
cnt++;
}
F.AddEdge(v,t,G[x][y].second-cnt);
} void downAndRight(int x,int y)
{
int u = (x-1)*M+y;
int j = y;
int cnt = 0;
for(int i=x+1;i<=N;++i){
if(G[i][j].first!=-1) break;
int v = (i-1)*M+j;
F.AddEdge(u,v,8);
cnt++;
}
F.AddEdge(s,u,G[x][y].first-cnt); int v = (x-1)*M+y+M*N; //这种点需要拆点
int i = x;
cnt = 0;
for(int j = y+1;j<=M;++j){
if(G[i][j].first!=-1) break;
int u = (i-1)*M+j;
F.AddEdge(u,v,8);
cnt++;
}
F.AddEdge(v,t,G[x][y].second-cnt);
} vector<pair<int,int> > col,row,both; int main()
{
#ifndef ONLINE_JUDGE
freopen("in.txt","r",stdin);
freopen("out.txt","w",stdout);
#endif
while(scanf("%d %d",&N, &M)==2){
F.init();
col.clear();
row.clear();
both.clear();
for(int i=1;i<=N;++i){
for(int j=1;j<=M;++j){
scanf("%s",str);
if(str[0]=='X') G[i][j].first = -2;
else if(str[0]=='.') G[i][j].first = -1;
else{
G[i][j].first = (str[0]-'0')*100+ (str[1]-'0')*10 + (str[2]-'0');
}
if(str[4]=='X') G[i][j].second = -2;
else if(str[4]=='.') G[i][j].second = -1;
else{
G[i][j].second = (str[4]-'0')*100 +(str[5]-'0')*10 + (str[6]-'0');
}
if(G[i][j].first>0 && G[i][j].second>0){
both.push_back(make_pair(i,j));
}
else if(G[i][j].first>0){
col.push_back(make_pair(i,j));
}
else if(G[i][j].second>0){
row.push_back(make_pair(i,j));
}
}
}
//debug();
memset(res,0,sizeof(res));
s= 0 ,t = N*M*2+2;
for(auto & v:col){
down(v.first,v.second);
}
for(auto & v:row){
right(v.first,v.second);
}
for(auto &v:both){
downAndRight(v.first,v.second);
}
int flow = F.maxFlow_sap(s,t,t+1);
//cout<<flow<<endl;
for(int i= F.head[s];~i;i=F.edg[i].next){
int u = F.edg[i].to;
//cout<<u<<endl;
for(int j= F.head[u];~j;j=F.edg[j].next){
EDGE & e =F.edg[j];
int v = e.to;
if(v==s) continue;
int x = (v-1)/M+1;
int y = v%M; if(y==0) y = M;
//cout<<e.flow+1<<" ";
//cout<<v<<" "<<x<<" "<<y<<endl;
res[x][y] = 9-e.cap;
}
//cout<<endl;
} for(int i=1;i<=N;++i){
for(int j=1;j<=M;++j){
if(G[i][j].first!=-1) printf("_");
else printf("%d",res[i][j]);
if(j==M) printf("\n");
else printf(" ");
}
}
}
return 0;
}

HDU - 3338 Kakuro Extension (最大流求解方格填数)的更多相关文章

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

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

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

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

  3. HDU 3338 Kakuro Extension

    网络最大流 TLE了两天的题目.80次Submit才AC,发现是刘汝佳白书的Dinic代码还可以优化.....瞬间无语..... #include<cstdio> #include< ...

  4. DFS(深度优先搜索遍历求合格条件总数)--07--DFS--蓝桥杯方格填数

    此题方法多种,我用规范的DFS来求解 题目:方格填数 如下的10个格子,填入0~9的数字.要求:连续的两个数字不能相邻. (左右.上下.对角都算相邻)一共有多少种可能的填数方案?   输出 请填写表示 ...

  5. java算法 蓝桥杯(题+答案) 方格填数

    6.方格填数  (结果填空) 如下的10个格子 (如果显示有问题,也可以参看[图1.jpg]) 填入0~9的数字.要求:连续的两个数字不能相邻.(左右.上下.对角都算相邻) 一共有多少种可能的填数方案 ...

  6. 蓝桥杯比赛javaB组练习《方格填数》

    方格填数 如下的10个格子   +--+--+--+   |  |  |  |+--+--+--+--+|  |  |  |  |+--+--+--+--+|  |  |  |+--+--+--+ ( ...

  7. java算法 第七届 蓝桥杯B组(题+答案) 6.方格填数

    6.方格填数  (结果填空) 如下的10个格子 (如果显示有问题,也可以参看[图1.jpg]) 填入0~9的数字.要求:连续的两个数字不能相邻.(左右.上下.对角都算相邻) 一共有多少种可能的填数方案 ...

  8. c++_方格填数(最新方法)

      方格填数 如下的10个格子 +--+--+--+ | | | |+--+--+--+--+| | | | |+--+--+--+--+| | | |+--+--+--+ (如果显示有问题,也可以参 ...

  9. 第七届蓝桥杯试题c/c++A组方格填数 回溯法

    方格填数如下的10个格子   +--+--+--+   |  |  |  |+--+--+--+--+|  |  |  |  |+--+--+--+--+|  |  |  |+--+--+--+(如果 ...

随机推荐

  1. freemarker1 一些内建函数和用法

    ${"   green mouse"?cap_first} -->   Green mouse  //字符串中的第一个单词的首字母大写 ${"ABCDF" ...

  2. Laravel5.1 关联模型之后操作

    之前写过关于模型关联的笔记,但是模型关联好后的一些使用没有介绍,今天补上 1 写入关联模型 1.1 使用Save方法(一对多) 我们准备了两个模型:Post和Comment. 它们的关系是一对多关系. ...

  3. Linux 文件夹含义(转)

    1./bin :获得最小的系统可操作性所需要的命令 2./boot :内核和加载内核所需的文件 3./dev :终端.磁盘.调制解调器等的设备项 4./etc :关键的启动文件和配置文件 5./hom ...

  4. 【BZOJ1412】[ZJOI2009]狼和羊的故事 最小割

    [BZOJ1412][ZJOI2009]狼和羊的故事 Description “狼爱上羊啊爱的疯狂,谁让他们真爱了一场:狼爱上羊啊并不荒唐,他们说有爱就有方向......” Orez听到这首歌,心想: ...

  5. 面试之三:JVM类加载机制-类加载各阶段说明和类加载器

    一.类生命周期:共7个阶段 类从被加载到虚拟机内存中开始,到卸载出内存.整个生命周期包括:加载.验证.准备.解析.初始化.使用和卸载7个阶段. 其中验证.准备.解析3个部分统称为连接. 类加载的过程: ...

  6. 使用Git分支开发新特性或修复Bug与使用Git分支开发新特性或修复Bug

    使用Git分支开发新特性或修复Bug 通过分支,可以在不影响原有代码的前提下改变代码,主要用于开发新功能新特性.下 一代产品 为已经发布的正式版修复bug 团队开发时为每个人建立一个分支,从而避免相互 ...

  7. C语言实现双链表(带头节点)

    双链表和单链表性质相似只是在多加了一个前指针 1.定义结构体 typedef struct Node{ int data; struct Node *prior; struct Node *next; ...

  8. TCP requires two packet transfers to set up the connection before it can send data

    wHTTP重用现存连接来减少TCP建立时延. HTTP The Definitive Guide 4.2.3 TCP Connection Handshake Delays When you set ...

  9. 原!mysql存储过程 批量导入数据

    mysql需要导入某前缀例如12345为前缀的,后缀扩展2位 即00-99. 利用存储过程插入数据. DROP PROCEDURE IF EXISTS insert_popsms_code;DELIM ...

  10. JS获取浏览器信息及屏幕分辨率

    因为vue有自己的生命周期,初始化数据的时候,可以在钩子函数created()函数里初始化数据,也可以在mounted()函数里获取,但是两者是不同的,获取浏览器和屏幕分辨率的时候,不能在create ...