点我看题目

题意 :N个村子连成一条线,相邻的村子都有直接的地道进行相连,不相连的都由地道间接相连,三个命令,D x,表示x村庄被摧毁,R  ,表示最后被摧毁的村庄已经重建了,Q x表示,与x直接或间接相连的村庄有多少个,当然包括他自己。

思路 :这是一道用线段树,树状数组,还有STL都可以做的题。。。。因为用线段树的更新什么的太麻烦,。。。。。所以我用了树状数组

#include <iostream>
#include <stdio.h>
#include <string.h> using namespace std; const int maxn = ;
int tree[maxn],data[maxn] ;
bool vis[maxn] ;
int n , m ; int lowbit(int x)
{
return (x & (-x)) ;
} void add(int x,int value)
{
for(int i = x ; i <= n ; i += lowbit(i))
tree[i] += value ;
}
int get(int x)
{
int sum = ;
for(int i = x ; i ; i -= lowbit(i))
sum += tree[i] ;
return sum ;
} int binary_search(int v)
{
int l = , r = n+ ,i = n+;//n+2是为了防止最后一个村庄找不到的时候没有返回值
while(l <= r)
{
int mid = (l + r) >> ;
if(get(mid) >= v)
{
r = mid- ;
i = mid ;
}
else l = mid+ ;
}
return i ;
}
int main()
{
while(~scanf("%d %d",&n,&m))
{
int a , i = ;
while(m--)
{
getchar() ;
char ch ;
scanf("%c",&ch) ;
if(ch == 'D')
{
scanf("%d",&a) ;
data[++i] = ++a ;//被摧毁的村庄都存在data数组里
vis[a] = true ;
add(a,) ;
}
else if(ch == 'Q')
{
scanf("%d",&a) ;
a ++ ;
if(vis[a]) printf("0\n") ;
else
{
int sum1,sum2 ;
a = get(a) ;
sum1 = binary_search(a) ;
sum2 = binary_search(a+) ;
printf("%d\n",sum2-sum1-) ;
}
}
else if(ch == 'R')
{
a = data[i--] ;
vis[a] = false ;
add(a,-) ;
}
}
}
return ;
}

以前用树状数组做的,现在用线段树做的,因为没有初始化WA到死,杭电上是多组数据,所以一开始也WA了,因为每次重建的时候建的是最后一个村子,将被摧毁的村子放入栈中就可以了,然后要标记一下,所有的村子存在即为0,摧毁了即为1,更新用单点,查询用区间,查询x村子时,寻找左边离他最近的那个被摧毁的村庄即1,以及最右边那个被摧毁的村庄,然后两个相减加1即为结果,但是如果他本身就被摧毁了直接是0。

 //
#include <cstdio>
#include <cstring>
#include <iostream> using namespace std ; int lz[*],p[ * ],vis[ * ],stk[*] ; void pushup(int rt)
{
p[rt] = p[rt << ] + p[rt << | ] ;
}
void update(int x,int l,int r,int rt,int sc)
{
if(l == r)
{
p[rt] = sc ;
return ;
}
int mid = (l+r) >> ;
if(x <= mid)
update(x,l,mid,rt << , sc) ;
else
update(x,mid+,r,rt << | , sc) ;
pushup(rt) ;
}
int query(int L,int R,int l,int r,int rt)
{
int sum = ;
if(L <= l && r <= R)
{
return p[rt] ;
}
int mid = (l+r) >> ;
if(mid >= L)
sum += query(L,R,l,mid,rt << ) ;
if(mid < R)
sum += query(L,R,mid+,r,rt << | ) ;
return sum ;
}
int findd(int l,int r,int x,int rt)
{
if(l == r)
{
return l;
}
int mid = (l+r) >> ;
if(x <= p[rt << ])
return findd(l,mid,x,rt << ) ;
else return findd(mid+,r,x-p[rt << ],rt << | ) ;
}
int main()
{
int n, m,x;
char ch[] ;
while(~scanf("%d %d",&n,&m))
{
int top = ;
memset(p,,sizeof(p)) ;
memset(vis,,sizeof(vis)) ;
for(int i = ; i < m ; i++)
{
scanf("%s",ch) ;
if(ch[] == 'R')
{
if(top != )
{
vis[stk[top-]] = ;//村庄存在即为0,被摧毁即为1
update(stk[top-],,n,,) ;
top-- ;
}
}
else if(ch[] == 'D')
{
scanf("%d",&x) ;
vis[x] = ;
stk[top++] = x ;
update(x,,n,,) ;
}
else if(ch[] == 'Q')
{
scanf("%d",&x) ;
if(vis[x])
{
printf("0\n") ;
continue ;
}
int l,r ;
int l1 = query(,x,,n,) ;//x村庄左边的和
if(l1 == )//如果和为0,说明x村庄左边全部都存在
l = ;
else l = findd(,n,l1,)+;//找出左边离x最近的那个一,既不存在的那个村庄
int r1 = query(x,n,,n,) ;
if(r1 == )
r = n;
else r = findd(,n,l1+,)- ;
printf("%d\n",r-l+) ;
}
}
}
return ;
}

