给定一个含有n个元素的整型数组a,求a中所有元素的和。可能您会觉得很简单,是的,的确简单,但是为什么还要说呢,原因有二,第一,这道题要求用递归法,只用一行代码。第二,这是我人生中第一次面试时候遇到的题,意义特殊。
1. 如果数组元素个数为0,那么和为0。
2. 如果数组元素个数为n,那么先求出前n - 1个元素之和,再加上a[n - 1]即可
2)求数组的最大值和最小值
给定一个含有n个元素的整型数组a,找出其中的最大值和最小值
分析
常规的做法是遍历一次,分别求出最大值和最小值,但我这里要说的是分治法(Divide and couquer),将数组分成左右两部分,先求出左半部份的最大值和最小值,再求出右半部份的最大值和最小值,然后综合起来求总体的最大值及最小值。这是个递归过程,对于划分后的左右两部分,同样重复这个过程,直到划分区间内只剩一个元素或者两个元素。
代码
// 求数组的最大值和最小值,返回值在maxValue和minValue
void MaxandMin(int *a, int l, int r, int& maxValue, int& minValue)
{
if(l == r) // l与r之间只有一个元素
{
maxValue = a[l] ;
minValue = a[l] ;
return ;
}
if(l + 1 == r) // l与r之间只有两个元素
{
if(a[l] >= a[r])
{
maxValue = a[l] ;
minValue = a[r] ;
}
else
{
maxValue = a[r] ;
minValue = a[l] ;
}
return ;
}
int m = (l + r) / 2 ; // 求中点
int lmax ; // 左半部份最大值
int lmin ; // 左半部份最小值
MaxandMin(a, l, m, lmax, lmin) ; // 递归计算左半部份
int rmax ; // 右半部份最大值
int rmin ; // 右半部份最小值
MaxandMin(a, m + 1, r, rmax, rmin) ; // 递归计算右半部份
maxValue = max(lmax, rmax) ; // 总的最大值
minValue = min(lmin, rmin) ; // 总的最小值
}
3)
3)求数组的最大值和次大值
给定一个含有n个元素的整型数组,求其最大值和次大值
分析
思想和上一题类似,同样是用分治法,先求出左边的最大值leftmax和次大值leftsecond,再求出右边的最大值rightmax和次大值rightsecond,然后合并,如何合并呢?分情况考虑
1 如果leftmax > rightmax,那么可以肯定leftmax是最大值,但次大值不一定是rightmax,但肯定不是rightsecond,只需将leftsecond与rightmax做一次比较即可。
2 如果rightmax > leftmax,那么可以肯定rightmax是最大值,但次大值不一定是leftmax,但肯定不是leftsecond,所以只需将leftmax与rightsecond做一次比较即可。
注意
这种方法无法处理最大元素有多个的情况,比如3,5,7,7将返回7,7而不是7,5。感谢网友 从无到有靠谁人 指出。
代码
// 找出数组的最大值和次大值,a是待查找的数组,left和right是查找区间,max和second存放结果
void MaxandMin(int a[], int left, int right, int&max, int&second)
{
if(left == right)
{
max = a[left] ;
second = INT_MIN;
}
elseif(left +1== right)
{
max = a[left] > a[right] ? a[left] : a[right] ;
second = a[left] < a[right] ? a[left] : a[right] ;
}
else
{
int mid = left + (right - left) /2 ;
int leftmax ;
int leftsecond ;
MaxandMin(a, left, mid, leftmax, leftsecond) ;
int rightmax ;
int rightsecond ;
MaxandMin(a, mid +1, right, rightmax, rightsecond) ;
if (leftmax > rightmax)
{
max = leftmax ;
second = leftsecond > rightmax ? leftsecond : rightmax ;
}
else
{
max = rightmax ;
second = leftmax < rightsecond ? rightsecond : leftmax ;
}
}
}
采用最笨的办法,遍历,设置两个参数也是可以实现的,但是分治算法的使用,使得算法变得有技巧性。