Codevs_1690_开关灯(线段树)

描述


http://codevs.cn/problem/1690/

一排共 n 个灯,起初都是关着的,现在有 m 次操作. 0 开头的操作代表将 l ~ r 的开关按一遍,1 开头的操作代表询问 l ~ r 共多少个开着的灯.

分析



刚开始想的是用 0 , 1 , -1 分别代表区间都是关着的,都是开着的,有开的又有关的,跟新和查询的时候都是查找到非 -1 的区间再做处理,但是 TLE 了.

想一下如果灯是 1 0 1 0 1 0 1 0 1...这样排列的,那就退化得不成样子了.

所以用 on 和 off 分别代表区间内开着的和关着的等的数量,如果按开关,就把这两个值交换.

最后可以不用 off ,只用一个 on ,可以通过区间长度进行计算.

#include<cstdio>

const int maxn=100005;
struct node { int l,r; int on;bool d; }a[3*maxn];
int n,m;

void build_tree(int l,int r,int k)
{
    a[k].l=l; a[k].r=r; a[k].on=0; a[k].d=false;
    if(l==r) return;
    int mid=l+(r-l)/2;
    build_tree(l,mid,2*k);
    build_tree(mid+1,r,2*k+1);
}

inline void turn(int k)
{
    a[k].on=(a[k].r-a[k].l+1)-a[k].on;
    a[k].d^=true;
}

void update(int l,int r,int k)
{
    if(a[k].l==l&&a[k].r==r)
    {
        turn(k);
        return;
    }
    if(a[k].d)
    {
        turn(2*k);
        turn(2*k+1);
        a[k].d=false;
    }
    int mid=a[k].l+(a[k].r-a[k].l)/2;
    if(r<=mid) update(l,r,2*k);
    else if(l>mid) update(l,r,2*k+1);
    else { update(l,mid,2*k); update(mid+1,r,2*k+1); }
    a[k].on=a[2*k].on+a[2*k+1].on;
}

int search(int l,int r,int k)
{
    if(a[k].l==l&&a[k].r==r) return a[k].on;
    if(a[k].d)
    {
        turn(2*k);
        turn(2*k+1);
        a[k].d=false;
    }
    int mid=a[k].l+(a[k].r-a[k].l)/2;
    if(r<=mid) return search(l,r,2*k);
    else if(l>mid) return search(l,r,2*k+1);
    else return search(l,mid,2*k)+search(mid+1,r,2*k+1);
}

int main()
{
    scanf("%d%d",&n,&m);
    build_tree(1,n,1);

    int qry,l,r;
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d%d",&qry,&l,&r);
        switch(qry)
        {
            case 0:
                update(l,r,1);
                break;
            case 1:
                printf("%d\n",search(l,r,1));
                break;
        }
    }
    return 0;
}
时间: 2024-10-09 03:51:07

Codevs_1690_开关灯(线段树)的相关文章

BZOJ 1230: [Usaco2008 Nov]lites 开关灯( 线段树 )

线段树.. --------------------------------------------------------------------------------- #include<cstdio> #include<algorithm> #include<cstring> #include<iostream> #define rep( i , n ) for( int i = 0 ; i < n ; ++i ) #define clr( x

codevs 1690 开关灯 线段树+延迟标记

1690 开关灯 时间限制: 1 s 空间限制: 128000 KB 题目描述 Description YYX家门前的街上有N(2<=N<=100000)盏路灯,在晚上六点之前,这些路灯全是关着的,六点之后,会有M(2<=m<=100000)个人陆续按下开关,这些开关可以改变从第i盏灯到第j盏灯的状态,现在YYX想知道,从第x盏灯到第y盏灯中有多少是亮着的(1<=i,j,x,y<=N) 输入描述 Input Description 第 1 行: 用空格隔开的两个整数N和

BZOJ 1230--lites 开关灯(线段树)