POJ 2892 Tunnel Warfare || HDU 1540(树状数组+二分 || 线段树的单点更新+区间查询)的更多相关文章

  1. 树状数组+二分||线段树 HDOJ 5493 Queue

    题目传送门 题意:已知每个人的独一无二的身高以及排在他前面或者后面比他高的人数,问身高字典序最小的排法 分析:首先对身高从矮到高排序,那么可以知道每个人有多少人的身高比他高,那么取较小值(k[i], ...

  2. BZOJ 3173 最长上升子序列(树状数组+二分+线段树)

    给定一个序列,初始为空.现在我们将1到N的数字插入到序列中,每次将一个数字插入到一个特定的位置.每插入一个数字,我们都想知道此时最长上升子序列长度是多少? 由于序列是顺序插入的,所以当前插入的数字对之 ...

  3. POJ 1195 Mobile phones (二维树状数组或线段树)

    偶然发现这题还没A掉............速速解决了............. 树状数组和线段树比较下,线段树是在是太冗余了,以后能用树状数组还是尽量用......... #include < ...

  4. HDU 5618 Jam's problem again(三维偏序,CDQ分治,树状数组,线段树)

    Jam's problem again Time Limit: 5000/2500 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Othe ...

  5. HDU 5877 2016大连网络赛 Weak Pair(树状数组,线段树,动态开点,启发式合并,可持久化线段树)

    Weak Pair Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others) Tota ...

  6. st表、树状数组与线段树 笔记与思路整理

    已更新(2/3):st表.树状数组 st表.树状数组与线段树是三种比较高级的数据结构,大多数操作时间复杂度为O(log n),用来处理一些RMQ问题或类似的数列区间处理问题. 一.ST表(Sparse ...

  7. bzoj 3110: [Zjoi2013]K大数查询 树状数组套线段树

    3110: [Zjoi2013]K大数查询 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1384  Solved: 629[Submit][Stat ...

  8. [BZOJ 3196] 213平衡树 【线段树套set + 树状数组套线段树】

    题目链接:BZOJ - 3196 题目分析 区间Kth和区间Rank用树状数组套线段树实现,区间前驱后继用线段树套set实现. 为了节省空间,需要离线,先离散化,这样需要的数组大小可以小一些,可以卡过 ...

  9. [BZOJ 1901] Dynamic Rankings 【树状数组套线段树 || 线段树套线段树】

    题目链接:BZOJ - 1901 题目分析 树状数组套线段树或线段树套线段树都可以解决这道题. 第一层是区间,第二层是权值. 空间复杂度和时间复杂度均为 O(n log^2 n). 线段树比树状数组麻 ...

随机推荐

  1. Linux 普通用户su命令切换控制

    1.编辑配置文件/etc/pam.d/su .将下面配置文件"#“去掉: # auth           required        pam_wheel.so use_uid 改成 a ...

  2. Redis操作字符串工具类封装,Redis工具类封装

    Redis操作字符串工具类封装,Redis工具类封装 >>>>>>>>>>>>>>>>>>& ...

  3. JQuery事件的绑定

    关于jQuery事件绑定html: <a href="#" onclick="addBtn()">addBtn</a> <div ...

  4. 自己动手实现SharePointList的分页展示

    SharePoin列表里对条目的展示只有上一页下一页,不能够跳转,不能够一次导航到第一页和最后一页. 项目需要,所以对列表的数据展示进行了定制化开发来实现如上的功能. 前端用GridView展示,用L ...

  5. 腾讯海量数据处理平台TDW

    TDW是腾讯海量数据处理平台中最核心的模块,它有以下几个作用: 提供海量的离线计算和存储服务.TDW是腾讯内部规模最大的离线数据处理平台,公司内大多数业务的产品报表.运营分析.数据挖掘等的存储和计算都 ...

  6. MVC中HtmlHelper用法大全参考

    MVC中HtmlHelper用法大全参考 解析MVC中HtmlHelper控件7个大类中各个控件的主要使用方法(1) 2012-02-27 16:25 HtmlHelper类在命令System.Web ...

  7. JS实现各种页面的刷新

    JS实现各种页面的刷新功能 1.刷新当前页面 opener.location.replace(opener.location.href); 或者window.opener.window.locatio ...

  8. row_number() OVER (PARTITION BY COL1 ORDER BY COL2)

    select *,ROW_NUMBER() over(partition by deviceID order by RecordDate desc row_number() OVER (PARTITI ...

  9. mysql 根据某个字段将多条记录的某个字段拼接成一个字段

    未合并情况 SELECT a.id, b.name AS "role" FROM sys_user a INNER JOIN sys_user_role c ON a.id=c.u ...

  10. Linux的前世今生

    Linux的起源 说到Linux[/ˈlɪnəks/],想必大家也会自然而然地想到他的创始人——被称为“Linux之父”的林纳斯·托瓦兹(Linus Torvalds).其实,在Linux出现之前,还 ...