[atARC119D]Grid Repainting 3
将每一行和每一列分别作为一个点,当第$i$行第$j$列的格子为红色时,将第$i$行与第$j$列连边
此时,考虑选择第$i$行的红色格子并将第$i$行的格子全部改成白色:
关于这一操作的条件,即需要第$i$行有红色格子,从图中来看也即第$i$行对应的点度非0
关于这一条件的影响,即第$i$行的红色格子都没了,从图中来看也即删去第$i$行对应的点所有出边
根据上述分析,每一次操作即在图中选择一个度非0的点,并删除其所有出边,并且设最终选择了$x$个行对应的点和$y$个列对应的点,最大化$nm-(n-x)(m-y)$
考虑其中的一个连通块(连通块之间显然是独立的),我们可以仅保留这个连通块中的任意一个点,并选择该连通块中其余的点
具体来说,任取该连通块的一个生成树,并以需要保留的点为根,每一次不断选择一个叶子即可
另一方面,显然无论如何我们都不可能删除一个连通块中的所有点,因此这已经最优了
同时,我们仅关心每一个连通块中未选的点是行还是列,一个点的连通块显然不需要考虑,多个点的连通块必然同时含有行和列的点
由此,求出所有连通块(不考虑一个点的连通块)的数量$k$以及$x-1$和$y-1$之和$X$和$Y$(其中$x$和$y$为该连通块中行和列对应的点数量),问题即求$f(x)=nm-(n-X-x)(m-Y-k+x)$在$x\in [0,k]$的最大值,利用二次函数简单讨论即可
方案根据上面的分析也不难构造,时间复杂度为$o(nm)$,可以通过

