【BZOJ4823】[CQOI2017]老C的方块(网络流)

题面

BZOJ

题解

首先还是给棋盘进行黑白染色,然后对于特殊边左右两侧的格子单独拎出来考虑。

为了和其他格子区分,我们把两侧的这两个格子染成灰色。

于是一个不合法的状态就是两个相邻的灰色点如果还和一个其他的点相连就是非法的。

我们先把黑白点分开,源点连向黑点,汇点连向白点,边权是删去这个点的代价。

因为灰点可以两两配对,非两两配对之间的没有影响,然后所有黑点连向对应的灰点,另一半灰点连向对应的白点。而要删去一组不合法的要么删去一个灰点,要么删去所有黑点或者所有白点,而连向源汇已经处理了删去所有黑点或者白点,所以只需要考虑删去一个灰点,显然删哪个都是一样的所以两个匹配的灰点之间连一条权值为两个点权值较小值的边。

这样子最小割就是答案。

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<vector>
  4. #include<queue>
  5. #include<map>
  6. using namespace std;
  7. #define MAX 100100
  8. const int inf=2e9;
  9. inline int read()
  10. {
  11. int x=0;bool t=false;char ch=getchar();
  12. while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
  13. if(ch=='-')t=true,ch=getchar();
  14. while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
  15. return t?-x:x;
  16. }
  17. struct Line{int v,next,w;}e[MAX*20];
  18. int h[MAX],cnt=2;
  19. inline void Add(int u,int v,int w)
  20. {
  21. e[cnt]=(Line){v,h[u],w};h[u]=cnt++;
  22. e[cnt]=(Line){u,h[v],0};h[v]=cnt++;
  23. }
  24. int S,T,cur[MAX],level[MAX];
  25. queue<int> Q;
  26. bool bfs()
  27. {
  28. for(int i=S;i<=T;++i)level[i]=0;
  29. Q.push(S);level[S]=1;
  30. while(!Q.empty())
  31. {
  32. int u=Q.front();Q.pop();
  33. for(int i=h[u];i;i=e[i].next)
  34. if(e[i].w&&!level[e[i].v])
  35. level[e[i].v]=level[u]+1,Q.push(e[i].v);
  36. }
  37. return level[T];
  38. }
  39. int dfs(int u,int flow)
  40. {
  41. if(u==T||!flow)return flow;
  42. int ret=0;
  43. for(int &i=cur[u];i;i=e[i].next)
  44. {
  45. int v=e[i].v,d;
  46. if(e[i].w&&level[v]==level[u]+1)
  47. {
  48. d=dfs(v,min(flow,e[i].w));
  49. ret+=d;flow-=d;
  50. e[i].w-=d;e[i^1].w+=d;
  51. if(!flow)break;
  52. }
  53. }
  54. if(!ret)level[u]=0;
  55. return ret;
  56. }
  57. int Dinic()
  58. {
  59. int ret=0;
  60. while(bfs())
  61. {
  62. for(int i=S;i<=T;++i)cur[i]=h[i];
  63. ret+=dfs(S,inf);
  64. }
  65. return ret;
  66. }
  67. map<pair<int,int>,int> M;
  68. int col[MAX];
  69. int d[4][2]={1,0,0,1,-1,0,0,-1};
  70. int C,R,n,X[MAX],Y[MAX],W[MAX];
  71. bool check(int x,int y)
  72. {
  73. if(x&1)return y%4==1||y%4==2;
  74. else return y%4==3||y%4==0;
  75. }
  76. int main()
  77. {
  78. R=read();C=read();n=read();
  79. for(int i=1;i<=n;++i)
  80. {
  81. Y[i]=read();X[i]=read();W[i]=read();
  82. M[make_pair(X[i],Y[i])]=i;
  83. if((X[i]+Y[i]+1)&1)col[i]=check(X[i],Y[i])?3:1;
  84. else col[i]=check(X[i],Y[i])?2:0;
  85. }
  86. S=0;T=n+1;
  87. for(int i=1;i<=n;++i)
  88. if(col[i]==0)Add(i,T,W[i]);
  89. else if(col[i]==1)Add(S,i,W[i]);
  90. else if(col[i]==2)
  91. for(int j=0;j<4;++j)
  92. {
  93. int x=X[i]+d[j][0],y=Y[i]+d[j][1];
  94. if(M.find(make_pair(x,y))==M.end())continue;
  95. int d=M[make_pair(x,y)];
  96. if(col[d]==1)Add(d,i,inf);
  97. else if(col[d]==3)Add(i,d,min(W[d],W[i]));
  98. }
  99. else if(col[i]==3)
  100. for(int j=0;j<4;++j)
  101. {
  102. int x=X[i]+d[j][0],y=Y[i]+d[j][1];
  103. if(M.find(make_pair(x,y))==M.end())continue;
  104. int d=M[make_pair(x,y)];
  105. if(col[d]==0)Add(i,d,inf);
  106. }
  107. printf("%d\n",Dinic());
  108. return 0;
  109. }

