题目:http://acm.hdu.edu.cn/showproblem.php?pid=4046

 

题意:给出一个字符串,统计这个字符串任意区间中“wbw”出现的次数。

规定两种操作,一是查询任意区间“wbw”出现次数;二是修改某一位置的字符。

 

分析:比较明显的线段树,单点更新,区间查询。

线段树记录的信息是区间中出现“wbw”字符的个数,线段树的叶子节点[i,i]记录字符串str中

str[i-2][i-1][i]是否是“wbw“。

对于字符的修改,这里就有要注意了。修改的位置pos, 可能对线段树中sum[pos-1]   sum[pos] 或sum[pos+1]

造成影响。这里要分情况。

 

代码

#include<cstdio>
#include<cstring>

#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1

using namespace std;

const int maxn=50010;
int n,m,sum[maxn<<2];
char str[maxn];
int num[maxn];

void pushup(int rt)
{
	sum[rt]=sum[rt<<1]+sum[rt<<1|1];
}

void build(int l,int r,int rt)
{
	if(l==r)
	{
		sum[rt]=num[l];
		return;
	}
	int m=(l+r)>>1;
	build(lson);
	build(rson);
	pushup(rt);
}

int query(int L,int R,int l,int r,int rt)
{
	if(L<=l&&r<=R)
	{
		return sum[rt];
	}
	int m=(l+r)>>1;
	int ans=0;
	if(L<=m) ans+=query(L,R,lson);
	if(R>m)  ans+=query(L,R,rson);
	return ans;
}

void update(int p,int val,int l,int r,int rt)
{
	if(l==r)
	{
		sum[rt]=val;
		return;
	}
	int m=(l+r)>>1;
	if(p<=m) update(p,val,lson);
	else 	 update(p,val,rson);
	pushup(rt);
}

int main()
{
	int t,T;
	scanf("%d",&T);
	for(int t=1;t<=T;t++)
	{
		printf("Case %d:\n",t);
		scanf("%d%d%s",&n,&m,str+1);

		memset(num,0,sizeof(num));
		for(int i=3;i<=n;i++)
			if(str[i-2]=='w'&&str[i-1]=='b'&&str[i]=='w')  num[i]=1;
		build(1,n,1);

		int k,a,b;
		char ch[5];
		while (m--)
		{
			scanf("%d",&k);
			if(k==0)
			{
				scanf("%d%d",&a,&b);
				a++;b++;   //字符串从1开始,所以下标都加1
				if(b-a<2)  printf("0\n");
				else       printf("%d\n",query(a+2,b,1,n,1));

			}
			else
			{
				scanf("%d%s",&a,ch);
				a++;
				if(ch[0]==str[a]) continue;//修改的和以前一样,这不用任何操作
				if(a>=3)
				{
					if(str[a-2]=='w'&&str[a-1]=='b'&&str[a]=='w')
						update(a,0,1,n,1);
					if(str[a-2]=='w'&&str[a-1]=='b'&&str[a]=='b')
						update(a,1,1,n,1);
				}
				if(a>=2&&a+1<=n)
				{
					if(str[a-1]=='w'&&str[a]=='b'&&str[a+1]=='w')
						update(a+1,0,1,n,1);
					if(str[a-1]=='w'&&str[a]=='w'&&str[a+1]=='w')
						update(a+1,1,1,n,1);
				}
				if(a+2<=n)
				{
					if(str[a]=='w'&&str[a+1]=='b'&&str[a+2]=='w')
						update(a+2,0,1,n,1);
					if(str[a]=='b'&&str[a+1]=='b'&&str[a+2]=='w')
						update(a+2,1,1,n,1);
				}
				str[a]=ch[0];

			}

		}
	}
	return 0;
}

HDU—4046 Panda (线段树)的更多相关文章

  1. hdu 4046 Panda [线段树]

    Panda Time Limit: 10000/4000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Subm ...

  2. HdU 4046 Panda 段树

    链接:http://acm.hdu.edu.cn/showproblem.php?pid=4046 意甲冠军:到了bw组成的长度为n的字符串(n<=50000).有m次操作(m<=1000 ...

  3. HDU 4046 Panda(树状数组)

    Panda Time Limit: 10000/4000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Sub ...

  4. HDU 4046 Panda (ACM ICPC 2011北京赛区网络赛)

    HDU 4046 Panda (ACM ICPC 2011北京赛区网络赛) Panda Time Limit: 10000/4000 MS (Java/Others)    Memory Limit: ...

  5. hdu 4031 attack 线段树区间更新

    Attack Time Limit: 5000/3000 MS (Java/Others)    Memory Limit: 65768/65768 K (Java/Others)Total Subm ...

  6. hdu 4288 离线线段树+间隔求和

    Coder Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Su ...

  7. hdu 3016 dp+线段树

    Man Down Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total S ...

  8. HDU 5877 dfs+ 线段树(或+树状树组)

    1.HDU 5877  Weak Pair 2.总结:有多种做法,这里写了dfs+线段树(或+树状树组),还可用主席树或平衡树,但还不会这两个 3.思路:利用dfs遍历子节点,同时对于每个子节点au, ...

  9. HDU 3308 LCIS (线段树区间合并)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3308 题目很好懂,就是单点更新,然后求区间的最长上升子序列. 线段树区间合并问题,注意合并的条件是a[ ...

随机推荐

  1. es批量索引

    使用Python操作Elasticsearch数据索引的教程 这篇文章主要介绍了使用Python操作Elasticsearch数据索引的教程,Elasticsearch处理数据索引非常高效,要的朋友可 ...

  2. Java-Class-FC:java.lang.StringBuilder

    ylbtech-Java-Class-FC:java.lang.StringBuilder 1.返回顶部   2.返回顶部 1. @Override public String toString() ...

  3. HTTPS 加密机制

    目录 1. HTTPS 概述 2. 对称加密 3. 非对称加密 4. 非对称加密改良方案 5. 非对称加密 + 对称加密 6. 中间人攻击 7. 数字证书 8. 数字签名 9. HTTPS 工作原理 ...

  4. inode缓存与dentry缓存

    1. inode缓存 1: struct inode { 2: /* RCU path lookup touches following: */ 3: umode_t i_mode; 4: uid_t ...

  5. jQuery 实现复选框全选、反选及获取选中的值

    实现复选框全选,反选及获取选中的值: 代码如下: <!doctype html> <html lang="en"> <head> <met ...

  6. Netty 源码学习——客户端流程分析

    Netty 源码学习--客户端流程分析 友情提醒: 需要观看者具备一些 NIO 的知识,否则看起来有的地方可能会不明白. 使用版本依赖 <dependency> <groupId&g ...

  7. struts基础

    六个基本包 struts2-core-2.1.6.jar :开发的核心类库 freemarker-2.3.13.jar :struts2的UI标签的模板使用freemarker编写 commons-l ...

  8. erlang在windows下和虚拟机节点通信

    版权声明:博客将逐步迁移到 http://cwqqq.com https://blog.csdn.net/cwqcwk1/article/details/24738599 在Linux下部署erlan ...

  9. 如何让contenteditable元素只能输入纯文本

    本文出自张旭鑫博客,要知详情,请戳右侧地址:http://www.zhangxinxu.com/wordpress/?p=5120 一.温故而知新 很多年以前,稍等,让我搜一下contentedita ...

  10. GetModuleHandleW 分析

    首先查询MSDN,可以清楚地看到 位于kernel32 dll 里面. 有目标就好办,找到这个dll,然后,开工,进入IDA. 跳啊 就到下面那块了. 遗憾的是...显然不是这里阿,实际上下一块调用的 ...