読者です 読者をやめる 読者になる 読者になる

beet's soil

競プロのことなど

AtCoder Grand Contest 008

AtCoderコンテスト納め

【A - Simple Calculator】
つらい問題だった、、、、
絶対値と正負で場合分けする

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
int main(){
  ll x,y;cin>>x>>y;
  if(x<y){
    if(0<=x||y<=0){
      cout<<y-x<<endl;
    }else{
      cout<<min(1+abs(abs(x)-abs(y)),y-x)<<endl;
    }
  }else{
    if(x<0||0<y){
      cout<<2+x-y<<endl;
    }else{
      cout<<1+abs(abs(x)-abs(y))<<endl;
    }
  }
  return 0;
}


【B - Contiguous Repainting】
最後のK個以外は自由に塗れるので累積和でえいってやる。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
int main(){
  ll n,k;
  cin>>n>>k;
  vector<ll> a(n),b(n+1,0),c(n+1,0);
  for(ll i=0;i<n;i++) cin>>a[i];
  for(ll i=0;i<n;i++){
    b[i+1]=b[i]+(a[i]>0?a[i]:0);
    c[i+1]=c[i]+a[i];
  }
  ll ans=0,tmp=0;
  for(int i=0;i<=n-k;i++){
    tmp=b[n]-b[i+k]+b[i];
    ans=max(ans,tmp);
    tmp+=c[i+k]-c[i];
    ans=max(ans,tmp);
  }
  cout<<ans<<endl;
  return 0;
}


【C - Tetromino Tiling】
時間切れで通せなかった。
O(1)なので先にやればよかった。
組み合わせを一個取るか全部二個ずつかにする。

#include<bits/stdc++.h>
using namespace std;
#define int long long
signed main(){
  int a[7];
  for(int i=0;i<7;i++) cin>>a[i];
  int ans,tmp=0;
  ans=a[1]+a[0]/2*2+a[3]/2*2+a[4]/2*2+(a[0]%2&&a[3]%2&&a[4]%2)*3;
  if(a[0]&&a[3]&&a[4]) tmp=a[1]+(a[0]-1)/2*2+(a[3]-1)/2*2+(a[4]-1)/2*2+3;
  cout<<max(ans,tmp)<<endl;
  return 0;
}

【D - K-th K】
配点と通してる人の数から先にDをやった。
ソートして前と後ろから埋めていく。
assertかけて提出デバッグしてしまったのでペナルティがひどいことになってしまった。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int,int> P;
int main(){
  int n,p;
  cin>>n;
  int x[n],a[n*n],u[n];
  vector<P> v;
  for(int i=0;i<n*n;i++) a[i]=0;
  for(int i=0;i<n;i++){
    cin>>x[i];
    u[i]=1;
    a[x[i]-1]=i+1;
    v.push_back(P(x[i]-1,i+1));
  }
  sort(v.begin(),v.end());
  bool f=1;
  p=0;
  for(int i=0;i<n;i++){
    int c=v[i].second;
    while(u[c-1]<c){
      if(a[p]){
	p++;
	continue;
      }
      if(p>v[i].first){
	f=0;
	break;
      }
      a[p]=c;
      u[c-1]++;
    }
  }
  reverse(v.begin(),v.end());
  p=n*n-1;
  for(int i=0;i<n;i++){
    int c=v[i].second;
    while(u[c-1]<n){
      while(a[p]) p--;
      if(p<v[i].first){
	f=0;
	break;
      }
      a[p]=c;
      u[c-1]++;
    }
  }

  //assertion
  if(f){
    for(int i=1;i<=n;i++){
      int t=0;
      for(int j=0;j<n*n;j++){
	if(a[j]==i) t++;
	if(i==t) {
	  f&=(j+1==x[i-1]);
	  break;
	}
      }
    }
  }
  if(f){
    for(int i=1;i<=n;i++){
      int t=0;
      for(int j=0;j<n*n;j++){
	if(a[j]==i) t++;
      }
      //cout<<i<<" "<<t<<endl;
      f&=(t==n);
    }
  }
  if(!f){
    cout<<"No"<<endl;
  }else{
    cout<<"Yes"<<endl;
    for(int i=0;i<n*n;i++) cout<<a[i]<<" \n"[i==n*n-1];
    //for(int i=0;i<n*n;i++) cout<<i<<":"<<a[i]<<endl;
  }
  return 0;
}


【全体】
1842 -> 1880 (+38)
正の変化で終われたのでよかった。
あと5分あればコンテスト中にC通せたなあという気持ち、、
来年も精進します