题意:一个无限大的棋盘, 有n个小兵, 给出了n个兵的坐标, 现在有一个长为width 高为height的炸弹放在棋盘上, 炸弹只能上下左右平移, 不能旋转。

且放炸弹的区域不能含有士兵, 炸弹可以一次炸掉与它同一行同一列的所有士兵。 放一颗炸弹, 使得炸死的士兵最多。输出最大值。

思路:先不考虑离散化, 可以计算出水平方向和竖直方向上所有长度为width和height区间内士兵的个数, 得到一个数组prefixX, prefixY。

然后一次遍历prefixY数组, 假设区间[i, i+height-1]对应prefixY[i], 而且[i, i+height-1]内所有士兵的x坐标对应的prefixX都减去一个大于n的数字MAX(这样保证炸弹不会放在士兵上),这个时候求prefixX上的一个最大值max再加上prefixY[i]就可以更新答案result了, 如果max是一个负数那么说明当前这个区间不能放炸弹。 之后再把第i行的所有士兵的x坐标对应的prefixX都加上MAX。 这样一次下去就可以了。线段树区间操作。

还有一个需要特判的地方是 上面并没有找到一个可以放炸弹的地方。 这个时候答案就是prefixX和prefixY的最大值。

思路一下子就想出来了,可是怎么离散化想了好久。。

 #include <bits/stdc++.h>
using namespace std;
const int maxn = 2e5 + ;
struct SegTree{
int seg[maxn << ], lazy[maxn << ];
void build(int l, int r, int pos, int val[]){
lazy[pos] = ;
if (l == r){
seg[pos] = val[l];
return;
}
int mid = (l + r) >> ;
build(l, mid, pos<<, val);
build(mid+, r, pos<<|, val);
seg[pos] = max(seg[pos<<], seg[pos<<|]);
}
void push_down(int pos){
if (lazy[pos]){
seg[pos<<] += lazy[pos];
seg[pos<<|] += lazy[pos];
lazy[pos<<] += lazy[pos];
lazy[pos<<|] += lazy[pos];
lazy[pos] = ;
}
}
void update (int l, int r, int pos, int ua, int ub, int val){
if (ua > ub){
return;
}
if (ua <= l && ub >= r){
seg[pos] += val;
lazy[pos] += val;
return;
}
push_down(pos);
int mid = (l + r) >> ;
if (ua <= mid){
update(l, mid, pos<<, ua, ub, val);
}
if (ub > mid){
update(mid+, r, pos<<|, ua, ub, val);
}
seg[pos] = max(seg[pos<<], seg[pos<<|]);
}
}tree;
int pawX[maxn], pawY[maxn];
int lshX[maxn], lshY[maxn], tot1, tot2;
int prefixX[maxn], prefixY[maxn], valX[maxn], valY[maxn];
vector <int> vec1[maxn], vec2[maxn];
void init(){
memset(prefixX, , sizeof prefixX);
memset(prefixY, , sizeof prefixY);
for (int i = ; i < maxn; i++){
vec1[i].clear();
vec2[i].clear();
}
}
int main(){
int T, cas = ;
scanf ("%d", &T);
while (T--){
int n, width, height;
scanf ("%d%d%d", &n, &width, &height);
tot1 = tot2 = ;
init();
for (int i = ; i < n; i++){
int x, y;
scanf ("%d%d", &x, &y);
pawX[i] = x, pawY[i] = y;
lshX[tot1++] = x;
lshX[tot1++] = x - width+;
lshY[tot2++] = y;
lshY[tot2++] = y-height+;
}
sort(lshX, lshX+tot1);
sort(lshY, lshY+tot2);
tot1 = unique(lshX, lshX+tot1) - lshX;
tot2 = unique(lshY, lshY+tot2) - lshY;
for (int i = ; i < n; i++){
int x1 = lower_bound(lshX, lshX+tot1, pawX[i]-width+) - lshX + ;
int x2 = lower_bound(lshX, lshX+tot1, pawX[i]) - lshX + ;
int y1 = lower_bound(lshY, lshY+tot2, pawY[i]-height+) - lshY + ;
int y2 = lower_bound(lshY, lshY+tot2, pawY[i]) - lshY + ;
prefixX[x1]++; prefixX[x2+]--;
prefixY[y1]++; prefixY[y2+]--;
vec1[y1].push_back(i);
vec2[y2].push_back(i);
}
int res = ;
for (int i = ; i <= tot1; i++){
prefixX[i] += prefixX[i-];
res = max(res, prefixX[i]); // 这里要特判一下
}
for (int i = ; i <= tot2; i++){
prefixY[i] += prefixY[i-];
res = max(res, prefixY[i]); // 这里要特判一下
}
tree.build(, tot1, , prefixX);
const int tenThousand = 1e5;
for (int i = ; i <= tot2; i++){
for (int j = ; j < vec1[i].size(); j++){
int idx = vec1[i][j];
int x = lower_bound(lshX, lshX+tot1, pawX[vec1[i][j]]) - lshX + ;
int y = lower_bound(lshX, lshX+tot1, pawX[vec1[i][j]]-width+) - lshX + ;
tree.update(, tot1, , y, x, -tenThousand);
}
int ret = tree.seg[];
if (ret > )
res = max(res, prefixY[i]+ret);
for (int j = ; j < vec2[i].size(); j++){
int x = lower_bound(lshX, lshX+tot1, pawX[vec2[i][j]]) - lshX + ;
int y = lower_bound(lshX, lshX+tot1, pawX[vec2[i][j]]-width+) - lshX + ;
tree.update(, tot1, , y, x, tenThousand);
}
}
printf("Case #%d: %d\n", cas++, res);
}
return ;
}

