3343: 教主的魔法

Time Limit: 10 Sec  Memory Limit: 256 MB
Submit: 631  Solved: 272
[Submit][Status][Discuss]

Description

教主最近学会了一种神奇的魔法,能够使人长高。于是他准备演示给XMYZ信息组每个英雄看。于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1、2、……、N
每个人的身高一开始都是不超过1000的正整数。教主的魔法每次可以把闭区间[LR](1≤LRN)内的英雄的身高全部加上一个整数W。(虽然L=R时并不符合区间的书写规范,但我们可以认为是单独增加第LR)个英雄的身高)
CYZ、光哥和ZJQ等人不信教主的邪,于是他们有时候会问WD闭区间 [LR] 内有多少英雄身高大于等于C,以验证教主的魔法是否真的有效。
WD巨懒,于是他把这个回答的任务交给了你。
 

Input

       第1行为两个整数NQQ为问题数与教主的施法数总和。
       第2行有N个正整数,第i个数代表第i个英雄的身高。
       第3到第Q+2行每行有一个操作:
(1)       若第一个字母为“M”,则紧接着有三个数字LRW。表示对闭区间 [LR] 内所有英雄的身高加上W
(2)       若第一个字母为“A”,则紧接着有三个数字LRC。询问闭区间 [LR] 内有多少英雄的身高大于等于C
 

Output

       对每个“A”询问输出一行,仅含一个整数,表示闭区间 [LR] 内身高大于等于C的英雄数。
 

Sample Input

5 3
1 2 3 4 5
A 1 5 4
M 3 5 1
A 1 5 4

Sample Output

2
3

HINT

【输入输出样例说明】
原先5个英雄身高为1、2、3、4、5,此时[1, 5]间有2个英雄的身高大于等于4。教主施法后变为1、2、4、5、6,此时[1, 5]间有3个英雄的身高大于等于4。
 
【数据范围】
对30%的数据,N≤1000,Q≤1000。
对100%的数据,N≤1000000,Q≤3000,1≤W≤1000,1≤C≤1,000,000,000。

  新开一个数组b
  b中的每块中的元素有序
  每次updata:整块tag记录,小块暴力修改
  求query:小块暴力,大块二分.
 #include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath> #define maxn 1000001 using namespace std; int n,a[maxn],team[maxn],b[maxn],str[maxn],end[maxn],tag[]; inline int in()
{
int x=;char ch=getchar();
while(ch<''||ch>'')ch=getchar();
while(ch<=''&&ch>='')x=x*+ch-'',ch=getchar();
return x;
} void pre()
{
int T=sqrt(n),cnt;
for(int i=;i<=n;i++)
team[i]=(i-)/T+;
cnt=team[n];
for(int i=;i<=n;i++)end[team[i]]=i,b[i]=a[i];
for(int i=n;i>=;i--)str[team[i]]=i;
for(int i=;i<=cnt;i++)
sort(str[i]+b,b+end[i]+);
} void add(int l,int r,int c)
{
if(team[l]==team[r]){
for(int i=l;i<=r;i++)a[i]+=c;
for(int i=str[team[l]];i<=end[team[r]];i++)b[i]=a[i];
sort(str[team[l]]+b,end[team[l]]+b);
return;
}
for(int i=l;i<=end[team[l]];i++)
a[i]+=c;
for(int j=str[team[r]];j<=r;j++)
a[j]+=c;
for(int i=str[team[l]];i<=end[team[l]];i++)b[i]=a[i];
for(int i=str[team[r]];i<=end[team[r]];i++)b[i]=a[i];
sort(str[team[l]]+b,end[team[l]]+b+);
sort(str[team[r]]+b,end[team[r]]+b+);
for(int i=team[l]+;i<=team[r]-;i++)
tag[i]+=c;
} int binary_search(int pos,int c)
{
int l=str[pos],r=end[pos];
if(b[r]<c)return r+;
while(l<=r)
{
int mid=(l+r)>>;
if(b[mid]<c)l=mid+;
else r=mid-;
}
return l;
} int query(int l,int r,int c)
{
int ans=;
if(team[l]==team[r]){
for(int i=l;i<=r;i++)if(a[i]+tag[team[l]]>=c)ans++;
return ans;
}
for(int i=l;i<=end[team[l]];i++)
if(a[i]+tag[team[l]]>=c)ans++;
for(int j=str[team[r]];j<=r;j++)
if(a[j]+tag[team[r]]>=c)ans++;
for(int i=team[l]+;i<=team[r]-;i++)
ans+=end[i]-binary_search(i,c-tag[i])+;
return ans;
} int main()
{
int q,u,v,c;
char sb;
n=in();q=in();
for(int i=;i<=n;i++)a[i]=in();
pre();
for(int i=;i<=q;i++)
{
sb=;
while(sb!='A'&&sb!='M')sb=getchar();
u=in(),v=in(),c=in();
if(sb=='A')printf("%d\n",query(u,v,c));
else add(u,v,c);
}
return ;
}

最新文章

  1. C语言中 指向函数的指针 简介
  2. js中typeof和instanceof
  3. Android-adb指令
  4. Spring day01笔记
  5. CAS实现无锁模式
  6. tomcat datasource
  7. UVa 12627 (递归 计数 找规律) Erratic Expansion
  8. java计算两个日期相差多少天
  9. doT模板
  10. Python初学——多线程Threading
  11. Java 拓展之调用其他语言
  12. xshell6 评估期已过 解决办法
  13. P3396 哈希冲突
  14. WinRAR从入门到高级的操作技巧集合
  15. 【原创】大叔经验分享(37)CM清理磁盘空间
  16. 关于HttpModule和HttpHandler以及HttpApplication
  17. java学习笔记(二):枚举值
  18. runtime.Gosched()的作用分析
  19. 搭建eclipse开发环境
  20. python安装与IO编程

热门文章

  1. Java Concurrency - 浅析 Phaser 的用法
  2. 【转】Log4.NET mark
  3. fuse挂载hdfs目录到linux本地
  4. Agile.Net 组件式开发平台 - 组件开发示例
  5. 【转载】干货来袭!Linux小白最佳实践:《超容易的Linux系统管理入门书》(连载七)LAMP集成安装
  6. LINQ中实现 In 与 Not In
  7. DDL_数据库模式定义语言
  8. 关于apache Alias斜杠/的实验
  9. javascript之DOMReady
  10. js-shortid:优雅简洁地实现短ID