题目链接  C.Butterfly

令$fd[i][j]$为以$s[i][j]$为起点开始往下走最大连续的‘X’个数

令$fl[i][j]$为以$s[i][j]$为起点开始往左下走最大连续的‘X’个数

令$fr[i][j]$为以$s[i][j]$为起点开始往左下走最大连续的‘X’个数

令$a[i][j] = min(fd[i][j], fl[i][j])$, $b[i][j] = min(fd[i][j], fr[i][j])$

对于每一个$(i, j)$, 我们要找到$(i, k)$

使得$k$最大,并且$k >= j$, $k - j + 1 <= min(a[i][j], b[i][k])$

$k - j + 1$必须为奇数

把所有条件整理出来就是

$k - j + 1 <= a[i][j]$

$k - j + 1 <= b[i][k]$

移项得到

$k <= a[i][j] + j - 1$

$k - b[i][k] + 1 <= j$

所以我们可以对$k - b[i][k] + 1$排序,当$j$从$1$扫到$m$的时候每一次确保所有满足$k - b[i][k] + 1$的$k$都已经被添加到线段树中

那么我们在$[j, a[i][j] + j - 1]$这段区间里面查找最大值(也就是符合条件的最大的$k$)更新答案即可。

注意分奇偶讨论

#include <bits/stdc++.h>

using namespace std;

#define rep(i, a, b)	for (int i(a); i <= (b); ++i)
#define dec(i, a, b) for (int i(a); i >= (b); --i)
#define fi first
#define se second
#define MP make_pair
#define lson i << 1, L, mid
#define rson i << 1 | 1, mid + 1, R typedef pair <int, int> PII; const int N = 2010; char s[N][N];
int fd[N][N], fl[N][N], fr[N][N], a[N][N], b[N][N];
int n, m, now;
int t[N << 4];
int cnt, mx, ans = 0; PII c[N]; inline void pushup(int i){ t[i] = max(t[i << 1], t[i << 1 | 1]); } void update(int i, int L, int R, int x, int val){
if (L == R && L == x){
t[i] = max(t[i], val);
return;
} int mid = (L + R) >> 1;
if (x <= mid) update(lson, x, val);
else update(rson, x, val);
pushup(i);
} int query(int i, int L, int R, int l, int r){
if (l <= L && R <= r) return t[i];
int mid = (L + R) >> 1;
if (r <= mid) return query(lson, l, r);
else if (l > mid) return query(rson, l, r);
else return max(query(lson, l, r), query(rson, l, r));
} int main(){ scanf("%d%d", &n, &m);
rep(i, 1, n) scanf("%s", s[i] + 1);
dec(i, n, 1) rep(j, 1, m){
fd[i][j] = s[i][j] == 'X' ? fd[i + 1][j] + 1 : 0;
fl[i][j] = s[i][j] == 'X' ? fl[i + 1][j - 1] + 1 : 0;
fr[i][j] = s[i][j] == 'X' ? fr[i + 1][j + 1] + 1 : 0;
} rep(i, 1, n) rep(j, 1, m) a[i][j] = min(fd[i][j], fr[i][j]), b[i][j] = min(fd[i][j], fl[i][j]); mx = m << 1;
ans = 0; rep(i, 1, n){
cnt = 0;
for (int j = 1; j <= m; j += 2) if (b[i][j]) c[++cnt] = MP(j - b[i][j] + 1, j);
sort(c + 1, c + cnt + 1); memset(t, 0, sizeof t);
now = 0;
for (int j = 1; j <= m; j += 2){
if (!a[i][j]) continue;
while (now < cnt){
if (c[now + 1].fi <= j){
++now;
update(1, 1, mx, c[now].se, c[now].se);
if (now >= cnt) break;
}
else break;
} int et = query(1, 1, mx, j, a[i][j] + j - 1);
ans = max(ans, et - j + 1);
} cnt = 0;
for (int j = 2; j <= m; j += 2) if (b[i][j]) c[++cnt] = MP(j - b[i][j] + 1, j);
sort(c + 1, c + cnt + 1); memset(t, 0, sizeof t);
now = 0;
for (int j = 2; j <= m; j += 2){
if (!a[i][j]) continue;
while (now < cnt){
if (c[now + 1].fi <= j){
++now;
update(1, 1, mx, c[now].se, c[now].se);
if (now >= cnt) break;
}
else break;
} int et = query(1, 1, mx, j, a[i][j] + j - 1);
ans = max(ans, et - j + 1);
}
} printf("%d\n", ans);
return 0;
}

  

