首页 > 其他分享 >[HEOI2016TJOI2016]排序

[HEOI2016TJOI2016]排序

时间:2023-10-13 19:12:21浏览次数:33  
标签:rs int void HEOI2016TJOI2016 mid ls 排序 change

P2824 [HEOI2016/TJOI2016] 排序

直接模拟复杂度爆炸,有观察到它只要求一个数。

思维十分清奇。

我们先考虑一个序列,如果全是 0/1,该怎么做。

发现这个问题很好做,修改区间时只需要先查询一的个数,然后将前面/后面全部置1,其他置0。

然后我们考虑怎么转化。

发现可以二分答案,对于小于mid的就认为0,大于等于mid的就认为1。(因为它是排列,都是连续的值)

然后最后执行完操作后,如果为1,说明答案在mid及以上的范围内,如果为0,说明答案在mid以下。

要维护01序列,用线段树,然后区间修改区间查询,最后单点查询位置q的值。

复杂度 \(O(m\log^2n+n\log n)\)。

#include<iostream>
#define ls p<<1
#define rs p<<1|1
#define up v[p]=v[ls]+v[rs]
using namespace std;
const int N=100010,M=4*N;
int n,m,Q,l[M],r[M],v[M],a[N],mid;
char f[M];
struct ch{
	bool op;
	int l,r;
}q[N];
void build(int p,int L,int R){
	l[p]=L,r[p]=R,f[p]=-1;
	if(L==R){
		v[p]=a[L]>=mid;
		return;
	}
	int mid=L+R>>1;
	build(ls,L,mid),build(rs,mid+1,R);
	up;
}
void change(int p,bool t){
	v[p]=t?r[p]-l[p]+1:0;
	f[p]=t;
}
void down(int p){
	char &t=f[p];
	if(~t){
		change(ls,t),change(rs,t);
		t=-1;
	}
}
void update(int p,int L,int R,bool t){
	if(L<=l[p]&&r[p]<=R){
		change(p,t);
		return;
	}
	down(p);
	int mid=l[p]+r[p]>>1;
	if(L<=mid)update(ls,L,R,t);
	if(R>mid)update(rs,L,R,t);
	up;
}
int query(int p,int L,int R){
	if(L<=l[p]&&r[p]<=R)return v[p];
	down(p);
	int mid=l[p]+r[p]>>1,ans=0;
	if(L<=mid)ans=query(ls,L,R);
	if(R>mid)ans+=query(rs,L,R);
	return ans;
}
bool check(int mid){
	::mid=mid;
	build(1,1,n);
	// cout<<v[1]<<'\n';
	for(int i=1;i<=m;++i){
		auto[op,l,r]=q[i];
		// cout<<l<<' '<<r<<'\n';
		int t1=query(1,l,r);
		// cout<<t1<<'\n';
		if(op)t1&&(update(1,l,l+t1-1,1),0),
		l+t1<=r&&(update(1,l+t1,r,0),0);
		else t1&&(update(1,r-t1+1,r,1),0),l+t1<=r&&(update(1,l,r-t1,0),0);
	}
	return query(1,Q,Q);
}
int main(){
	ios::sync_with_stdio(0);
	cin.tie(0),cout.tie(0);
	cin>>n>>m;
	for(int i=1;i<=n;++i)cin>>a[i];
	for(int i=1;i<=m;++i)cin>>q[i].op>>q[i].l>>q[i].r;
	cin>>Q;
	int l=1,r=n;
	while(l<r){
		int mid=l+r+1>>1;
		// cout<<mid<<'\n';
		if(check(mid))l=mid;
		else r=mid-1;
	}
	cout<<r;
	return 0;
}

笑话

本题使用桶排与氧气反应可以得到100分(Unaccepted)

标签:rs,int,void,HEOI2016TJOI2016,mid,ls,排序,change
From: https://www.cnblogs.com/wscqwq/p/17762940.html

相关文章

  • 插入排序
    原数组为915623 1voidprint(vector<int>&a,intn,inti){2cout<<"step"<<i<<":";3for(intj=0;j<n;j++){4cout<<a[j]<<"";5}6......
  • 选择排序
    核心思想:以第k趟为例:将余下的元素最小者放在第k个位置,如果这个最小者原本不在第k个位置则需要和第k个位置上的元素交换1voidselectSort(vector<int>&nums){2intlen=nums.size();3intminIndex=0;4for(inti=0;i<len;++i){5minIndex=i;6......
  • 冒泡排序
    冒泡排序算法原理1、每一次循环结束之后,都要找出最大的数据,放到参与比较的这堆数据的最右边。(冒出最大的那个气泡)2、 拿着左边的数字和右边的数字比对,当左边>右边的时候,交换位置。例如:9,8,10,7,6第1次循环:比较的数据981076891076第1次比较:交换891076......
  • Shell(五):文件的排序、合并和分割
    Linux文本处理命令是Shell编程中的常用命令,文本处理包含对文件记录的排序、文件的合并和分割等。1、sort命令sort命令是一种对文件排序的工具,sort命令将输入文件看做由多条记录组成的数据流,而记录由可变宽度的字段组成,以换行符作为定界符。sort命令,可将记录分成多......
  • java算法之排序算法大全
    ①排序所谓排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。排序算法,就是如何使得记录按照要求排列的方法。排序算法在很多领域得到相当地重视,尤其是在大量数据的处理方面。一个优秀的算法可以节省大量的资源。在各个领域中考虑到数据的各种限制......
  • 计算机程序设计艺术(第3卷)-排序和查找(英文影印版) pdf电子版epub
    计算机程序设计艺术(第3卷)-排序和查找(英文影印版)pdf电子版epub作者: (美)DonaldE.KnuthISBN: 9787302058168点击下l载数学分析算法,没有比这更好的了......
  • mysql复制数据库,数据库排序规则不一致问题
    mysql复制数据库步骤1.导出数据库sql文件mysqldump数据库名-h数据库地址-P数据库端口(3306可省略)-u账号-p密码--add-drop-table>/路径/sql文件名.sql 2.确认导出和导入数据库编码和排序规则是否一致showglobalvariableslike'%coll%'showglobalvariabl......
  • 基于凸多边形离散点排序的研究
    OrderBy(){varvertices1=_.cloneDeep(this.polygon);varxArray=vertices1.map((item)=>item.x);varyArray=vertices1.map((item)=>item.y);const[minX,maxX,minY,maxY]=[_.min(xArray),_.max(xArray),_.min(yArray),_.m......
  • python列表中的元素按照自身某个索引的元素排序
    title:aliases:-python列表按元素排序tags:-Python/数据处理category:stars:url:creation-time:2023-07-3115:26modification-time:#!/usr/bin/python#-*-coding:UTF-8-*-#获取列表的第二个元素deftakeSecond(elem):returnelem[1]#列表r......
  • rust HashMap 排序
    按照key和value升序、降序、自定义排序示例usestd::collections::HashMap;usestd::cmp::Ordering;fnmain(){letmutdf=HashMap::new();forxin5..=12{letk=format!("key_{}",x);letv=format!("value_{}",x);......