`
蒙面考拉
  • 浏览: 161210 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

数对之差的最大值(8)

 
阅读更多

题目:在数组中,数字减去它右边的数字得到一个数对之差。求所有数对之差的最大值。例如在数组{2, 4, 1, 16, 7, 5, 11, 9}中,数对之差的最大值是11,是16减去5的结果。

分析:看到这个题目,很多人的第一反应是找到这个数组的最大值和最小值,然后觉得最大值减去最小值就是最终的结果。这种思路忽略了题目中很重要的一点:数对之差是一个数字减去它右边的数字。由于我们无法保证最大值一定位于数组的左边,因此这个思路不管用。

于是我们接下来可以想到让每一个数字逐个减去它右边的所有数字,并通过比较得到数对之差的最大值。由于每个数字需要和它后面的O(n)个数字作减法,因此总的时间复杂度是O(n2)

解法一:分治法

通常蛮力法不会是最好的解法,我们想办法减少减法的次数。假设我们把数组分成两个子数组,我们其实没有必要拿左边的子数组中较小的数字去和右边的子数组中较大的数字作减法。我们可以想象,数对之差的最大值只有可能是下面三种情况之一:(1)被减数和减数都在第一个子数组中,即第一个子数组中的数对之差的最大值;(2)被减数和减数都在第二个子数组中,即第二个子数组中数对之差的最大值;(3)被减数在第一个子数组中,是第一个子数组的最大值。减数在第二个子数组中,是第二个子数组的最小值。这三个差值的最大者就是整个数组中数对之差的最大值。

在前面提到的三种情况中,得到第一个子数组的最大值和第二子数组的最小值不是一件难事,但如何得到两个子数组中的数对之差的最大值?这其实是原始问题的子问题,我们可以递归地解决。下面是这种思路的参考代码:

int MaxDiff_Solution1(int numbers[], unsigned length)

{

    if(numbers == NULL || length < 2)

        return 0;

 

    int max, min;

    return MaxDiffCore(numbers, numbers + length - 1, &max, &min);

}

 

int MaxDiffCore(int* start, int* end, int* max, int* min)

{

    if(end == start)

    {

        *max = *min = *start;

        return 0x80000000;

    }

 

    int* middle = start + (end - start) / 2;

 

    int maxLeft, minLeft;

    int leftDiff = MaxDiffCore(start, middle, &maxLeft, &minLeft);

 

    int maxRight, minRight;

    int rightDiff = MaxDiffCore(middle + 1, end, &maxRight, &minRight);

 

    int crossDiff = maxLeft - minRight;

 

    *max = (maxLeft > maxRight) ? maxLeft : maxRight;

    *min = (minLeft < minRight) ? minLeft : minRight;

 

    int maxDiff = (leftDiff > rightDiff) ? leftDiff : rightDiff;

    maxDiff = (maxDiff > crossDiff) ? maxDiff : crossDiff;

    return maxDiff;

}

在函数MaxDiffCore中,我们先得到第一个子数组中的最大的数对之差leftDiff,再得到第二个子数组中的最大数对之差rightDiff。接下来用第一个子数组的最大值减去第二个子数组的最小值得到crossDiff。这三者的最大值就是整个数组的最大数对之差。

解法二:转化成求解子数组的最大和问题

接下来再介绍一种比较巧妙的解法。如果输入一个长度为n的数组numbers,我们先构建一个长度为n-1的辅助数组diff,并且diff[i]等于numbers[i]-numbers[i+1]0<=i<n-1)。如果我们从数组diff中的第i个数字一直累加到第j个数字(j > i),也就是diff[i] + diff[i+1] + … + diff[j] = (numbers[i]-numbers[i+1]) + (numbers[i + 1]-numbers[i+2]) + ... + (numbers[j] – numbers[j + 1]) = numbers[i] – numbers[j + 1]

分析到这里,我们发现原始数组中最大的数对之差(即numbers[i] – numbers[j + 1])其实是辅助数组diff中最大的连续子数组之和。我们在本系列的博客的第3《求子数组的最大和》中已经详细讨论过这个问题的解决方法。基于这个思路,我们可以写出如下代码:

int MaxDiff_Solution2(int numbers[], unsigned length)

{

    if(numbers == NULL || length < 2)

        return 0;

 

    int* diff = new int[length - 1];

    for(int i = 1; i < length; ++i)

        diff[i - 1] = numbers[i - 1] - numbers[i];

 

    int currentSum = 0;

    int greatestSum = 0x80000000;

    for(int i = 0; i < length - 1; ++i)

    {

        if(currentSum <= 0)

            currentSum = diff[i];

        else

            currentSum += diff[i];

 

        if(currentSum > greatestSum)

            greatestSum = currentSum;

    }

 

    delete[] diff;

 

    return greatestSum;

} 

解法三:动态规划法

既然我们可以把求最大的数对之差转换成求子数组的最大和,而子数组的最大和可以通过动态规划求解,那我们是不是可以通过动态规划直接求解呢?下面我们试着用动态规划法直接求数对之差的最大值。

我们定义diff[i]是以数组中第i个数字为减数的所有数对之差的最大值。也就是说对于任意hh < i),diff[i]number[h]-number[i]diff[i]0≤i<n)的最大值就是整个数组最大的数对之差。

假设我们已经求得了diff[i],我们该怎么求得diff[i+1]呢?对于diff[i],肯定存在一个hh < i),满足number[h]减去number[i]之差是最大的,也就是number[h]应该是number[i]之前的所有数字的最大值。当我们求diff[i+1]的时候,我们需要找到第i+1个数字之前的最大值。第i+1个数字之前的最大值有两种可能:这个最大值可能是第i个数字之前的最大值,也有可能这个最大值就是第i个数字。第i+1个数字之前的最大值肯定是这两者的较大者。我们只要拿第i+1个数字之前的最大值减去number[i+1],就得到了diff[i+1]

int MaxDiff_Solution3(int numbers[], unsigned length)

{

    if(numbers == NULL || length < 2)

        return 0;

 

    int max = numbers[0];

    int maxDiff =  max - numbers[1];

 

    for(int i = 2; i < length; ++i)

    {

        if(numbers[i - 1] > max)

            max = numbers[i - 1];

 

        int currentDiff = max - numbers[i];

        if(currentDiff > maxDiff)

            maxDiff = currentDiff;

    }

 

    return maxDiff;

}

在上述代码中,max表示第i个数字之前的最大值,而currentDiff表示diff[i] 0≤i<n),diff[i]的最大值就是代码中maxDiff

解法小结

上述三种代码,虽然思路各不相同,但时间复杂度都是O(n)(第一种解法的时间复杂度可以用递归公式表示为T(n)=2(n/2)+O(1),所以总体时间复杂度是O(n))。我们也可以注意到第一种方法是基于递归实现,而递归调用是有额外的时间、空间消耗的(比如在调用栈上分配空间保存参数、临时变量等)。第二种方法需要一个长度为n-1的辅助数组,因此其空间复杂度是O(n)。第三种方法则没有额外的时间、空间开销,并且它的代码是最简洁的,因此这是最值得推荐的一种解法。

分享到:
评论

相关推荐

    数对之差的最大值

    在这个问题中,我们面临的是一个关于数组操作的挑战,具体来说是寻找数组中数对之差的最大值。这个问题可以归类为数组处理中的简单算法问题,通常可以通过一次遍历或排序来解决。 首先,我们要理解“数对之差”的...

    1112 最大值和最小值的差.cpp

    1112:最大值和最小值的差 时间限制: 1000 ms 内存限制: 65536 KB 提交数: 24329 通过数: 14187 【题目描述】 ...输出M个数中最大值和最小值的差。 【输入样例】 5 2 5 7 4 2 【输出样例】 5 【来源】 NO

    1.给出一个整数数组,求其中任意两个元素之差的最大值。

    ### 第一部分:求整数数组中任意两个元素之差的最大值 #### 题目背景与需求 题目要求给定一个整数数组,计算并返回数组中任意两个元素之差的最大值。 #### 解决方案 为了找到数组中任意两个元素之差的最大值,可以...

    数组中数对差最大

    在这个问题中,我们关注的是“数组中数对差最大”的计算,这是一种在数组中寻找特定数值差异的算法问题。 该问题的描述指出,我们需要找到数组中所有数对之间的最大差值。数对由数组中的两个不同元素组成,其中一个...

    比较数的最大值和最小值

    根据给定的文件信息,我们...该程序提供了一种有效的方法来比较数组中的最大值和最小值,通过预先调整数组中的某些元素位置,减少了查找过程中的比较次数,从而提高了效率。这种算法适用于对效率有一定要求的应用场景。

    C语言程序设计-求一批数中最大值和最小值的差.c

    C语言程序设计-求一批数中最大值和最小值的差

    C#写的2个数字的比较,输出其中的最大值的算法

    这段代码首先导入`System`命名空间,然后定义了两个整数变量`num1`和`num2`,接着使用`Math.Max`函数比较这两个数并存储结果到`maxNum`变量,最后通过`Console.WriteLine`输出最大值。 另一种方法是使用条件运算符...

    数的距离差

    1. **距离差**:给定一组正整数,其中的最大值记作`max`,最小值记作`min`,对于这组数中的任意一个数`x`,其到`max`和`min`的距离差`D`定义为`abs(abs(x - max) - abs(x - min))`,其中`abs()`表示求一个数的绝对值...

    求最大值和最小值

    从键盘任意输入10个整数,用指针变量作函数参数编程计算最大值和最小值,并返回它们所在数组中的位置。

    三位数重排求差“数学黑洞”

    这里的“重排求差”操作指的是将一个三位数的三个数字重新排列,形成最大的数和最小的数,然后计算它们的差值。例如,对于数字107,重排求差序列是710-17=693,963-369=594,954-459=495。 为了编程验证这个规律,...

    Python中如何查看Pandas DataFrame对象列的最大值、最小值、平均值、标准差、中位数等

    如何查看Pandas DataFrame对象列的最大值、最小值、平均值、标准差、中位数等 我们举个例子说明一下,先创建一个dataframe对象df,内容如下: 1.使用sum函数获得函数列的和,用法:df.sum() 2.使用max获取最大值,用法...

    用直方图算平均数中位数众数标准差PPT课件.pptx

    然而,由于其对极端值敏感,平均数可能会因为异常值的存在而产生误导。在经济学中,平均数常用于计算平均工资或平均消费水平。 标准差是一个衡量数据离散程度的统计量,它通过计算每个数据点与平均数之间的差值的...

    西西弗斯黑洞【123数字黑洞】 卡普雷卡尔黑洞(重排求差黑洞):三位数黑洞495

    西西弗斯黑洞【123数字黑洞】 /// ​设定一个任意数字串,数出这个数中的偶数个数,奇数个数,及这个数中所包含的所有位数的总数 ... /// 对三位数继续计算最大值 和 最小值,最终差值(终结黑洞值)为495

    给定n个不同的整数,问这些数中有多少对整数,它们的值正好相差1

     给定n个不同的整数,问这些数中有多少对整数,它们的值正好相差1。 输入格式  输入的第一行包含一个整数n,表示给定整数的个数。  第二行包含所给定的n个整数。 输出格式  输出一个整数,表示值正好相差1的数对...

    平均数众数中位数方差极差标准差典型题.doc

    极差是描述数据离散程度的统计量,它是数据中最大值和最小值的差。极差的计算公式为: $$R=x_{max}-x_{min}$$ 其中,$R$为极差,$x_{max}$为最大值,$x_{min}$为最小值。 例如,一组数据:4、—1、5、9、7、6、7...

    求值_python_方差_中位数_平均值_

    **方差**衡量的是数据集中的数值与平均值的偏离程度,它计算每个数值与平均值的差的平方的平均数。方差越大,数据点越分散;方差越小,数据点越集中。在Python中,可以这样计算: ```python # 计算方差 variance = ...

    数列极差问题

    总结来说,数列极差问题是一个涉及贪心策略的算法问题,通过对数列进行特定操作并逐步减少元素,可以计算出最大值和最小值的极差。通过设计和实现对应的算法,我们可以有效地解决这类问题,并进行充分的测试以验证其...

    求极差即一组数据中最大值与最小值的差PPT学习教案.pptx

    1. 极差计算:极差是一组数据中的最大值与最小值之差,用于衡量数据的波动范围。例如,在电子元件电阻值的例子中,我们需要找到电阻值的最大值和最小值,然后计算它们之间的差值,以了解这批元件电阻值的分散程度。 ...

    平均数、中位数、众数、极差、方差标准差课件.pptx

    极差是数据中的最大值减去最小值,它体现了数据的波动范围,但并不考虑数据的分布情况。极差简单易计算,但受极端值影响较大。 方差是各数据值与其平均数之差的平方的平均数,表示数据离散程度的平方。方差计算公式...

    平均数、中位数、众数、极差、方差标准差的概念讲解.ppt

    极差( Range)是指一组数据中的最大值和最小值之间的差值。极差反映了一组数据的分散程度。 方差和标准差 方差( Variance)是指一组数据中的每个值与平均值的差值的平方的平均值。标准差( Standard Deviation)...

Global site tag (gtag.js) - Google Analytics