【BZOJ4823】[CQOI2017]老C的方块(网络流)的更多相关文章

  1. bzoj4823: [Cqoi2017]老C的方块(最小割)

    4823: [Cqoi2017]老C的方块 题目:传送门 题解: 毒瘤题ORZ.... 太菜了看出来是最小割啥边都不会建...狂%大佬强强强   黑白染色?不!是四个色一起染,四层图跑最小割... 很 ...

  2. BZOJ4823 CQOI2017老C的方块(最小割)

    如果将其转化为一个更一般的问题即二分图带权最小单边点覆盖(最小控制集)感觉是非常npc的.考虑原题给的一大堆东西究竟有什么奇怪的性质. 容易发现如果与特殊边相邻的两格子都放了方块,并且这两个格子都各有 ...

  3. BZOJ4823 [Cqoi2017]老C的方块 【最小割】

    题目 老C是个程序员. 作为一个懒惰的程序员,老C经常在电脑上玩方块游戏消磨时间.游戏被限定在一个由小方格排成的R行C列网格上,如果两个小方格有公共的边,就称它们是相邻的,而且有些相邻的小方格之间的公 ...

  4. [bzoj4823][Cqoi2017]老C的方块

    来自FallDream的博客,未经允许,请勿转载,谢谢. 挺有意思的一道题.... 看完题面比较明确是最小割,考虑怎么建图 想了比较久 突破口应该是题目中那张奇怪的图 观察这个奇怪的图和方块,很容易发 ...

  5. bzoj千题计划300:bzoj4823: [Cqoi2017]老C的方块

    http://www.lydsy.com/JudgeOnline/problem.php?id=4823 讨厌的形状就是四联通图 且左右各连一个方块 那么破坏所有满足条件的四联通就好了 按上图方式染色 ...

  6. [CQOI2017]老C的方块 网络流

    ---题面--- 题解: 做这题做了好久,,,换了4种建图QAQ 首先我们观察弃疗的形状,可以发现有一个特点,那就是都以一个固定不变的特殊边为中心的,如果我们将特殊边两边的方块分别称为s块和t块, 那 ...

  7. BZOJ 4823 [Cqoi2017]老C的方块 ——网络流

    lrd的题解:http://www.cnblogs.com/liu-runda/p/6695139.html 我还是太菜了.以后遇到这种题目应该分析分析性质的. 网络流复杂度真是$O(玄学)$ #in ...

  8. 洛谷$P3756\ [CQOI2017]$老$C$的方块 网络流

    正解:网络流 解题报告: 传送门$QwQ$ 看到不能出现给定的讨厌的图形,简单来说就,特殊边两侧的方格不能同时再连方格. 所以如果出现,就相当于是四种方案?就分别炸四个格子. 然后冷静分析一波之后发现 ...

  9. bzoj 4823 [Cqoi2017]老C的方块——网络流

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4823 一个不合法方案其实就是蓝线的两边格子一定选.剩下两部分四相邻格子里各选一个. 所以这个 ...

  10. BZOJ 4823 Luogu P3756 [CQOI2017]老C的方块 (网络流、最小割)

    题目链接 (Luogu) https://www.luogu.org/problem/P3756 (BZOJ) http://lydsy.com/JudgeOnline/problem.php?id= ...

随机推荐

  1. java全栈项目

    文档地址:https://course.7yue.pro/lin/sleeve/ http://talelin.unna.com.cn/ 1.小程序里,我把结构分为三部分:wxml(view).pag ...

  2. if(response.isSuccess){}else{}的方式,如果我们由于忽略没有设置success字段的值,就可能导致

    在日常开发中,我们会经常要在类中定义布尔类型的变量,比如在给外部系统提供一个RPC接口的时候,我们一般会定义一个字段表示本次请求是否成功的. 关于这个”本次请求是否成功”的字段的定义,其实是有很多种讲 ...

  3. Django信号机制相关解释与示例

    Django 信号# django自带一套信号机制来帮助我们在框架的不同位置之间传递信息.也就是说,当某一事件发生时,信号系统可以允许一个或多个发送者(senders)将通知或信号(signals)发 ...

  4. Markdown数学公式语法

    详细网址:Markdown数学公式语法

  5. 【RTOS】基于V7开发板的最新版FreeRTOS V10.2.0程序模板,含MDK和IAR,支持串口打印任务执行情况

    模板下载: 链接:https://pan.baidu.com/s/1N32Hx7cTbDoRinuzTUB3zw   提取码:6aox 1.MDK使用MDK5.26及其以上版本. 2.IAR使用IAR ...

  6. UTXO和Account模型一个都不能少

    UTXO对于非区块链从业人员来说可能比较陌生,UTXO的全称是Unspent Transaction Output,这中本聪在比特币中的一个天才设计.而Account模型就很常见,也很容易理解,你银行 ...

  7. 当h5页面图片加载失败后,给定一个默认图

    本文主要讨论页面中图片加载失败后替换默认图片的几种方式 重点来了:一定要记住error事件不冒泡. 相关的知识点:jquery的ready方法.$("img").error().i ...

  8. javaWeb核心技术第七篇之HTTP、Tomcat、Servlet、Request和Response

    - Web服务器 - 概念: - web资源: "英文直译"网"的意思 资源:一切数据文件 web资源:通过网络可以访问到的资源,通常指的是一切放在服务器上的文件&quo ...

  9. Oracle merge into的优势

    简介 Oracle merge into命令,顾名思义就是“有则更新,无则插入”,这个也是merge into 命令的核心思想,在实际开发过程中,我们会经常遇到这种通过两表互相关联匹配更新其中一个表的 ...

  10. [转]UiPath: How to Capture a Mouse Event on Hover Menus?

    本文转自:https://www.uipath.com/kb-articles/how-to-capture-mouse-event-on-hover-menus he Knowledgebase a ...