题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=4422

我真服了。。这题我能调一天半,最后还是对拍拍出来的。。。脑子还是有病啊

题解: 首先可以dp, 分情况讨论: 若下面右面都有栅栏则值为零,若仅下面有栅栏则dp值等于右面,若仅右面有栅栏则dp值等于下面,若\((i,j)\)满足存在一矩形\((i+1,j+1)-(x,y)\)则dp[i][j]=dp[i+1][j]+dp[i][j+1]-dp[x+1][y+1],否则dp[i][j]=dp[i+1][j]+dp[i][j+1]-dp[i+1][j+1]

然后考虑用线段树+扫描线优化。差分之后推一波发现只需要支持: 单点加、区间覆盖(为0)、区间求和。

扫描线写得还是不熟。注意如果从右往左扫,同一竖列一定要先加入栅栏再删除栅栏!如数据:

5
2 6 6 7
1 9 9 9
8 4 9 5
2 1 3 3
10 2 10 2
10
4 5
10 4
8 10
4 3
1 10
10 3
9 4
8 10
6 6
5 10
1
1 3

代码

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cassert>
#include<set>
#include<iostream>
#include<algorithm>
using namespace std; inline int read()
{
int x=0; bool f=1; char c=getchar();
for(;!isdigit(c);c=getchar()) if(c=='-') f=0;
for(; isdigit(c);c=getchar()) x=(x<<3)+(x<<1)+(c^'0');
if(f) return x;
return -x;
} const int N = 2e5;
const int C = 1e6;
struct SegmentTree
{
struct SgTNode
{
int sum,tag;
SgTNode() {sum = 0,tag = -1;}
} sgt[(C<<2)+3];
void pushdown(int pos,int le,int ri)
{
int mid = (le+ri)>>1;
if(sgt[pos].tag>-1)
{
sgt[pos<<1].sum = sgt[pos].tag*(mid-le+1); sgt[pos<<1].tag = sgt[pos].tag;
sgt[pos<<1|1].sum = sgt[pos].tag*(ri-mid); sgt[pos<<1|1].tag = sgt[pos].tag;
sgt[pos].tag = -1;
}
}
void addval(int pos,int le,int ri,int lrb,int val)
{
if(le==lrb && ri==lrb) {sgt[pos].sum += val; sgt[pos].tag = -1; return;}
pushdown(pos,le,ri);
int mid = (le+ri)>>1;
if(lrb<=mid) addval(pos<<1,le,mid,lrb,val);
else addval(pos<<1|1,mid+1,ri,lrb,val);
sgt[pos].sum = sgt[pos<<1].sum+sgt[pos<<1|1].sum;
}
void modify(int pos,int le,int ri,int lb,int rb,int val)
{
if(le>=lb && ri<=rb) {sgt[pos].sum = (ri-le+1)*val; sgt[pos].tag = val; return;}
pushdown(pos,le,ri);
int mid = (le+ri)>>1;
if(lb<=mid) {modify(pos<<1,le,mid,lb,rb,val);}
if(rb>mid) {modify(pos<<1|1,mid+1,ri,lb,rb,val);}
sgt[pos].sum = sgt[pos<<1].sum+sgt[pos<<1|1].sum;
}
int querysum(int pos,int le,int ri,int lb,int rb)
{
if(le>=lb && ri<=rb) {return sgt[pos].sum;}
pushdown(pos,le,ri);
int mid = (le+ri)>>1,ret = 0;
if(lb<=mid) {ret += querysum(pos<<1,le,mid,lb,rb);}
if(rb>mid) {ret += querysum(pos<<1|1,mid+1,ri,lb,rb);}
sgt[pos].sum = sgt[pos<<1].sum+sgt[pos<<1|1].sum;
return ret;
}
} sgt;
struct Event
{
int opt,x,y,xx,yy,ans,id; //1Õ¤À¸×ó²à£¬2Õ¤À¸ÓҲ࣬3»¨£¬4Å£
bool operator <(const Event &arg) const
{
if(x<arg.x) return true; else if(x>arg.x) return false;
if(opt>arg.opt) return true; else if(opt<arg.opt) return false;
return y>arg.y;
}
} qr[(N<<2)+3];
bool cmp_id(Event x,Event y) {return x.id<y.id;}
int id[(N<<2)+3];
set<int> s;
int n,m,p,q; int getval(int y) //yµ½ÏÂÃæµÚÒ»¸öÅ£À¸-1µÄºÍ
{
int tmp = *s.upper_bound(y);
int ret = sgt.querysum(1,1,C,y,tmp-1);
return ret;
} int main()
{
int q = 0;
scanf("%d",&p);
for(int i=1; i<=p; i++)
{
int lbx,rbx,lby,rby; scanf("%d%d%d%d",&lby,&lbx,&rby,&rbx);
q++; qr[q].x = lbx-1; qr[q].y = lby; qr[q].opt = 2; qr[q].id = q;
q++; qr[q].x = rbx; qr[q].y = rby; qr[q].opt = 1; qr[q].id = q;
}
scanf("%d",&m); for(int i=1; i<=m; i++) {int x,y; scanf("%d%d",&y,&x); q++; qr[q].x = x; qr[q].y = y; qr[q].opt = 3; qr[q].id = q;}
scanf("%d",&n); for(int i=1; i<=n; i++) {int x,y; scanf("%d%d",&y,&x); q++; qr[q].x = x; qr[q].y = y; qr[q].opt = 4; qr[q].id = q;}
sort(qr+1,qr+q+1); int j = q;
for(int i=1; i<=q; i++) id[qr[i].id] = i;
s.insert(C+1);
for(int i=C; i>=1; i--)
{
while(j>0 && qr[j].x==i)
{
if(qr[j].opt==2)
{
int k = id[qr[j].id+1]; int xx = qr[k].x,yy = qr[k].y,x = i,y = qr[j].y;
if(y>1)
{
int cur = getval(y-1)-qr[k].ans;
sgt.modify(1,1,C,y-1,y-1,cur);
}
sgt.modify(1,1,C,y,yy,0);
s.erase(y); if(yy<C) s.erase(yy+1);
}
else if(qr[j].opt==1)
{
int k = id[qr[j].id-1]; int x = qr[k].x,y = qr[k].y,xx = i,yy = qr[j].y;
if(y>1)
{
int cur = getval(y-1);
sgt.modify(1,1,C,y-1,y-1,cur);
}
if(yy<C) qr[j].ans = getval(yy+1);
sgt.modify(1,1,C,y,yy,0);
s.insert(y); if(yy<C) s.insert(yy+1);
}
else if(qr[j].opt==3)
{
sgt.addval(1,1,C,qr[j].y,1);
}
else if(qr[j].opt==4)
{
qr[j].ans = getval(qr[j].y);
}
j--;
}
}
sort(qr+1,qr+q+1,cmp_id);
for(int i=1; i<=q; i++) {if(qr[i].opt==4) printf("%d\n",qr[i].ans);}
return 0;
}

中考超QDEZ线\(32\)分,心里最大的一块石头终于落地了……

BZOJ 4422 Cow Confinement (线段树、DP、扫描线、差分)的更多相关文章

  1. 【BZOJ-4422】Cow Confinement 线段树 + 扫描线 + 差分 (优化DP)

    4422: [Cerc2015]Cow Confinement Time Limit: 50 Sec  Memory Limit: 512 MBSubmit: 61  Solved: 26[Submi ...

  2. Tsinsen A1219. 采矿(陈许旻) (树链剖分,线段树 + DP)

    [题目链接] http://www.tsinsen.com/A1219 [题意] 给定一棵树,a[u][i]代表u结点分配i人的收益,可以随时改变a[u],查询(u,v)代表在u子树的所有节点,在u- ...

  3. HDU 3016 Man Down (线段树+dp)

    HDU 3016 Man Down (线段树+dp) Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Ja ...

  4. 【POJ1151】Atlantis(线段树,扫描线)

    [POJ1151]Atlantis(线段树,扫描线) 题面 Vjudge 题解 学一学扫描线 其实很简单啦 这道题目要求的就是若干矩形的面积和 把扫描线平行于某个轴扫过去(我选的平行\(y\)轴扫) ...

  5. 【Loj#535】花火(线段树,扫描线)

    [Loj#535]花火(线段树,扫描线) 题面 Loj 题解 首先如果不考虑交换任意两个数这个操作,答案就是逆序对的个数. 那么暴力就是枚举交换哪个两个数,然后用数据结构之类的东西动态维护逆序对. 但 ...

  6. Bzoj 2752 高速公路 (期望,线段树)

    Bzoj 2752 高速公路 (期望,线段树) 题目链接 这道题显然求边,因为题目是一条链,所以直接采用把边编上号.看成序列即可 \(1\)与\(2\)号点的边连得是. 编号为\(1\)的点.查询的时 ...

  7. hdu 1542 线段树之扫描线之面积并

    点击打开链接 题意:给你n个矩形,求它们的面积,反复的不反复计算 思路:用线段树的扫描线完毕.将X坐标离散化后,从下到上扫描矩形,进行各种处理,看代码凝视把 #include <stdio.h& ...

  8. hdu1542 矩形面积并(线段树+离散化+扫描线)

    题意: 给你n个矩形,输入每个矩形的左上角坐标和右下角坐标. 然后求矩形的总面积.(矩形可能相交). 题解: 前言: 先说说做这道题的感受: 刚看到这道题顿时就懵逼了,几何 烂的渣渣.后来从网上搜题解 ...

  9. 【POJ 2482】 Stars in Your Window(线段树+离散化+扫描线)

    [POJ 2482] Stars in Your Window(线段树+离散化+扫描线) Time Limit: 1000MS   Memory Limit: 65536K Total Submiss ...

随机推荐

  1. [转帖]小米手环采用RISC-V 指令集芯片

    小米手环4或用“黄山一号”芯片,雷军再回前线,未来走向如何 静心科技 06-1111:19 忘记来源地址了 不过国内的很多东西都是有中国特色的 比如飞腾 比如麒麟(银河麒麟 还有华为的麒麟 980) ...

  2. 性能库 Oracle数据库 连接不上问题的解决

    今天性能库的centos 上面的oracle18c 数据库连接不上了 这里进行了一次简单处理 针对不同问题进行说明: 1. 实例时 blocked 的状态 问题现象: LSNRCTL> stat ...

  3. Python的入门(day1)

    一:Python的起源 Python的创始人为Guido van Rossum.1989年圣诞节期间,在阿姆斯特丹,Guido为了打发圣诞节的无趣,决心开发一个新的脚本解释程序,做为ABC 语言的一种 ...

  4. 移除list里面的值

    public class IteratorTest { public static void main(String[] args) { List<String> list = new A ...

  5. DataFrame.to_dict(orient='dict')

    DataFrame.to_dict(orient=’dict’) >>> df = pd.DataFrame({'name':[1,2,3],"class":[1 ...

  6. 剑指offer-字符串的排列-数组-递归-动态规划-python

    题目描述 输入一个字符串,按字典序打印出该字符串中字符的所有排列.例如输入字符串abc,则打印出由字符a,b,c所能排列出来的所有字符串abc,acb,bac,bca,cab和cba. 输入描述: 输 ...

  7. 用SQL存储过程生成唯一单据号

    用SQL存储过程生成唯一单据号     在一些系统中,经理要生成单据号,为了不使多台客户端生成的单据号重复,一般要在服务端生成这种流水号,本文是在数据库中生成流水号,并且可以生成多种类型的单据号(比如 ...

  8. 国内高速下载Docker

    一般情况下,我们可以从Docker官网下载docker安装文件,但是官方网站由于众所周知的原因,不是访问慢,就是下载慢.下载docker安装包动不动就要个把小时,真是极大的影响工作效率. 在这里推荐一 ...

  9. 解决错误:无法在web.xml或使用此应用程序部署的jar文件中解析绝对uri:[http://shiro.apache.org/tags]

    服务器错误信息如下: 解决方法: 把shiro包中的tld文件(shiro.tld)解压出来放到WEB-INF文件夹下即解决问题. 参考:http://blog.sina.com.cn/s/blog_ ...

  10. 2019-11-29-WPF-轻量级-MVVM-框架入门-2.1.2

    title author date CreateTime categories WPF 轻量级 MVVM 框架入门 2.1.2 lindexi 2019-11-29 10:16:10 +0800 20 ...