给出一个数组A,找出一对 (i, j)使得A[i] <= A[j] (i < j)并且j-i最大

简介:

题目:给出一个数组A,找出一对 (i, j)使得A[i] <= A[j] (i <= j)并且j-i最大 ,若有多个这样的位置对,返回i最小的那一对。

最直接的想法就是对于每一个 i 从数组最尾端开始向前找到第一个大于等于 A[i] 的位置 j ,时间复杂度O(n^2)。

复制代码
1.    pair<int, int> find(const vector<int> &A)  
2.    {  
3.        int n = A.size();  
4.        if(n == 0)  
5.            throw new invalid_argument("Array's size can't be 0!");  
6.      
7.        int target_i = 0, target_j = 0;  
8.        int max_len = 0;  
9.        for(int i = 0; i < n; ++i)  
10.        {  
11.            int j;  
12.            for(j = n-1; j >= i; --j)  
13.                if(A[j] >= A[i])  
14.                    break;  
15.            if(j-i+1 > max_len)  
16.            {  
17.                target_i = i;  
18.                target_j = j;  
19.                max_len = j-i+1;  
20.            }  
21.        }  
22.      
23.        return make_pair<int, int>(target_i, target_j);  
24.    }  
复制代码

我们对上述算法稍作优化。当i=0时,我们假设找到的大于A[i]的最右位置是j0,那么对于i=1时,我们根本就不需要考虑小于j0的位置,因为它们的区间长度都小于j0+1,不可能成为最优解。

复制代码
1.    pair<int, int> find(const vector<int> &A)  
2.    {  
3.        int n = A.size();  
4.        if(n == 0)  
5.            throw new invalid_argument("Array's size can't be 0!");  
6.      
7.        int target_i = 0, target_j = 0;  
8.        int max_len = 0;  
9.        for(int i = 0; i < n; ++i)  
10.        {  
11.            int j;  
12.            for(j = n-1; j > target_j; --j) // 此处只需检查到target_j  
13.                if(A[j] >= A[i])  
14.                    break;  
15.            if(j-i+1 > max_len)  
16.            {  
17.                target_i = i;  
18.                target_j = j;  
19.                max_len = j-i+1;  
20.            }  
21.        }  
22.      
23.        return make_pair<int, int>(target_i, target_j);  
24.    }  
复制代码

但时间复杂度仍然是O(n^2)的。我们可以继续接着上面的思路优化。其实对于位置 i 求最后一个大于等于它的位置,不需要每次都从数组尾部向前找,我们可以通过改进这个地方将时间复杂度变为O(n)。

过程是这样的,对于 i ,我们先找到 i 及其右端的最大元素的位置 j ,检查是否比当前记录的最优解更优,更新。然后考虑 j+1及其右端的最大元素位置是否大于等于A[i],若是,令 j 等于该位置,重复如上过程,若否,那么从位置i+1重新开始,但j仍然从当前位置考虑即可,原因上面已说明。这样时间复杂度就成O(n)的了。

具体请参考代码

 


本文转自农夫山泉别墅博客园博客,原文链接:http://www.cnblogs.com/yaowen/p/4529124.html,如需转载请自行联系原作者

相关文章
|
12月前
【Leetcode -696.计数二进制字串 -697.数组的度】
【Leetcode -696.计数二进制字串 -697.数组的度】
29 0
|
5月前
|
存储
leetcode2744. 最大字符串配对数目
leetcode2744. 最大字符串配对数目
30 0
|
5月前
【每日一题Day317】LC2605从两个数字数组里生成最小数字 | 哈希表
【每日一题Day317】LC2605从两个数字数组里生成最小数字 | 哈希表
38 0
|
12月前
|
Serverless
华为机试HJ62:查找输入整数二进制中1的个数
华为机试HJ62:查找输入整数二进制中1的个数
|
12月前
华为机试HJ58:输入n个整数,输出其中最小的k个
华为机试HJ58:输入n个整数,输出其中最小的k个
|
12月前
华为机试HJ9:提取不重复的整数
华为机试HJ9:提取不重复的整数
|
12月前
|
C++
华为机试HJ2:计算某字母出现次数
华为机试HJ2:计算某字母出现次数
每日一题---输出100个1~6的随机整数,并求出每个数出现的概率
每日一题---输出100个1~6的随机整数,并求出每个数出现的概率
每日一题---输出100个1~6的随机整数,并求出每个数出现的概率
leetcode 1356 根据数字二进制下1的数目排序
leetcode 1356 根据数字二进制下1的数目排序
66 0
leetcode 1356 根据数字二进制下1的数目排序
C语言经典实例:11-20例:使用结构体输出学生成绩、编制万年历、验证哥德巴赫猜想、求二维数组最大最小值、数组求素数、数组元素排序、进制数的转换进制数的转换、找出次大值、重组数组(下)
C语言经典实例:11-20例:使用结构体输出学生成绩、编制万年历、验证哥德巴赫猜想、求二维数组最大最小值、数组求素数、数组元素排序、进制数的转换进制数的转换、找出次大值、重组数组(下)
C语言经典实例:11-20例:使用结构体输出学生成绩、编制万年历、验证哥德巴赫猜想、求二维数组最大最小值、数组求素数、数组元素排序、进制数的转换进制数的转换、找出次大值、重组数组(下)