UVAlive7141 BombX 14年上海区域赛D题 线段树+离散化的更多相关文章

  1. UVALive 7148 LRIP 14年上海区域赛K题 树分治

    题意 n个点组成一棵树, 带有点权. 求最长不降的路径的长度, 且路径上最大值最小值之差不超过D. 显然是树分治, 但是分治之后如何维护答案呢. 假设当前重心为g, 分别记录g出发不降路径的长度,以及 ...

  2. hdu 5475 模拟计算器乘除 (2015上海网赛H题 线段树)

    给出有多少次操作 和MOD 初始值为1 操作1 y 表示乘上y操作2 y 表示除以第 y次操作乘的那个数 线段树的叶子结点i 表示 第i次操作乘的数 将1替换成y遇到操作2 就把第i个结点的值 替换成 ...

  3. ACM 2015年上海区域赛A题 HDU 5572An Easy Physics Problem

    题意: 光滑平面,一个刚性小球,一个固定的刚性圆柱体 ,给定圆柱体圆心坐标,半径 ,小球起点坐标,起始运动方向(向量) ,终点坐标 ,问能否到达终点,小球运动中如果碰到圆柱体会反射. 学到了向量模板, ...

  4. UVALive 8519 Arrangement for Contests 2017西安区域赛H 贪心+线段树优化

    题意 等价于给一个数列,每次对一个长度为$K$的连续区间减一 为最多操作多少次 题解: 看样例猜的贪心,10分钟敲了个线段树就交了... 从1开始,找$[i,i+K]$区间的最小值,然后区间减去最小值 ...

  5. hdu 4031 2011成都赛区网络赛A题 线段树 ***

    就是不知道时间该怎么处理,想了好久,看了别人的题解发现原来是暴力,暴力也很巧妙啊,想不出来的那种  -_-! #include<cstdio> #include<iostream&g ...

  6. HDU - 6521 Party (SYSU校赛K题)(线段树)

    题目链接 题意:n个人排成一列,一开始他们互不认识,每次选[l,r]上的人开party,使他们互相认识,求出每次party之后新互相认识的人的对数. 思路:把“互相认识”变成单向连边,只考虑左边的人对 ...

  7. 2019 ICPC 上海区域赛总结

    2019上海区域赛现场赛总结 补题情况(以下通过率为牛客提交): 题号 标题 已通过代码 通过率 我的状态 A Mr. Panda and Dominoes 点击查看 5/29 未通过 B Prefi ...

  8. HDU-5532//2015ACM/ICPC亚洲区长春站-重现赛-F - Almost Sorted Array/,哈哈,水一把区域赛的题~~

    F - Almost Sorted Array Time Limit:2000MS     Memory Limit:262144KB     64bit IO Format:%I64d & ...

  9. hdu5080:几何+polya计数(鞍山区域赛K题)

    /* 鞍山区域赛的K题..当时比赛都没来得及看(反正看了也不会) 学了polya定理之后就赶紧跑来补这个题.. 由于几何比较烂写了又丑又长的代码,还debug了很久.. 比较感动的是竟然1Y了.. * ...

随机推荐

  1. MvvmCross[翻译] 使用Xamarin与MvvmCross完成一个跨平台App

    总览 原文:https://github.com/MvvmCross/MvvmCross/wiki/Tip-Calc-A-first-app 我们所做的第一个Model-View-ViewModel( ...

  2. SQL,学习基础2

    列=字段, 记录=实体  事物日志文件(用来记录数据库的增删情况,扩展名LDF) 数据库文件(但是只有一个是主数据库文件(即用它来启动的),其余为次数据库文件)mdf 数据类型: 整形(整数)——in ...

  3. CSS 分组 和 嵌套 选择器

    Grouping Selectors 在样式表中有很多具有相同样式的元素. h1{color:green;}h2{color:green;}p{color:green;} 为了尽量减少代码,你可以使用 ...

  4. POJ 1936 All in All(模拟)

    All in All 题目链接:http://poj.org/problem?id=1936 题目大意:判断从字符串s2中能否找到子串s1.字符串长度为10W. Sample Input sequen ...

  5. 翻纸牌 高校俱乐部 英雄会 csdn

    题目描述 有一种纸牌游戏,很有意思,给你N张纸牌,一字排开,纸牌有正反两面,开始的纸牌可能是一种乱的状态(有些朝正,有些朝反),现在你需要整理这些纸牌.但是麻烦的是,每当你翻一张纸牌(由正翻到反,或者 ...

  6. 读书笔记之 - javascript 设计模式 - 观察者模式

    在事件驱动的环境中,比如浏览器这种持续寻求用户关注的环境中,观察者模式是一种管理人与其任务(确切的讲,是对象及其行为和状态之间的关系)之间的关系的得力工具.用javascript的话来讲,这种模式的实 ...

  7. CentOS7 开关防火墙

    systemctl start firewalld.service#启动firewallsystemctl stop firewalld.service#停止firewallsystemctl dis ...

  8. Linux环境下添加ftp账号步骤

    (1)远程登录Linux服务器所用的工具,免费开源,可以从网站上很容易就下载到. (2)打开putty,输入服务器IP,进入后按提示进入用户名和密码输入超级管理员 root,然后系统让输入密码,注意此 ...

  9. wordpress4.0.1源码学习和摘录--项目设置

    1.静态变量日期 define( 'MINUTE_IN_SECONDS', 60 ); define( 'HOUR_IN_SECONDS', 60 * MINUTE_IN_SECONDS ); def ...

  10. python使用pyapns进行ios推送消息

    Pyapns 提供了通用的Apple Push Notification Service (APNS).该解决方案使用了开源的Twisted server,支持原生的Python和Ruby API. ...