codeforces#1148E. Earth Wind and Fire(贪心)

 

题目链接:

http://codeforces.com/contest/1148/problem/E

题意:

 给出两个长度为$n$的序列,将第一个序列变成第二个序列,顺序不重要,只需要元素完全相同即可

只有一种修改操作

  • $a_i=a_i+d,a_j=a_j-d$满足 $a_j-a_i\geq 2 \cdot d$

数据范围:

 $1\leq n\leq 3e^{5}$

分析: 

先说结论:

  • 对$a,b$数组排序,结果一定是$a_i$变成$b_i$,也就是一一对应的关系
  • 有些数需要增加,有些数需要减少,第一个增加的数和第一个减少的数对应操作
  • 如果(需要减少的数的目标位置)小于(需要增大的数的目标位置),那么就输出$NO$

在草稿纸上模拟几次就好了,$QAQ$

ac代码:

#include<bits/stdc++.h>
#define ll long long
#define pa pair<int,int>
using namespace std;
const int maxn=3e5+10;
struct Ans
{
    int a,b,c;
}ans[maxn*5];
int cnt;
struct Move
{
    int id,now,to;
    bool operator <(const Move &a)const
    {
        if(to==a.to)return id<a.id;
        return to<a.to;
    }
};
pa a[maxn];
int b[maxn];
set<Move>se1,se2;
int main()
{
    int n;
    scanf("%lld",&n);
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&a[i].first);
        a[i].second=i;
    }
    for(int i=1;i<=n;i++)
        scanf("%d",&b[i]);
    sort(a+1,a+1+n);
    sort(b+1,b+1+n);
    for(int i=1;i<=n;i++)
    {
        //cout<<a[i].first<<" "<<b[i]<<endl;
        if(a[i].first<b[i])
            se1.insert((Move){a[i].second,b[i]-a[i].first,b[i]});
        else if(a[i].first>b[i])
            se2.insert((Move){a[i].second,a[i].first-b[i],b[i]});
        //cout<<se1.size()<<" "<<se2.size()<<endl;
    }
    while(1)
    {
        //cout<<se1.size()<<" "<<se2.size()<<endl;
        //cout<<"sadf0"<<endl;
        if(se1.size()==0&&se2.size()==0)break;
        if(se1.size()==0||se2.size()==0)
        {
            printf("NO\n");
            return 0;
        }
        Move x=(*se1.begin());
        Move y=(*se2.begin());
        if(x.to>y.to)
        {
            printf("NO\n");
            return 0;
        }
      //  cout<<x.id<<" "<<y.id<<endl;
        se1.erase(x);
        se2.erase(y);
        if(x.now==y.now)
        {
            ans[++cnt]=(Ans){x.id,y.id,x.now};
        }
        else if(x.now<y.now)
        {
            ans[++cnt]=(Ans){x.id,y.id,x.now};
            y.now-=x.now;
            se2.insert(y);
        }
        else if(x.now>y.now)
        {
            ans[++cnt]=(Ans){x.id,y.id,y.now};
            x.now-=y.now;
            se1.insert(x);
        }
    }
    printf("YES\n");
    printf("%d\n",cnt);
    for(int i=1;i<=cnt;i++)
        printf("%d %d %d\n",ans[i].a,ans[i].b,ans[i].c);
    return 0;
}

  

posted @ 2019-06-05 22:25  czh~  阅读(347)  评论(0编辑  收藏  举报