luogu P4062 [Code+#1]Yazid 的新生舞会(线段树+套路)
今天原来是平安夜啊
感觉这题是道好题。
一个套路枚举权值\(x\),把权值等于\(x\)的设为1,不等于的设为-1,然后问题转化为多少个区间权值和大于。
发现并不是很好做,还有一个套路,用前缀和查分来表示区间。然后就是
\]
\]
然后树状数组可以做\(a[i]\leq7\)的数据。
那么\(a[i]\)那么大该怎么办?
考虑我们构建的\(1,-1\)数列中连续-1的数列很多。
然后这些连续-1不会互相影响的贡献,然后我们考虑直接算出这些连续-1的贡献。
设这段连续-1的两端的sum为\(l,r\)。贡献为:
\]
然后我们在线段树上维护\(i\)和\(cnt[i]\),然后用线段树可以把这个-1序列\(logn\)处理掉。
然后每一个1单独处理,最后总复杂度\(O(nlogn)\)
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<vector>
using namespace std;
#define int long long
#define mid ((l+r)>>1)
#define ls now<<1
#define rs now<<1|1
const int N=501000;
vector<int> vec[N];
int sum[N*2*5],sumi[N*2*5],lazy[N*2*5],a[N],b[N],last,lastsum,ans,n;
int read(){
int sum=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){sum=sum*10+ch-'0';ch=getchar();}
return sum*f;
}
void update(int now){
sum[now]=sum[ls]+sum[rs];
sumi[now]=sumi[ls]+sumi[rs];
}
void pushdown(int l,int r,int now){
if(lazy[now]==0)return;
if(l==r)return;
sum[ls]+=(mid-l+1)*lazy[now];
sumi[ls]+=(mid+l)*(mid-l+1)/2*lazy[now];
sum[rs]+=(r-(mid+1)+1)*lazy[now];
sumi[rs]+=(r+mid+1)*(r-(mid+1)+1)/2*lazy[now];
lazy[ls]+=lazy[now];
lazy[rs]+=lazy[now];
lazy[now]=0;
}
void add(int l,int r,int L,int R,int k,int now){
if(R<l||L>r)return;
pushdown(l,r,now);
if(L<=l&&r<=R){
sum[now]+=(r-l+1)*k;
sumi[now]+=(l+r)*(r-l+1)/2*k;
lazy[now]+=k;
return;
}
add(l,mid,L,R,k,ls);
add(mid+1,r,L,R,k,rs);
update(now);
}
int check(int l,int r,int L,int R,int now){
if(R<l||L>r)return 0;
pushdown(l,r,now);
if(L<=l&&r<=R)return sum[now];
return check(l,mid,L,R,ls)+check(mid+1,r,L,R,rs);
}
int checki(int l,int r,int L,int R,int now){
if(R<l||L>r)return 0;
pushdown(l,r,now);
if(L<=l&&r<=R)return sumi[now];
return checki(l,mid,L,R,ls)+checki(mid+1,r,L,R,rs);
}
signed main(){
n=read();int hh=read();
for(int i=1;i<=n;i++)a[i]=read(),b[i]=a[i];
sort(b+1,b+1+n);
int tot=unique(b+1,b+1+n)-b-1;
for(int i=1;i<=n;i++)a[i]=lower_bound(b+1,b+1+tot,a[i])-b;
for(int i=1;i<=n;i++)vec[a[i]].push_back(i);
add(1,2*n,n,n,1,1);
for(int i=1;i<=tot;i++){
vec[i].push_back(n+1);
last=0;lastsum=0;
for(int j=0;j<vec[i].size();j++){
int L=vec[i][j]-last-1;
int l=lastsum-L;
int r=lastsum-1;
if(vec[i][j]!=last+1){
ans+=L*check(1,2*n,1,l+n-1,1);
ans+=(r+n)*check(1,2*n,l+n,r+n,1);
ans-=checki(1,2*n,l+n,r+n,1);
add(1,2*n,l+n,r+n,1,1);
}
if(vec[i][j]==n+1)continue;
ans+=check(1,2*n,1,l+n,1);
add(1,2*n,l+1+n,l+1+n,1,1);
last=vec[i][j];lastsum=l+1;
}
last=0;lastsum=0;
for(int j=0;j<vec[i].size();j++){
int L=vec[i][j]-last-1;
int l=lastsum-L;
int r=lastsum-1;
if(vec[i][j]!=last+1)add(1,2*n,l+n,r+n,-1,1);
if(vec[i][j]==n+1)continue;
add(1,2*n,l+1+n,l+1+n,-1,1);
last=vec[i][j];lastsum=l+1;
}
}
printf("%lld\n",ans);
return 0;
}
luogu P4062 [Code+#1]Yazid 的新生舞会(线段树+套路)的更多相关文章
- 【BZOJ5110】[CodePlus2017]Yazid 的新生舞会 线段树
[BZOJ5110][CodePlus2017]Yazid 的新生舞会 Description Yazid有一个长度为n的序列A,下标从1至n.显然地,这个序列共有n(n+1)/2个子区间.对于任意一 ...
- P4062 [Code+#1]Yazid 的新生舞会
思路:分治 提交:2次 错因:数组开小 题解: 我们枚举一下众数\(x\). 设\(s[n]=\sum_{i=1}^n [a[i]==x]\) 那么对于区间\((l,r]\),有\(s[r]-s[l] ...
- 洛谷 P4062 - [Code+#1]Yazid 的新生舞会 的线性做法
洛谷题面传送门 一个线性做法. \(n\log n\) 解法可以戳这里查看 首先回顾一下 \(n\log n\) 解法的过程:我们对于每一个数 \(x\),考察其出现位置,设为 \(t_1,t_2,t ...
- 洛谷 P4062 - [Code+#1]Yazid 的新生舞会(权值线段树)
题面传送门 题意: 给出一个序列 \(a\),求 \(a\) 有多少个子区间 \([l,r]\),满足这个区间中出现次数最多的数出现次数 \(>\dfrac{r-l+1}{2}\) \(1 \l ...
- BZOJ.5110.[CodePlus2017]Yazid 的新生舞会(线段树/树状数组/分治)
LOJ BZOJ 洛谷 又来发良心题解啦 \(Description\) 给定一个序列\(A_i\).求有多少个子区间,满足该区间众数出现次数大于区间长度的一半. \(n\leq5\times10^5 ...
- 【线段树】【P4062】 [Code+#1]Yazid 的新生舞会
Description 给定一个长度为 \(n\) 的序列,求有多少子区间满足区间众数严格大于区间长度的一半.如果区间有多个出现次数最多且不同的数则取较小的数为众数. Limitation 对于全部的 ...
- [题解] [Code+#1]Yazid 的新生舞会
题面 题解 upd : \(cnt_i\) 代表值为 \(i\) 的个数 我们可以暴力枚举众数 \(k\) 把等于 \(k\) 的赋值成 1 , 不等于 \(k\) 的赋值成 -1 这样原序列就变成了 ...
- 「CodePlus 2017 11 月赛」Yazid 的新生舞会(树状数组/线段树)
学习了新姿势..(一直看不懂大爷的代码卡了好久T T 首先数字范围那么小可以考虑枚举众数来计算答案,设当前枚举到$x$,$s_i$为前$i$个数中$x$的出现次数,则满足$2*s_r-r > 2 ...
- luogu P3919 [模板]可持久化数组(可持久化线段树/平衡树)(主席树)
luogu P3919 [模板]可持久化数组(可持久化线段树/平衡树) 题目 #include<iostream> #include<cstdlib> #include< ...
随机推荐
- 编程范式(Programming Paradigm)-[ 程序员的编程世界观 ]
编程范式(Programming Paradigm)是某种编程语言典型的编程风格或者说是编程方式.随着编程方法学和软件工程研究的深入,特别是OO思想的普及,范式(Paradigm)以及编程范式等术语渐 ...
- 页面定制CSS代码初探(二):自定义h2标题样式 添加阴影 添加底色 等
故事的开始 先说一下<h2></h2>原先默认是空白的,很难看 然后今天无意中看到一个博友的标题很好看啊,一直就想要这种效果有没有? 好的东西自然要拿过来啦 通过审查元素,果然 ...
- BZOJ 2150 部落战争 (二分图匹配)
题目大意:给你一个n*m的棋盘,有一些坏点不能走,你有很多军队,每支军队可以像象棋里的马一样移动,不过马是1*2移动的,而军队是r*c移动的,军队只能从上往下移动,如果一个点已经被一直军队经过,那么其 ...
- Rsync和Sersync(企业实时同步方案)
注:本文章依据参考文章中的信息资料结合自己的实践操作而成 一.实验环境介绍 系统版本:Cent OS 7.4 X64 内核版本:3.10.0-693.5.2.el7.x86_64 系统采用最小化安装, ...
- 在centos系统中增、删交换分区
Swap介绍: Linux 将物理内存分为内存段,叫做页面.交换是指内存页面被复制到预先设定好的硬盘空间(叫做交换空间)的过程,目的是释放对于页面的内存.物理内存和交换空间的总大小是可用的虚拟内存的总 ...
- CSS的引入方式和样式
CSS的引入方式和样式 一.样式 行内样式 内接样式 外接样式(1.链接式 2.导入式) <!--行内样式--> <div> <p style="color: ...
- Spring MVC中 提交表单报错400
背景: 在写SpringMVC表单提交的代码的时,在最后点击提交的时候总是会出现400的错误 原因: 主要原因就是表单提交的数据和对应实体类的属性无法完全匹配 解决方案: 查看我们提交的数据是否完全和 ...
- C语言 将十六进制字符串转为十六进制数 (二进制、十进制都适用)
主要利用 long int strtol(const char *nptr,char **endptr,int base); 函数 函数说明: 参数base范围从2至36,或0.参数base代表采用的 ...
- Thunder9(迅雷9)去掉右侧浏览器广告的方法
1.打开文件夹C:\Program Files (x86)\Thunder Network\Thunder9\Program\TBC 2.找到 ThunderBrowser.exe 3.重命名为任意名 ...
- static方法调用
Static方法调用,类名.方法名 int number = Integer.ParseInt(String ); 将字符串参数作为有符号的十进制整数进行解析 将数字解析成字节数组 Character ...