SP1108题解——二分和前缀和的应用-创新互联
第一次写题解,求赞求关注qwq
成都创新互联是一家专注网站建设、网络营销策划、小程序定制开发、电子商务建设、网络推广、移动互联开发、研究、服务为一体的技术型公司。公司成立10余年以来,已经为上1000+柔性防护网各业的企业公司提供互联网服务。现在,服务的上1000+客户与我们一路同行,见证我们的成长;未来,我们一起分享成功的喜悦。原题地址~
为了更好的理解题意我们可以看一下这个模拟:
假设一开始有 5 5 5 张牌,初始顺序分别是 1 , 2 , 3 , 4 , 5 1,2,3,4,5 1,2,3,4,5。
第 1 1 1 次把最顶层的 1 1 1 依次拿到下面,顺序变成 2 , 3 , 4 , 5 , 1 2,3,4,5,1 2,3,4,5,1;接着把最上面的 2 2 2 拿走,第一轮操作后的数列为 3 , 4 , 5 , 1 3,4,5,1 3,4,5,1。
第 2 2 2 次把最顶层的 3 , 4 3,4 3,4 依次拿到下面,顺序变成 5 , 1 , 3 , 4 5,1,3,4 5,1,3,4;接着把最上面的 5 , 1 5,1 5,1 拿走,第一轮操作后的数列为 3 , 4 3,4 3,4。
最后一次需要进行 3 3 3 次操作,可是只有 2 2 2 个数,怎么办呢?把 3 , 4 3,4 3,4 依次拿到下面三次,也就是 3 , 4 → 4 , 3 → 3 , 4 → 4 , 3 3,4 \rightarrow 4,3 \rightarrow 3,4 \rightarrow 4,3 3,4→4,3→3,4→4,3 。
最后的数列为 2 , 5 , 1 , 4 , 3. 2,5,1,4,3. 2,5,1,4,3.
而题目想让你求的是使最后的数列单调上升,即最后的数列为 1 , 2 , 3 , 4 , 5 1,2,3,4,5 1,2,3,4,5.
我们再用样例来测试一下, n = 5 n = 5 n=5,初始顺序为 3 , 1 , 4 , 5 , 2 3,1,4,5,2 3,1,4,5,2.
第 1 1 1 次把最顶层的 3 3 3 依次拿到下面,顺序变成 1 , 4 , 5 , 2 , 3 1,4,5,2,3 1,4,5,2,3;接着把最上面的 1 1 1 拿走,第一轮操作后的数列为 4 , 5 , 2 , 3 4,5,2,3 4,5,2,3。
第 2 2 2 次把最顶层的 4 , 5 4,5 4,5 依次拿到下面,顺序变成 2 , 3 , 4 , 5 2,3,4,5 2,3,4,5;接着把最上面的 2 , 3 2,3 2,3 拿走,第一轮操作后的数列为 4 , 5 4,5 4,5。
最后一次把 4 , 5 4,5 4,5 依次拿到下面,也就是 4 , 5 → 5 , 4 → 4 , 5 4,5 \rightarrow 5,4 \rightarrow 4,5 4,5→5,4→4,5 。
最后的数列刚好为 1 , 2 , 3 , 4 , 5 1,2,3,4,5 1,2,3,4,5,符合题意。XD
理解了题意,我们看数据范围:
n
≤
2
×
1
0
4
n \le 2 \times 10^4
n≤2×104 ,另一篇题解用
O
(
能
过
)
O(能过)
O(能过) 的暴力,很明显运用了区间求和的操作,这时候我们会想到 前缀和/树状数组/线段树。
因为我有强迫症刚学树状数组,所以我就自然要用求和复杂度为
O
(
log
n
)
\mathcal{O} (\operatorname{log}n)
O(logn) 的树状数组解决啦~
附赠树状数组板子:
// P.S:f为树状数组
int lowbit(int i){return i&-i;
}
int getsum(int n){int ans = 0;
while(n>0){ans+=f[n];
n-=lowbit(n);
}
return ans;
}
void change(int i,int d,int n){while(i<=n){f[i]+=d;
i+=lowbit(i);
}
}
通过题目得知最后的序列是单调上升的,还原时自然会想到复杂度为 O ( log n ) \mathcal{O}(\operatorname{log} n) O(logn) 的二分来处理前缀和。
int BinarySearch(int val,int n){int l = 1,r = n;
while(lint mid = (l+r)>>1;
if(getsum(mid)>=val) r = mid;
else l = mid+1;
}
return l;
}
所以这种做法总的时间复杂度是 O ( n l o g 2 n ) \mathcal{O}(n \operatorname{log^2} n) O(nlog2n)。
最后就是核心代码啦,注释都在代码里。
void work(int n){memset(f,0,sizeof f);//多测不清空,抱灵两行泪
int p = 0,c = n,pl,pr,step;//p是当前处理的元素(就是指针)
for(int i = 1;i<=n;i++){change(i,1,n);//将初始的1,2,3...n存入到树状数组中
}
for(int i = 1;i<=n;i++){step = (i+1)%c;//处理步数比元素多的情况
pl = getsum(p),pr = c-pl;//需要更改的范围
int nxt;
nxt = (step<=pr)?step+pl:step-pr;二分范围
if(nxt==0) nxt = c;
p = BinarySearch(nxt,n);
ans[p] = i;//填入答案
change(p,-1,n);//填完了~
c--;
}
for(int i = 1;i<=n;i++) cout<
代码就不提供了,主函数没有什么东西了qwq
你是否还在寻找稳定的海外服务器提供商?创新互联www.cdcxhl.cn海外机房具备T级流量清洗系统配攻击溯源,准确流量调度确保服务器高可用性,企业级服务器适合批量采购,新人活动首月15元起,快前往官网查看详情吧
文章名称:SP1108题解——二分和前缀和的应用-创新互联
文章位置:http://pcwzsj.com/article/dgshpi.html