E. Permutation by Sum(思维+构造)

98 篇文章 3 订阅

https://codeforces.com/contest/1512/problem/E


思路:

首先考虑-1,那么我们把1+2+3....这种和n,n-1,n-2...这种,看一下是不是分别>s,<s。不然就能构造出来。因为答案在范围里嘛。

然后先拿最小的1 2 3 4 5..去构造,然后修正差值。每次从后面的这个最大的'5'开始,先减了,然后算>=和s的总差值的数,然后更新上去。n=500嘛..O(n)再去找总共也就O(n^2)。问题不大。

二分总复杂度O(nlogn)。

#include<iostream>
#include<vector>
#include<queue>
#include<cstring>
#include<cmath>
#include<map>
#include<set>
#include<cstdio>
#include<algorithm>
#define debug(a) cout<<#a<<"="<<a<<endl;
using namespace std;
const int maxn=550;
typedef long long LL;
inline LL read(){LL x=0,f=1;char ch=getchar();	while (!isdigit(ch)){if (ch=='-') f=-1;ch=getchar();}while (isdigit(ch)){x=x*10+ch-48;ch=getchar();}
return x*f;}
LL a[maxn];
multiset<LL>s1;
int main(void){
   cin.tie(0);std::ios::sync_with_stdio(false);
   LL t;cin>>t;
   while(t--){
      s1.clear();
      LL n,l,r,s;cin>>n>>l>>r>>s;
      for(LL i=1;i<=n;i++) s1.insert(i);
      LL len=r-l+1;
      for(LL i=0;i<=n+10;i++) a[i]=0;
      if((1+len)*len/2>s){
         cout<<"-1"<<"\n";
         continue;
      }
      if((n+(n-len+1))*len/2<s){
         cout<<"-1"<<"\n";
         continue;
      }
      LL cnt=s-(1+len)*len/2;
     // debug(cnt);
      LL num=0;
      for(LL i=l;i<=r;i++) a[i]=++num;
      multiset<LL>:: iterator it;
      for(LL i=r;i>=l;i--){
          if(cnt<=0){
            break;
          }
          cnt+=a[i];
          it=s1.lower_bound(cnt);
          a[i]=*it;
          cnt-=a[i];
          s1.erase(a[i]);
          ///s1.insert(a[i]);
      }
      for(LL i=l;i<=r;i++){
          if(s1.count(a[i])){
            s1.erase(a[i]);
          }
      }
      for(LL i=1;i<l;i++){
          cout<<(*s1.begin())<<" ";
          s1.erase(*s1.begin());
      }
      for(LL i=l;i<=r;i++){
          cout<<a[i]<<" ";
      }
      for(LL i=r+1;i<=n;i++){
          cout<<(*s1.begin())<<" ";
          s1.erase(*s1.begin());
      }
      cout<<"\n";
   }
   return 0;
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值