Wannafly挑战赛2 C.Butterfly(线段树优化枚举)的更多相关文章

  1. [USACO2005][POJ3171]Cleaning Shifts(DP+线段树优化)

    题目:http://poj.org/problem?id=3171 题意:给你n个区间[a,b],每个区间都有一个费用c,要你用最小的费用覆盖区间[M,E] 分析:经典的区间覆盖问题,百度可以搜到这个 ...

  2. Weak Pair---hud5877大连网选(线段树优化+dfs)

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=5877  题意:给你一颗树,有n个节点,每个节点都有一个权值v[i]:现在求有多少对(u,v ...

  3. CodeForces 558E(计数排序+线段树优化)

    题意:一个长度为n的字符串(只包含26个小字母)有q次操作 对于每次操作 给一个区间 和k k为1把该区间的字符不降序排序 k为0把该区间的字符不升序排序 求q次操作后所得字符串 思路: 该题数据规模 ...

  4. HDU4719-Oh My Holy FFF(DP线段树优化)

    Oh My Holy FFF Time Limit: 5000/2500 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others) T ...

  5. UOJ#77. A+B Problem [可持久化线段树优化建边 最小割]

    UOJ#77. A+B Problem 题意:自己看 接触过线段树优化建图后思路不难想,细节要处理好 乱建图无果后想到最小割 白色和黑色只能选一个,割掉一个就行了 之前选白色必须额外割掉一个p[i], ...

  6. Codeforces 558E A Simple Task (计数排序&&线段树优化)

    题目链接:http://codeforces.com/contest/558/problem/E E. A Simple Task time limit per test5 seconds memor ...

  7. 2019.03.09 codeforces833B. The Bakery(线段树优化dp)

    传送门 线段树优化dpdpdp入门题. 要求把nnn个数分成kkk段,每段价值为里面不相同的数的个数,求所有段的价值之和最大值.n≤35000,k≤50n\le35000,k\le50n≤35000, ...

  8. Codeforces 1045. A. Last chance(网络流 + 线段树优化建边)

    题意 给你 \(n\) 个武器,\(m\) 个敌人,问你最多消灭多少个敌人,并输出方案. 总共有三种武器. SQL 火箭 - 能消灭给你集合中的一个敌人 \(\sum |S| \le 100000\) ...

  9. bzoj千题计划311:bzoj5017: [Snoi2017]炸弹(线段树优化tarjan构图)

    https://www.lydsy.com/JudgeOnline/problem.php?id=5017 暴力: 对于每一个炸弹,枚举所有的炸弹,看它爆炸能不能引爆那个炸弹 如果能,由这个炸弹向引爆 ...

随机推荐

  1. webpack4搭建Vue开发环境笔记~~持续更新

    项目git地址 一.node知识 __dirname: 获取当前文件所在路径,等同于path.dirname(__filename) console.log(__dirname); // Prints ...

  2. python并发编程之线程剩余内容(线程队列,线程池)及协程

    1. 线程的其他方法 import threading import time from threading import Thread,current_thread def f1(n): time. ...

  3. POJ:2060-Taxi Cab Scheme(最小路径覆盖)

    传送门:http://poj.org/problem?id=2060 Taxi Cab Scheme Time Limit: 1000MS Memory Limit: 30000K Total Sub ...

  4. Linux网络文件系统NFS详解

    什么是文件系统,NFS文件系统又是什么? 简单的说,文件系统就是通过软件对磁盘上的数据进行组织和管理的一种机制,对其的一种封装或透视. 你女朋友拍了美美的暧昧照片,放一个文件夹里发送给了A服务器,当你 ...

  5. 64位程序调用32DLL解决方案

    最近做一个.NETCore项目,需要调用以前用VB6写的老程序,原本想重写,但由于其调用了大量32DLL,重写后还需要编译为32位才能运行,于是干脆把老代码整个封装为32DLL,然后准备在64位程序中 ...

  6. 转:Generating PDFs from Web Pages on the Fly with jsPDF

    The Portable Document Format has been one the major innovations in the fields of desktop publishing ...

  7. VC6.0与Office2007~2010不兼容问题及解决方法

    一.问题描述 启动打开文件对话框中,在 Visual C++ 使用的键盘快捷键或从文件菜单上将导致以下错误: 在 DEVSHL 中的访问冲突 (0xC0000005).在 0x5003eaed 的 D ...

  8. [python学习篇][廖雪峰][1]高级特性--创建生成器 方法2 yield

    def fib(max): n, a, b = 0, 0, 1 while n < max: print b a, b = b, a + b n = n + 1 将print b 改成yield ...

  9. [linux time命令学习篇] time 统计命令执行的时间

    注意: 命令后面一定要有分号; http://codingstandards.iteye.com/blog/798788 用途说明 time命令常用于测量一个命令的运行时间,注意不是用来显示和修改系统 ...

  10. 九度oj 题目1496:数列区间

    题目描述: 有一段长度为n(1<=n<=1000000)的数列,数列中的数字从左至右从1到n编号.初始时数列中的数字都是0. 接下来我们会对其进行m(1<=m<=100000) ...