1230: [Usaco2008 Nov]lites 开关灯 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1682  Solved: 876[Submit][Status][Discuss] Description Farmer John尝试通过和奶牛们玩益智玩具来保持他的奶牛们思维敏捷. 其中一个大型玩具是牛栏中的灯. N (2 <= N <= 100,000) 头奶牛中的每一头被连续的编号为1..N, 站在一个彩色的灯下面.刚到傍晚的时候

B1230 [Usaco2008 Nov]lites 开关灯 线段树

就是线段树维护异或和.之前我线段树区间修改down都是修改当前区间,结果debug出不来,改成每次向下了. 题干: Description Farmer John尝试通过和奶牛们玩益智玩具来保持他的奶牛们思维敏捷. 其中一个大型玩具是牛栏中的灯. N (2 <= N <= 100,000) 头奶牛中的每一头被连续的编号为1..N, 站在一个彩色的灯下面.刚到傍晚的时候, 所有的灯都是关闭的. 奶牛们通过N个按钮来控制灯的开关; 按第i个按钮可以改变第i个灯的状态.奶牛们执行M (1 <=

【codevs1690】开关灯 线段树 区间修改+区间求和(标记)

[codevs1690]开关灯 2014年2月15日4930 题目描述 Description YYX家门前的街上有N(2<=N<=100000)盏路灯,在晚上六点之前,这些路灯全是关着的,六点之后,会有M(2<=m<=100000)个人陆续按下开关,这些开关可以改变从第i盏灯到第j盏灯的状态,现在YYX想知道,从第x盏灯到第y盏灯中有多少是亮着的(1<=i,j,x,y<=N) 输入描述 Input Description 第 1 行: 用空格隔开的两个整数N和M 第

【BZOJ1230】 [Usaco2008 Nov]lites 开关灯 线段树

递归不要加inline 会出现神奇的错误..... 1 #include <iostream> 2 #include <cstdio> 3 #define N 100010 4 using namespace std; 5 struct Tree 6 { 7 int size,rev,l,r; 8 }seg[N*4]; 9 int n,m; 10 inline int read() 11 { 12 char c; 13 int ans=0,f=1; 14 while (!isdig

CODEVS1690|开关灯|线段树(带lazy)

题目描述 Description    YYX家门前的街上有N(2<=N<=100000)盏路灯,在晚上六点之前,这些路灯全是关着的,六点之后,会有M(2<=m<=100000)个人陆续按下开关,这些开关可以改变从第i盏灯到第j盏灯的状态,现在YYX想知道,从第x盏灯到第y盏灯中有多少是亮着的(1<=i,j,x,y<=N)输入描述 Input Description第 1 行: 用空格隔开的两个整数N和M第 2..M+1 行: 每行表示一个操作, 有三个用空格分开的整数

线段树--codevs 1690 开关灯

codevs 1690 开关灯 USACO 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description YYX家门前的街上有N(2<=N<=100000)盏路灯,在晚上六点之前,这些路灯全是关着的,六点之后,会有M(2<=m<=100000)个人陆续按下开关,这些开关可以改变从第i盏灯到第j盏灯的状态,现在YYX想知道,从第x盏灯到第y盏灯中有多少是亮着的(1<=i,j,x,y<=N) 输入描述 Input

【线段树】Bzoj1230 [Usaco2008 Nov]lites 开关灯

Description Farmer John尝试通过和奶牛们玩益智玩具来保持他的奶牛们思维敏捷. 其中一个大型玩具是牛栏中的灯. N (2 <= N <= 100,000) 头奶牛中的每一头被连续的编号为1..N, 站在一个彩色的灯下面.刚到傍晚的时候, 所有的灯都是关闭的. 奶牛们通过N个按钮来控制灯的开关; 按第i个按钮可以改变第i个灯的状态.奶牛们执行M (1 <= M <= 100,000)条指令, 每个指令都是两个整数中的一个(0 <= 指令号 <= 1).