我正在从一本书中学习算法' Algorthms简介'。我想实现一个分而治之的算法来找到一个数组的最大子数组。这是我的解决方案,但我得到了错误的结果。
任何帮助将不胜感激。请解释一下,因为我更感兴趣的是理解它而不是让它发挥作用。谢谢。
def maxNumber(int a, int b){
return a > b ? a : b;
}
def maxNumber(List a, List b, List c){
return maxNumber(a.sum(), maxNumber(b.sum(), c.sum()))
}
//int sum(List list){
// int sum = 0
// list.each {
// sum+= it
// }
// sum
//}
def maxCrossing(ArrayList<Integer> list, int low, int mid, int high){
int sum = 0
int leftSum = Integer.MIN_VALUE
int maxLeftIndex = -1
/*for (int i = low; i <= mid ; i++) {
sum += list[i]
if (sum > leftSum) {
leftSum = sum
maxLeftIndex = i
}
}*/
for (int i = mid; i >= low ; i--) {
sum += list[i]
if (sum > leftSum) {
leftSum = sum
maxLeftIndex = i
}
}
sum = 0
int rightSum = Integer.MIN_VALUE
int maxRightIndex = -1
for (int i = mid + 1; i <= high ; i++) {
sum += list[i]
if (sum > rightSum) {
rightSum = sum
maxRightIndex = i
}
}
def returnList = []
for (int i = maxLeftIndex; i < maxRightIndex + 1; i++) {
returnList.add(list[i])
}
return returnList
}
def maxSubArray(ArrayList<Integer> list,int low, int high){
if (low == high) return [list[low]]
int mid = (low + high) / 2
def leftResults = maxSubArray(list, low, mid)
def rightResults = maxSubArray(list, mid + 1, high)
def crossResults = maxCrossing(list, low, mid, high)
/*if (rightResults[2] > leftResults[2] && rightResults[2] > crossResults[2]) return rightResults
if (leftResults[2] > rightResults[2] && leftResults[2] > crossResults[2]) return leftResults
else return crossResults*/
maxNumber(leftResults, rightResults, crossResults)
}
//Testing Features
println("Enter array members")
ArrayList<Integer> myList = [-2, -5, 10, -2, -3, 1, 5, -6] //System.in.newReader().readLines()
int size = myList.size()
def maxSum = maxSubArray(myList, 0, size - 1)
println("Maximum sub-array is: " + maxSum)
答案 0 :(得分:1)
认为您的一个主要问题是if
maxCrossing
语句缺少括号
所以:
if (sum > leftSum) leftSum = sum
maxLeftIndex = i
应该是:
if (sum > leftSum) {
leftSum = sum
maxLeftIndex = i
}
另外,我相信在检查交叉路口的下半部分时,你需要从中间点开始工作(这可能是错误的)
此外,传回索引和总和没有多大意义......我将其更改为只返回每个步骤的最大数组(然后我们可以调用sum()
)
以下是您的代码中的(我认为)工作解决方案:
def maxResults(List a, List b, List c) {
a.sum() > b.sum() && a.sum() > c.sum() ? a :
b.sum() > c.sum() ? b :
c
}
def maxCrossing(List list, int low, int mid, int high){
int sum = 0
int leftSum = Integer.MIN_VALUE
int maxLeftIndex = -1
for (int i = mid; i >= low; i--) {
sum += list[i]
if (sum > leftSum) {
leftSum = sum
maxLeftIndex = i
}
}
sum = 0
int rightSum = Integer.MIN_VALUE
int maxRightIndex = -1
for (int i = mid + 1; i <= high ; i++) {
sum += list[i]
if (sum > rightSum) {
rightSum = sum
maxRightIndex = i
}
}
return list[maxLeftIndex..maxRightIndex]
}
def maxSubArray(List list, int low, int high){
if (low == high) return [list[low]]
int mid = (low + high) / 2
def leftResults = maxSubArray(list, low, mid)
def rightResults = maxSubArray(list, mid + 1, high)
def crossResults = maxCrossing(list, low, mid, high)
maxResults(rightResults, leftResults, crossResults)
}
//Testing Features
println("Enter array members")
ArrayList<Integer> myList = [-2, -5, 10, -2, -3, 1, 5, -6] //System.in.newReader().readLines()
int size = myList.size()
def maxSum = maxSubArray(myList, 0, size - 1)
println("Maximum sub-array is: " + maxSum)
顺便说一下,这个输出是:
Maximum sub-array is: [10, -2, -3, 1, 5]