这题用贪心来做,关键是找到贪心的思维
一开始做的时候贪心的思维不对,敲了好几份然后都错了
看了题解才知道是要从最重的和最轻的入手
int cmp(int*a,int*b)
{
return *a-*b;
}
int numRescueBoats(int* people, int peopleSize, int limit){
int ans=0;
qsort(people,peopleSize,sizeof(int),cmp);
int left=0;
int right=peopleSize-1;
while(left<=right)
{
if(people[left]+people[right]>limit)
{
right--;
}
else
{
left++;
right--;
}
ans++;
}
return ans;
}
实际上还有一个疑问
为什么最重的要和最轻的配对,直观上来看,最重的应该和能够和最重的人一起装进船里面的人中最重的配对
比如limit=15 最重的为10,则有5,1两个轻的人,直观上来看,应该要放5,这样1就在外面,留给外面船的压力就小一点
实际上两个是一样的
考虑一个固定的序列p0,p1,p2,……pn-1,与最轻的配对的方法称为方法1,上述方法称为方法2,
则当light,heavy指针所指的数满足people[light]+people[heavy]>limit时,两个方法都把heavy剔除,二者面对的子问题一致
当people[light]+people[heavy]<=limit时,
设序列中倒数有k个数,即p0,p1,p2……bk,bk-1.……b1满足p0+bi<=limit bk=max(bi)
即方法1中p0和b1配对
方法2中p0和bk配对,这两种情况是对应的
当方法2中p0+bk<=limit时,必有p1+bk-1<=limit,p2+bk-2<=limit,……pi+bk-i<=limit,在完成这些配对后,方法2的船的数量+k,面对的子情况是,pk,pk+1……bk+1
因为p0+bk<=limit
则在方法1中,p0+b1<=limit,p1+b2<=limit pi+bi+1<=limit,因为bi<=bk,pi<=p1
所以在完成这些配对后,方法1的船的数量+k,面对的子情况是pk,pk+1……bk+1
与方法1面对的子情况相同
所以方法2和方法1,都会产生相同的结果,方法2只是更加强烈一点