这题昨天晚上花了我1个小时50多分钟来搞,都没有搞定。。后来看别人代码,直接暴力枚举第一个数的值来做。。最多1000*1000的复杂度。当时怎么就没想到呢?还有为啥我的方法不对呢。。
暴力方法代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#define Mod 1000000007
using namespace std;
#define N 1007 int a[N]; int main()
{
int n,k,i,j,tmp,maxi,cnt,now,mini;
while(scanf("%d%d",&n,&k)!=EOF)
{
for(i=;i<=n;i++)
scanf("%d",&a[i]);
mini = Mod;
for(i=;i<=;i++)
{
cnt = ;
tmp = i;
for(j=;j<=n;j++)
{
if(a[j] != tmp)
cnt++;
tmp += k;
}
if(cnt < mini)
{
mini = cnt;
now = i;
}
}
printf("%d\n",mini);
tmp = now;
for(i=;i<=n;i++)
{
if(a[i] > tmp)
printf("- %d %d\n",i,a[i]-tmp);
else if(a[i] < tmp)
printf("+ %d %d\n",i,tmp-a[i]);
tmp += k;
}
}
return ;
}
自己的做法:
做一个参考数组b[],存放1,1+k,1+2k....1+(n-1)k,然后a,b两数组(a为原数组)个元素相减,得出差值数组cha,且将cha复制一份给cha2,将cha2从大到小排序,得出差值大于0的最大连续相同的差值(差值小于0肯定不行,因为从1开始的b数组是底线,比b还小说明会小于0,与题意不符),以最大连续差值的这些数为基准,这些数不变,将其他数变为使整个数组满足Ai+1-Ai = k,此时形成标准数组biao[],然后比较a[]和biao[],得出操作数和操作序列。但是就是不对,,不知道为啥,那位好心人如果发现这样那里不对欢迎向我指出,谢谢。
自己代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#define Mod 1000000007
using namespace std;
#define N 1007 int a[N],biao[N],cha[N],b[N],cha2[N]; int cmp(int ka,int kb)
{
return ka>kb;
} int main()
{
int n,k,i,j,flag;
while(scanf("%d%d",&n,&k)!=EOF)
{
for(i=;i<=n;i++)
scanf("%d",&a[i]);
for(i=;i<=n;i++)
b[i] = +(i-)*k;
for(i=;i<=n;i++)
{
cha[i] = a[i]-b[i];
cha2[i] = cha[i];
}
sort(cha2+,cha2+n+,cmp);
int cnt = ;
int maxi = ;
for(i=;i<=n;i++)
{
if(cha2[i] < )
break;
if(cha2[i] == cha2[i-])
cnt++;
else
{
if(cnt > maxi)
{
maxi = cnt;
flag = cha2[i-];
cnt = ;
}
}
}
if(cnt > maxi)
{
maxi = cnt;
flag = cha2[i-];
}
for(i=;i<=n;i++)
{
if(cha[i] == flag)
{
break;
}
}
for(j=i;j>=;j--)
biao[j] = a[i]-k*(i-j);
for(j=i+;j<=n;j++)
biao[j] = a[i]+k*(j-i);
cnt = ;
for(i=;i<=n;i++)
{
if(a[i]!=biao[i])
cnt++;
}
printf("%d\n",cnt);
for(i=;i<=n;i++)
{
if(a[i] != biao[i])
{
if(a[i] > biao[i])
{
printf("- %d %d\n",i,a[i]-biao[i]);
}
else
printf("+ %d %d\n",i,biao[i]-a[i]);
}
}
}
return ;
}