zoukankan      html  css  js  c++  java
  • AT2688 [ARC080C] Young Maids

    一道挺有意思的题目,在这里记录一下。

    题目大意

    给你一个长度为 (n) 的排列,每一次你可以取出相邻的两个数将其放在答案序列的开头,最后问你字典序最小的答案序列是什么。

    题解

    由于最后是求字典序最小,所以我们肯定需要倒序求解,所以我们需要考虑如何取才能使得剩下的序列合法。

    如果我们在区间 ([L,R]) 当前选择点 (l) 和点 (r) ,那么最后的区间很显然会被我们分成三部分, ([L,l-1])([l+1,r-1])([r+1,R]) 。而且这三个区间的长度必须为偶数才可以满足区间同时也可分。我们就可以用一个优先队列来模拟一下,每一次取出当前所以区间中可以选择字典最小的点对的区间,再将其拆分为三个区间。不断重复此操作直到最后所有的数都被取出即可。

    代码

    #include<bits/stdc++.h>
    using namespace std;
    const int N=2e5+5;
    int n,a[N],b[N];
    struct Point{int data,loc;};
    bool operator < (const Point a,const Point b){return a.data<b.data;}
    struct Seg_Tree
    {
    	struct Node{Point data;}tr[N<<2];
    	void up(int u){tr[u].data=min(tr[u<<1].data,tr[u<<1|1].data);}
    	void build(int u,int l,int r,int a[])
    	{
    		if(l==r)
    		{
    			tr[u].data.data=a[l];
    			tr[u].data.loc=l;
    			return ;
    		}
    		int mid=(l+r)>>1;
    		build(u<<1,l,mid,a);
    		build(u<<1|1,mid+1,r,a);
    		up(u);
    	}
    	void chg(int u,int l,int r,int x)
    	{
    		if(l==r){tr[u].data.data=1e9+7;return ;}
    		int mid=(l+r)>>1;
    		if(x<=mid) chg(u<<1,l,mid,x);
    		else chg(u<<1|1,mid+1,r,x);
    		up(u);
    	}
    	Point query(int u,int l,int r,int x,int y)
    	{
    		if(x<=l&&r<=y) return tr[u].data;
    		int mid=(l+r)>>1;Point res={1000000007,0};
    		if(x<=mid) res=min(res,query(u<<1,l,mid,x,y));
    		if(y>mid) res=min(res,query(u<<1|1,mid+1,r,x,y));
    		return res;
    	}
    }t[2];//0为偶,1为奇
    struct Data{int l,r;Point data;};
    bool operator < (const Data a,const Data b){return a.data.data>b.data.data;}
    priority_queue<Data> q;
    vector<int> res;
    int main()
    {
    	cin>>n;
    	for(int i=1,x;i<=n;++i)
    	{
    		scanf("%d",&x);
    		if(i&1) a[i]=1e9+7,b[i]=x;
    		else a[i]=x,b[i]=1e9+7;
    	}
    	t[0].build(1,1,n,a);
    	t[1].build(1,1,n,b);
    	Data tmp;
    	tmp.l=1,tmp.r=n;
    	tmp.data=t[1].query(1,1,n,1,n);
    	q.push(tmp);
    	for(int i=1;i<=n/2;++i)
    	{
    		tmp=q.top(),q.pop();
    		int tag=tmp.l%2;
    //		printf("%d
    ",tag);
    		Point L=tmp.data,R=t[tag^1].query(1,1,n,L.loc+1,tmp.r);
    //		printf("%d %d %d %d
    ",tmp.l,tmp.r,L.data,R.data);
    		res.push_back(L.data),res.push_back(R.data);
    		Data now;
    		now.l=tmp.l,now.r=L.loc-1;
    		if(now.l<=now.r)
    		{
    			now.data=t[tag].query(1,1,n,now.l,now.r);
    			q.push(now);
    		}
    		now.l=L.loc+1,now.r=R.loc-1;
    		if(now.l<=now.r)
    		{
    			now.data=t[tag^1].query(1,1,n,now.l,now.r);
    			q.push(now);
    		}
    		now.l=R.loc+1,now.r=tmp.r;
    		if(now.l<=now.r)
    		{
    			now.data=t[tag].query(1,1,n,now.l,now.r);
    			q.push(now);
    		}
    	}
    	for(int i=0;i<(int)res.size();++i) printf("%d ",res[i]);
    	printf("
    ");
    	return 0;
    }
    
  • 相关阅读:
    设计模式之建造者模式(简单)
    设计模式之简单工厂模式,工厂模式,抽象工厂模式
    设计模式之观察者模式
    Redis java操作客服端——jedis
    Redis基础
    java环境变量配置加maven配置
    设计模式之JDK动态代理
    springboot+mybatis+Thymeleaf
    OS问题管理系统开发
    复制CentOS虚拟机网络配置
  • 原文地址:https://www.cnblogs.com/Point-King/p/13993474.html
Copyright © 2011-2022 走看看