1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 2505
4 struct Edge{
5 int nex,to;
6 }edge[N*N*2];
7 struct Op{
8 int type,x,y;
9 };
10 vector<int>v[N<<1];
11 vector<Op>ans;
12 int E,n,m,x,y,scc,head[N<<1],vis[N<<1];
13 char s[N][N];
14 void add(int x,int y){
15 edge[E].nex=head[x];
16 edge[E].to=y;
17 head[x]=E++;
18 }
19 void dfs(int k){
20 if (vis[k])return;
21 vis[k]=1;
22 v[scc].push_back(k);
23 for(int i=head[k];i!=-1;i=edge[i].nex)dfs(edge[i].to);
24 }
25 void dfs(int k,int fa){
26 if (vis[k])return;
27 vis[k]=1;
28 for(int i=head[k];i!=-1;i=edge[i].nex)dfs(edge[i].to,k);
29 if (fa>=0){
30 if (k<n)ans.push_back(Op{0,k,fa-n});
31 else ans.push_back(Op{1,fa,k-n});
32 }
33 }
34 void write(){
35 printf("%d\n",ans.size());
36 for(int i=0;i<ans.size();i++){
37 ans[i].x++,ans[i].y++;
38 if (ans[i].type==0)printf("X %d %d\n",ans[i].x,ans[i].y);
39 else printf("Y %d %d\n",ans[i].x,ans[i].y);
40 }
41 }
42 int main(){
43 scanf("%d%d",&n,&m);
44 memset(head,-1,sizeof(head));
45 for(int i=0;i<n;i++){
46 scanf("%s",s[i]);
47 for(int j=0;j<m;j++)
48 if (s[i][j]=='R'){
49 add(i,j+n);
50 add(j+n,i);
51 }
52 }
53 for(int i=0;i<n+m;i++)
54 if (!vis[i]){
55 v[++scc].clear();
56 dfs(i);
57 if (v[scc].size()==1)scc--;
58 else{
59 x--,y--;
60 for(int j=0;j<v[scc].size();j++){
61 if (v[scc][j]<n)x++;
62 else y++;
63 }
64 }
65 }
66 memset(vis,0,sizeof(vis));
67 if (m-n+x-y<0){
68 for(int i=1;i<=scc;i++)
69 for(int j=0;j<v[i].size();j++)
70 if (v[i][j]<n){
71 dfs(v[i][j],-1);
72 break;
73 }
74 write();
75 }
76 else{
77 for(int i=1;i<=scc;i++)
78 for(int j=0;j<v[i].size();j++)
79 if (v[i][j]>=n){
80 dfs(v[i][j],-1);
81 break;
82 }
83 write();
84 }
85 }
[atARC119D]Grid Repainting 3的更多相关文章
- AtCoder Beginner Contest 088 D Grid Repainting
Problem statement We have an H×W grid whose squares are painted black or white. The square at the i- ...
- ExtJS 4.2 Grid组件的单元格合并
ExtJS 4.2 Grid组件本身并没有提供单元格合并功能,需要自己实现这个功能. 目录 1. 原理 2. 多列合并 3. 代码与在线演示 1. 原理 1.1 HTML代码分析 首先创建一个Grid ...
- WPF中Grid实现网格,表格样式通用类
/// <summary> /// 给Grid添加边框线 /// </summary> /// <param name="grid"></ ...
- 在 Windows Phone 中,为 Grid 添加 Tilt 效果
在 Windows Phone 中,Tilt 效果是比较经典的效果,我们可以很简单的为按钮等控件添加这样的效果(使用 Windows Phone Toolkit 的Tilt 效果),但是,如果我们想要 ...
- wpf 列表、菜单 收起与展开,通过Grid DoubleAnimation或者Expander实现
菜单收缩有很多种方法具体如何实现还是看个人想法: 第一种通过后台控制收起与展开: 效果图: 代码 : <Grid> <Grid.ColumnDefinitions> <C ...
- Sencha ExtJS 6 Widget Grid 入门
最近由于业务需要,研究了一下Sencha ExtJS 6 ,虽然UI和性能上据相关资料说都有提升,但是用起来确实不太顺手,而且用Sencha cmd工具进行测试和发布,很多内部细节都是隐藏的,出了问题 ...
- WPF CheckBox样式 ScrollViewer样式 WrapPanel、StackPanel、Grid布局
本节讲述布局,顺带加点样式给大家看看~单纯学布局,肯定是枯燥的~哈哈 那如上界面,该如何设计呢? 1.一些布局元素经常用到.Grid StackPanel Canvas WrapPanel等.如上这种 ...
- [转]ExtJS Grid 分页时保持选中的简单实现方法
原文地址 :http://www.qeefee.com/article/ext-grid-keep-paging-selection ExtJS中经常要用到分页和选择,但是当选择遇到分页的时候,杯具就 ...
- [转]extjs grid的Ext.grid.CheckboxSelectionModel默认选中解决方法
原文地址:http://379548695.iteye.com/blog/1167234 grid的复选框定义如下: var sm = new Ext.grid.CheckboxSelection ...
随机推荐
- HPE ProLiant 系列服务器Microsoft Windows 2008 R2系统下网卡绑定方法
HPE Network Configuration Utility(以下简称NCU) 网卡绑定工具,用户可以通过该工具很方便的把服务器的多个网卡捆绑到一起以达到容错和增加可用带宽的目的. 1.打开NC ...
- SpringBoot+WebSocket实时监控异常
写在前面 此异常非彼异常,标题所说的异常是业务上的异常. 最近做了一个需求,消防的设备巡检,如果巡检发现异常,通过手机端提交,后台的实时监控页面实时获取到该设备的信息及位置,然后安排员工去处理. 因为 ...
- 题解 「BZOJ2178」圆的面积并
题目传送门 题目大意 给出 \(n\) 个圆,求它们并的面积大小. \(n\le 10^3\) 思路 如果您不会自适应辛普森法,请戳这里学习 其实我们发现,如果我们设 \(f(x)\) 表示 \(x= ...
- SpringBoot配置文件application
配置文件 SpringBoot使用一个全局的配置文件 , 配置文件名称是固定的,有两种文件格式: application.properties 语法结构 :key=value application. ...
- Java泛型背后是什么?
文Java中泛型的应用,让大家更好地理解泛型,以及常说的泛型类型擦除是什么概念,举一个简单的例子,如下: 这里可以看出来在代码编写阶段就已经报错了,不能往string类型的集合中添加int类型的数据. ...
- Go语言核心36讲(Go语言进阶技术四)--学习笔记
10 | 通道的基本操作 作为 Go 语言最有特色的数据类型,通道(channel)完全可以与 goroutine(也可称为 go 程)并驾齐驱,共同代表 Go 语言独有的并发编程模式和编程哲学. D ...
- WEB安全指南
说明:本文是Mozilla Web应用部署文档,对运维或者后端开发团队的部署行为进行指导.该部署安全规范内容充实,对于部署有很大意义.同时也涉及到了许多web前端应用安全的基本知识,如CSP, TOK ...
- 80. 删除有序数组中的重复项 II
题目 给你一个有序数组 nums ,请你原地删除重复出现的元素(不需要考虑数组中超出新长度后面的元素),使每个元素最多出现两次 ,返回删除后数组的新长度. 不要使用额外的数组空间,你必须在原地修改输入 ...
- 看动画学算法之:队列queue
目录 简介 队列的实现 队列的数组实现 队列的动态数组实现 队列的链表实现 队列的时间复杂度 简介 队列Queue是一个非常常见的数据结构,所谓队列就是先进先出的序列结构. 想象一下我们日常的排队买票 ...
- [软工顶级理解组] Beta阶段事后分析
目录 设想和目标 计划 资源 变更管理 设计/实现 测试/发布 团队的角色,管理,合作 总结 质量提高 会议截图 设想和目标 我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰 ...