鉴于您的问题陈述:
实现一个动态规划算法来解决以下问题
问题。输入是一个 n × n 整数矩阵。对于输出:
计算水平,垂直,相邻条目的最大总和,
对角线和反对角线方向。返回总和作为输出
h + v + d + c(暗示性符号)这四个辅助结果。
仅由负数组成的数组的最大(或最大)和
整数为 0;也就是说,在这种情况下,我们选择空子数组。
具有解 45 的(小)示例输入矩阵:
|-2 5 3 2|
|9 -6 5 3|
|1 -8 2 -3|
|-1 2 -5 2|
然后,阅读您发布的示例输入,我做了一个简短的方法来解析文件中的信息:
public static List<List<Integer>> readFile(final String path) {
List<List<Integer>> result = new ArrayList<List<Integer>>();
Path p = Paths.get(path);
if (!Files.exists(p))
return null;
List<String> lines = null;
try {
lines = Files.readAllLines(p);
} catch (IOException e) {
e.printStackTrace();
}
if (lines == null)
return null;
for (String str : lines) {
List<Integer> row = new ArrayList<Integer>();
final String line = str.substring(1, str.length() - 1);
String[] arr = line.split(" ");
for (String s : arr)
row.add(Integer.valueOf(s.trim()));
result.add(row);
}
return result;
}
垂直方法是直截了当的,循环遍历嵌套数组。
public static int getVertical(final List<List<Integer>> list) {
int result = 0;
for (List<Integer> arr : list) {
int curr = 0;
for (Integer val : arr)
curr += val;
if (curr > result)
result = curr;
}
return result;
}
而且水平方向也相当直。注意:为了简单起见,这里我只是使用了一个计数器列表。还有更有效的方法。
public static int getHorizontal(final List<List<Integer>> list, final int len) {
List<Integer> sums = new ArrayList<Integer>(list.get(0));
for (int i = 1; i < len; ++i)
for (int j = 0; j < len; ++j)
sums.set(j, sums.get(j) + list.get(i).get(j));
return Collections.max(sums);
}
我快速搜索找到一个诊断/反诊断循环question。此代码有助于适应整数值(来自字符串/println's),并导致以下两种方法。注意:调试/system.out.println 用于帮助/显示。
public static int getDiagonal(final List<List<Integer>> list, final int len) {
int result = 0;
// [top half]
for (int i = len - 1; i > 0; --i) {
//String temp = "";
int tmp = 0;
for (int j = 0, x = i; x <= len - 1; ++j, ++x) {
final int val = list.get(x).get(j);
//temp = temp + " " + val;
tmp += val;
}
//System.out.println(temp);
if (tmp > result)
result = tmp;
}
// [lower half]
for (int i = 0; i <= len - 1; ++i) {
//String temp = "";
int tmp = 0;
for (int j = 0, y = i; y <= len - 1; ++j, ++y) {
final int val = list.get(j).get(y);
//temp = temp + " " + val;
tmp += val;
}
//System.out.println(temp);
if (tmp > result)
result = tmp;
}
return result;
}
public static int getAntiDiagonal(final List<List<Integer>> list, final int len) {
int result = 0;
// [top half]
for (int k = 0; k < len; ++k) {
int tmp = 0;
for (int j = 0; j <= k; ++j) {
int i = k - j;
int val = list.get(i).get(j);
//System.out.print(val + " ");
tmp += val;
}
if (tmp > result)
result = tmp;
//System.out.println();
}
// [lower half]
for (int k = len - 2; k >= 0; --k) {
int tmp = 0;
for (int j = 0; j <= k; ++j) {
int i = k - j;
int val = list.get(len - j - 1).get(len - i - 1);
//System.out.print(val + " ");
tmp += val;
}
if (tmp > result)
result = tmp;
//System.out.println();
}
return result;
}
这是解决原始问题的整个程序最终的样子:
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
public class App {
public static void main(String[] args) {
List<List<Integer>> data = readFile("C:\\Users\\Nick\\Desktop\\test.txt");
for (List<Integer> row : data)
System.out.println(row);
final int n = data.size();
int maxVertical = getVertical(data);
int maxHorizontal = getHorizontal(data, n);
int maxDiagonal = getDiagonal(data, n);
int maxAntiDiagonal = getAntiDiagonal(data, n);
System.out.println("max vertical = " + maxVertical);
System.out.println("max horizontal = " + maxHorizontal);
System.out.println("max diagonal = " + maxDiagonal);
System.out.println("max anti-diagonal = " + maxAntiDiagonal);
}
public static int getVertical(final List<List<Integer>> list) {
int result = 0;
for (List<Integer> arr : list) {
int curr = 0;
for (Integer val : arr)
curr += val;
if (curr > result)
result = curr;
}
return result;
}
public static int getHorizontal(final List<List<Integer>> list, final int len) {
List<Integer> sums = new ArrayList<Integer>(list.get(0));
for (int i = 1; i < len; ++i)
for (int j = 0; j < len; ++j)
sums.set(j, sums.get(j) + list.get(i).get(j));
return Collections.max(sums);
}
public static int getDiagonal(final List<List<Integer>> list, final int len) {
int result = 0;
// [top half]
for (int i = len - 1; i > 0; --i) {
int tmp = 0;
for (int j = 0, x = i; x <= len - 1; ++j, ++x) {
final int val = list.get(x).get(j);
tmp += val;
}
if (tmp > result)
result = tmp;
}
// [lower half]
for (int i = 0; i <= len - 1; ++i) {
int tmp = 0;
for (int j = 0, y = i; y <= len - 1; ++j, ++y) {
final int val = list.get(j).get(y);
tmp += val;
}
if (tmp > result)
result = tmp;
}
return result;
}
public static int getAntiDiagonal(final List<List<Integer>> list, final int len) {
int result = 0;
// [top half]
for (int k = 0; k < len; ++k) {
int tmp = 0;
for (int j = 0; j <= k; ++j) {
int i = k - j;
int val = list.get(i).get(j);
tmp += val;
}
if (tmp > result)
result = tmp;
}
// [lower half]
for (int k = len - 2; k >= 0; --k) {
int tmp = 0;
for (int j = 0; j <= k; ++j) {
int i = k - j;
int val = list.get(len - j - 1).get(len - i - 1);
tmp += val;
}
if (tmp > result)
result = tmp;
}
return result;
}
public static List<List<Integer>> readFile(final String path) {
List<List<Integer>> result = new ArrayList<List<Integer>>();
Path p = Paths.get(path);
if (!Files.exists(p))
return null;
List<String> lines = null;
try {
lines = Files.readAllLines(p);
} catch (IOException e) {
e.printStackTrace();
}
if (lines == null)
return null;
for (String str : lines) {
List<Integer> row = new ArrayList<Integer>();
final String line = str.substring(1, str.length() - 1);
String[] arr = line.split(" ");
for (String s : arr)
row.add(Integer.valueOf(s.trim()));
result.add(row);
}
return result;
}
}
最后,鉴于 test.txt 文件包含:
|-2 5 3 2|
|9 -6 5 3|
|1 -8 2 -3|
|-1 2 -5 2|
输出结果(应该是)是:
[-2, 5, 3, 2]
[9, -6, 5, 3]
[1, -8, 2, -3]
[-1, 2, -5, 2]
max vertical = 11
max horizontal = 7
max diagonal = 7
max anti-diagonal = 14
干杯
编辑我意识到我在技术上并没有完全回答原始问题,所以:
-
将这两行添加到 main() 方法的最后
int 结果 = maxVertical + maxHorizontal + maxDiagonal + maxAntiDiagonal;
System.out.println("结果 = " + 结果);
为每个相应的方法(vert/horiz/diag/anti-diag)添加检查,以便问题中所述的负面事件的情况
仅由负数组成的数组的最大(或最大)和
整数为 0;也就是说,在这种情况下,我们选择空子数组
也可以被覆盖。这不是一个巨大的代码大修论文,更是一个独特的检查。