Bubble Sort
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) Total Submission(s): 224 Accepted Submission(s): 147
for(int i=1;i<=N;++i)
for(int j=N,t;j>i;—j)
if(P[j-1] > P[j])
t=P[j],P[j]=P[j-1],P[j-1]=t;
After the sort, the array is in increasing order. ?? wants to know the absolute values of difference of rightmost place and leftmost place for every number it reached.
limits T <= 20 1 <= N <= 100000 N is larger than 10000 in only one case.
3
3 1 2
3
1 2 3
Case #2: 0 0 0
In first case, (3, 1, 2) -> (3, 1, 2) -> (1, 3, 2) -> (1, 2, 3)
the leftmost place and rightmost place of 1 is 1 and 2, 2 is 2 and 3, 3 is 1 and 3
In second case, the array has already in increasing order. So the answer of every number is 0.
题意:比赛的时候两度理解错题目意思。。后来才发现题目的意思是给你一个1~n的排列,问冒泡排序过程中,数字i(1<=i<=n)所到达的最左位置与最右位置的差值的绝对值是多少
参考了一下别人的,整理了一下思路,如下
判断一个数字到达的最左位置,可以打个草稿,会发现就min(a[i],i);
i是当前所在位置,a[i]为排序后的最终位置;
到达的最右位置就是当前位置i+他右边比他小的数的个数,N为100000,可以用树状数组来做
从右到左找,每个数a[i],我们只需判断它右边1~a[i]-1中有几个比他小数
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<iostream>
using namespace std;
const int N = ;
int a[N],l[N],r[N],c[N];
int lowbit(int t)
{
return t&(-t);
}
void update(int i,int x)
{
while(i<N)
{
c[i]=c[i]+x;
i+=lowbit(i);
}
}
int Sum(int n) //求1-n-1项的和.
{
int sum=;
while(n>)
{
sum+=c[n];
n-=lowbit(n);
}
return sum;
}
int main()
{
int t,n,p=,i;
scanf("%d",&t);
while(t--)
{
memset(c,,sizeof(c));
scanf("%d",&n);
for(i=;i<=n;i++)
{
scanf("%d",&a[i]);
l[a[i]]=min(i,a[i]);
}
for(i=n;i>;i--)
{
r[a[i]]=i+Sum(a[i]-);
update(a[i],);
}
printf("Case #%d:",p++);
for(i=;i<=n;i++)
printf(" %d",abs(l[i]-r[i]));
puts("");
}
return ;
}