usaco 2008 月赛 lites 开关灯 题解
所有的灯都是关闭的. 奶牛们通过N个按钮来控制灯的开关; 按第i个按钮可以 改变第i个灯的状态.奶牛们执行M (1 <= M <= 100,000)条指令, 每个指令都是两个整数中的一个(0 <= 指令号 <= 1). 第1种指令(用0表示)包含两个数字S_i和E_i (1 <= S_i <= E_i <= N), 它们表示起始开关 和终止开关. 奶牛们只需要把从S_i到E_i之间的按钮都按一次, 就可以完成这个指令. 第2种指令(用1表示)同样包含两个数字S_i和E_i (1 <= S_i <= E_i
<= N), 不过这种指令 是询问从S_i到E_i之间的灯有多少是亮着的. 帮助FJ确保他的奶牛们可以得到正确的答案.
这是我做的第一道线段树的题目,以前只了解过模板——下面我就来介绍一下。
-------------------------------------------线段树的介绍-------------------------------------------------------
线段树是什么?就是一种二叉树结构,而且它的每一个结点代表一条线段的和。
以下是百度百科的详细解释:
一般线段树我都是用堆来储存的,即a[i]的左儿子是a[i*2],右儿子是a[i*2+1]。
线段树快在哪里呢?他对于每个结点的修改和访问都是log(n)的效率。
相信到这里都很好理解,下面介绍一种更加强大的思想:lazy思想。
如果我要将a--b的区间的每一个值都加上p,那么效率就是(b-a+1)log(n)。有没有更快的方法呢?
通过研究,我们发现:如果它只修改而并不访问,我们岂不是不用更新了!
于是我们可以再开一个域cnt,表示目前有多少次累加被积压着。每次处理到k结点时(注意,这个处理包括访问和修改等),我们都把它积压的值给加上,然后传给它的儿子。此时它的儿子不必处理,直到要处理到它的儿子为止。
下面是这强大的update代码:
void update(long k)
{
if (a[k].cnt==0) return;
a[k].sum+=(a[k].r-a[k].l+1)*a[k].cnt;
a[k*2].cnt+=a[k].cnt;a[k*2+1].cnt+=a[k].cnt;
a[k].cnt=0;
}
每次处理到k时,先做一遍update(k)再进行操作即可。
------------------------------------------------回到原题-------------------------------------------------------
这里可能相对简单一些,下面说一些与模板不太相同的细节。
(1)初始建树时,每个值都是0.
(2)在插入时,每次cnt累加1即可,这个1当然不是加上1,而是执行操作数。
(3)在update中,如果cnt是偶数,就可以直接清零(因为偶数次不是抵消了嘛!);如果是奇数,我们把它相反一下,并向下传1。
代码:
#include<stdio.h>
using namespace std;
const long maxn=100001;
struct tree
{
long l,r; //l和r就是a[i]所表示的左右区间范围。
long cnt,sum; //cnt是累加的操作次数,sum是a[i]表示的线段中开着的灯的数量。
}a[4*maxn];
long n,m,i,x,y,ok;
void build(long k,long l,long r) //基础建树
{
a[k].cnt=0;a[k].l=l;a[k].r=r;
if (l==r) return;
long mid=(l+r)/2;
build(k*2,l,mid);build(k*2+1,mid+1,r);
}
void update(long k) //update操作。
{
if (a[k].cnt==0) return;
if (a[k].cnt%2==1)
{
a[k].sum=a[k].r-a[k].l+1-a[k].sum; //a[k]的区间是a[k].l到a[k].r,那么共有a[k].r-a[k].l+1个,这里取反。
a[k*2].cnt+=1;a[k*2+1].cnt+=1;
a[k].cnt=0;
}
else a[k].cnt=0;
}
void ins(long k,long l,long r)
{
update(k);
if (a[k].l>=l&&a[k].r<=r) {a[k].cnt+=1;return;} //当前区域被覆盖
long mid=(a[k].l+a[k].r)/2; //二分找被覆盖的区域。
if (l<=mid) ins(k*2,l,r);
if (r>mid) ins(k*2+1,l,r);
update(k*2);update(k*2+1); //对儿子的更新操作(很重要),因为下面一步要重新更新当前的值。
a[k].sum=a[k*2].sum+a[k*2+1].sum;
}
long find(long k,long l,long r) //这与ins大同小异。
{
update(k);
if (a[k].l>=l&&a[k].r<=r) return a[k].sum;
long mid=(a[k].l+a[k].r)/2,o=0;
if (l<=mid) o+=find(k*2,l,r);
if (r>mid) o+=find(k*2+1,l,r);
update(k*2);update(k*2+1);
a[k].sum=a[k*2].sum+a[k*2+1].sum;
return o;
}
int main()
{
//freopen("lites.in","r",stdin);
//freopen("lites.out","w",stdout);
scanf("%ld%ld",&n,&m);
build(1,1,n);
for (i=1;i<=m;i++)
{
scanf("%ld%ld%ld",&ok,&x,&y);
if (ok==0) ins(1,x,y);
else {long ans=find(1,x,y);printf("%ld\n",ans);}
}
//scanf("%ld",&n,&m);
return 0;
}
usaco 2008 月赛 lites 开关灯 题解的更多相关文章
- usaco 2002 月赛 Fiber Communications 题解
Description Farmer John wants to connect his N (1 <= N <= 1,000) barns (numbered 1..N) with a ...
- bzoj usaco 金组水题题解(1)
UPD:我真不是想骗访问量TAT..一开始没注意总长度写着写着网页崩了王仓(其实中午的时候就时常开始卡了= =)....损失了2h(幸好长一点的都单独开了一篇)....吓得赶紧分成两坨....TAT. ...
- 1230: [Usaco2008 Nov]lites 开关灯
1230: [Usaco2008 Nov]lites 开关灯 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 1162 Solved: 589[Sub ...
- Code[VS]1690 开关灯 题解
Code[VS]1690 开关灯 题解 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description: YYX家门前 ...
- BZOJ 1230: [Usaco2008 Nov]lites 开关灯( 线段树 )
线段树.. --------------------------------------------------------------------------------- #include< ...
- usaco 2002 月赛 Chores 题解
Description Farmer John's family pitches in with the chores during milking, doing all the chores as ...
- USACO全部月赛及GateWay数据
月赛: 以07年open为例,网站如下 http://contest.usaco.org/OPEN07 其他的格式是http://contest.usaco.org/月份(月份的英文前三位,比如1月是 ...
- ZOJ Monthly, June 2014 月赛BCDEFGH题题解
比赛链接:点击打开链接 上来先搞了f.c,,然后发现状态不正确,一下午都是脑洞大开,, 无脑wa,无脑ce...一样的错犯2次.. 硬着头皮搞了几发,最后20分钟码了一下G,不知道为什么把1直接当成不 ...
- USACO section 1.1 C++题解
USACO section1.1:DONE 2017.03.03 TEXT Submitting Solutions DONE 2017.03.04 PROB Your Ride Is Here [A ...
随机推荐
- Plupload上传插件自定义图片的修改
若自定义的一个上传图片效果,代码(可能不全),当用户再次点击所有或任意一个上传图片的input时,uploader.files已经多了客户再次上传的图片,但是你就想要最后的两张图片,这就可以使用到up ...
- RFM模型——构建数据库营销的商业战役!(转)
RFM模型:R(Recency)表示客户最近一次购买的时间有多远,F(Frequency)表示客户在最近一段时间内购买的次数,M (Monetary)表示客户在最近一段时间内购买的金额.一般原始数据为 ...
- 你对SpringMvc是如何理解的?
SpringMVC工作原理 SpringMvc是基于过滤器对servlet进行了封装的一个框架,我们使用的时候就是在web.xml文件中配置DispatcherServlet类:SpringMvc工作 ...
- 【T-SQL进阶】02.理解SQL查询的底层原理
本系列[T-SQL]主要是针对T-SQL的总结. [T-SQL基础]01.单表查询-几道sql查询题 [T-SQL基础]02.联接查询 [T-SQL基础]03.子查询 [T-SQL基础]04.表表达式 ...
- (原创)Maven+Spring+CXF+Tomcat7 简单例子实现webservice
这个例子需要建三个Maven项目,其中一个为父项目,另外两个为子项目 首先,建立父项目testParent,选择quickstart: 输入项目名称和模块名称,然后创建: 然后建立子项目testInt ...
- D. Powerful array
D. Powerful array 题意 给定一个数列:a[i] (1<= i <= n) K[j]表示 在区间 [l,r]中j出现的次数.有t个查询,每个查询l,r,对区间内所有a[i] ...
- Akka(5): ConsistentHashing Router - 可选定Routee的任务分配模式
上一篇讨论里我们介绍了几种任务分配(Routing)模式.Akka提供的几种现成智能化Routing模式大多数是通过对用户屏蔽具体的运算Routee选择方式来简化Router使用,提高智能程度,所以我 ...
- 做一枚精致的程序猿,Fighting!
这几天我和我们的团队正在做一个公司管理系统的项目,团队分工根据成员的水平高低来分工,这样看似公平,但其实不公平,如此这样一来,那些水平稍不如别人的成员就没有发展的机会?那么问题来了,对于水平稍逊色的程 ...
- Hadoop 2.7 伪分布式环境搭建
1.安装环境 ①.一台Linux CentOS6.7 系统 hostname ipaddress subnet mask ...
- Nodejs的http模块
一.http服务器 我们知道传统的HTTP服务器是由Aphche.Nginx.IIS之类的软件来搭建的,但是Nodejs并不需要,Nodejs提供了http模块,自身就可以用来构建服务器,例如: ...