- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我编写了一个算法来查找两点之间所有可能的路径。我用 c#.net 和 java 编写了该算法。该算法产生大约 40,000 条可能的路径。令我惊讶的是,C#.net 中的算法只花了 13 分钟就完成了(在 i3 处理器中),而 java 甚至在 4 小时内都无法完成。
递归在java中可能很慢,但我不这么认为。
我有什么遗漏的吗?
好的,这里是需求,迷宫是一个二维数组,包含数字0、1、2和3。0代表可行走点,1代表障碍物,2代表起点,3代表导出点。该程序只需一次触摸全 0 即可找到从点 2 到点 3 的所有路径。我们不能跨过 1,因为它们是障碍。
public class Maze
{
//Represents each point in the Maze.
private class Node
{
private int val;
public Node(int num)
{
val = num;
switch (num)
{
case 1: IsPit = true;
break;
case 2: IsEntranceNode = true;
break;
case 3:
IsExitNode = true;
break;
}
}
public string Name { get; set; }
public bool IsPit { get; private set; }
public bool IsVisited { get; set; }
public bool IsExitNode { get; private set; }
public bool IsEntranceNode { get; private set; }
public Node UpNode { get; set; }
public Node DownNode { get; set; }
public Node LeftNode { get; set; }
public Node RightNode { get; set; }
}
#endregion
//Number of pits in the maze.
private int numberOfPits;
//Stack traces the path traversed.
private Stack<String> pathTraversed;
//Represents the maze.All Nodes are wired to its adjacent nodes.
private Node[,] Maze { get; set; }
//Number of paths found.
private int pathCount = 1;
//displays a set.
static void printSet(int[,] set)
{
Console.Write(" \t");
for (var i = 0; i < set.GetLength(1); i++)
Console.Write(i + " ");
Console.WriteLine("\n--------------------------");
for (var i = 0; i < set.GetLength(0); i++)
{
Console.Write((char)(65 + i) + " | \t");
for (var j = 0; j < set.GetLength(1); j++)
Console.Write(set[i, j] + " ");
Console.WriteLine();
}
}
static void Main(string[] args)
{
int[,] set2 = new int[7, 7] { { 2, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 3, 1, 1 }
};
Console.WriteLine("\n\nSet 2");
Console.WriteLine("=====");
printSet(set2);
Console.WriteLine("\nPath found");
Console.WriteLine("==========");
startTime = DateTime.Now;
new Maze().AddPoints(set2).FindPaths();
endTime = DateTime.Now;
difference = endTime.Subtract(startTime).TotalMinutes;
Console.WriteLine("Time taken in mins : " + difference);
Console.WriteLine("\nPress any key to exit");
Console.ReadKey();
}
//Adds points to the maze.Transforms a matrix to 2 dimensional connected nodes.
public Program AddPoints(int[,] matrix)
{
Maze = new Node[matrix.GetLength(0), matrix.GetLength(1)];
for (int i = 0; i < matrix.GetLength(0); i++)
{
for (var j = 0; j < matrix.GetLength(1); j++)
{
Node node = Maze[i, j];
if (node == null)
{
node = new Node(matrix[i, j]);
Maze[i, j] = node;
}
if (!(i - 1 < 0))
Maze[i - 1, j].DownNode = node;
if (!(i + 1 > matrix.GetLength(0) - 1))
{
var downNode = Maze[i + 1, j];
if (downNode == null)
Maze[i + 1, j] = new Node(matrix[i + 1, j]);
Maze[i + 1, j].UpNode = node;
}
if (!(j - 1 < 0))
Maze[i, j - 1].RightNode = node;
if (!(j + 1 > matrix.GetLength(1) - 1))
{
var leftNode = Maze[i, j + 1];
if (leftNode == null)
Maze[i, j + 1] = new Node(matrix[i, j + 1]);
Maze[i, j + 1].LeftNode = node;
}
node.Name = ((char)(65 + i)).ToString() + j.ToString();
if (node.IsPit)
numberOfPits++;
}
}
return this;
}
//Finds path between the start node and end node.
public void FindPaths()
{
var startNode = Maze[0, 0];
if (!startNode.IsEntranceNode)
throw new Exception("Node is not starting node.");
pathTraversed = new Stack<string>();
pathTraversed.Push(startNode.Name);
Traverse(startNode.RightNode);
Traverse(startNode.DownNode);
Traverse(startNode.LeftNode);
Traverse(startNode.UpNode);
}
//Traverses a node.
private void Traverse(Node node)
{
if (node == null)
return;
if (node.IsEntranceNode || node.IsPit || node.IsVisited)
return;
if (node.IsExitNode)
{
pathTraversed.Push(node.Name);
if (pathTraversed.Count == Maze.Length - numberOfPits)
{
var msg = "Path " + pathCount++ + " : " + string.Join("->", pathTraversed.Reverse());
Console.WriteLine(msg);
}
pathTraversed.Pop();
return;
}
pathTraversed.Push(node.Name);
node.IsVisited = true;
Traverse(node.RightNode); //
Traverse(node.DownNode); // Move to Next Node
Traverse(node.LeftNode); //
Traverse(node.UpNode); //
if (node.Name != pathTraversed.Peek())
throw new Exception("Error in Logic.");
node.IsVisited = false;
pathTraversed.Pop();
}
}
public class PathFinder
{
public static void main(String[] args){
int[][] set1 = new int[][] { { 2, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 0, 0, 0 },
{ 0, 0, 0, 0, 3, 1, 1 }};
ObservableStack<String> observableStack = new ObservableStack<String>();
StackObserver stackObserver = new StackObserver(plotter);
Maze maze = new Maze(set1,(IPathFinderStack<String>)observableStack);
maze.findPaths();
}
}
public class Maze {
private Node[][] nodes;
IPathFinderStack<String> pathTraversed;
private int numberOfPits = 0;
private int result = 0;
public Node[][] getVector(){
return nodes;
}
public Maze(int[][] matrix ,IPathFinderStack<String> stack){
pathTraversed = stack;
addNodes(matrix);
}
public void addNodes(int[][] matrix) {
int rows = matrix.length;
int cols = matrix[0].length;
nodes = new Node[rows][cols];
for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
Node node = nodes[i][j];
if (node == null){
node = new Node(matrix[i][j]);
nodes[i][j] = node;
}
if (!(i - 1 < 0))
nodes[i-1][j].setDownNode(node);
if (!((i + 1) > rows - 1))
{
Node downNode = nodes[i+1][j];
if (downNode == null)
nodes[i + 1][j] = new Node(matrix[i+1][j]);
nodes[i + 1][j].setUpNode(node);
}
if (!(j - 1 < 0))
nodes[i][j-1].setRightNode(node);
if (!(j + 1 > cols - 1)){
Node leftNode = nodes[i][j+1];
if (leftNode == null)
nodes[i][j+1] = new Node(matrix[i][j+1]);
nodes[i][j+1].setLeftNode(node);
}
String name = new String(new char[] {(char) (65 + i), (char)(48 + j)});
node.setName(name);
if (node.isPit())
this.numberOfPits ++;
}
}
}
public void findPaths(){
Node startNode = nodes[0][0];
pathTraversed.push(startNode.getName());
Traverse(startNode.getRightNode());
Traverse(startNode.getDownNode());
Traverse(startNode.getLeftNode());
Traverse(startNode.getUpNode());
}
private void Traverse(Node node){
if (node == null)
return ;
if (node.isEntrance() || node.isVisited() || node.isPit())
return ;
if (node.isExit()){
pathTraversed.push(node.getName());
if (pathTraversed.getLength() == (nodes[0].length * nodes.length) - numberOfPits){
String msg = "Path found" + (++result);
Logger.Log(msg);
Logger.Log(pathTraversed.toString());
}
pathTraversed.pop();
return;
}
pathTraversed.push(node.getName());
node.setVisited(true);
Traverse(node.getRightNode());
Traverse(node.getDownNode());
Traverse(node.getLeftNode());
Traverse(node.getUpNode());
node.setVisited(false);
pathTraversed.pop();
}
}
public class Node
{
private String name;
private int val;
private boolean isPit;
private boolean isVisited;
private boolean isEntrance;
private boolean isExit;
private Node downNode;
private Node upNode;
private Node leftNode;
private Node rightNode;
public Node(int val){
this.val = val;
this.isPit = (val == 1);
this.isEntrance = (val == 2);
this.isExit = (val == 3);
this.isVisited = false;
}
public String getName(){
return this.name;
}
public void setName (String name) {
this.name = name;
}
public boolean isPit(){
return isPit;
}
public boolean isVisited(){
return isVisited;
}
public void setVisited(boolean isVisited) {
this.isVisited = isVisited;
}
public boolean isEntrance(){
return isEntrance;
}
public boolean isExit(){
return isExit;
}
public Node getDownNode(){
return downNode;
}
public void setDownNode(Node node){
downNode = node;
}
public Node getUpNode(){
return upNode;
}
public void setUpNode(Node node){
upNode = node;
}
public Node getLeftNode(){
return leftNode;
}
public void setLeftNode(Node node){
leftNode = node;
}
public Node getRightNode(){
return rightNode;
}
public void setRightNode(Node node) {
rightNode = node;
}
}
ObservableStack 是一个由数组组成的普通堆栈。我将其连接到小程序以实现遍历动画。
我希望我已经提供了足够的细节。
最佳答案
我采用了不同的编码方式,并且速度更快。
import java.util.ArrayList;
import java.util.List;
public class PathFinder {
public static void main(String[] args) {
int[][] set1 = new int[][]{{2, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 3, 1, 1}};
long start = System.nanoTime();
Maze maze = new Maze(set1);
List<String> paths = new ArrayList<String>();
maze.findPaths(0, 0, new StringBuilder(), paths);
System.out.printf("Paths found %,d%n", paths.size());
long time = System.nanoTime() - start;
System.out.printf("Took %,.1f seconds%n", time / 1e9);
}
}
class Node {
final int num;
private boolean traversed;
private final String name;
Node(int num, String name) {
this.num = num;
this.name = name;
}
public boolean isTraversed() {
return traversed;
}
public void setTraversed(boolean traversed) {
this.traversed = traversed;
}
public String getName() {
return name;
}
public boolean isExit() {
return num == 3;
}
public boolean isPit() {
return num == 1;
}
}
class Maze {
private final Node[][] nodes;
private final int rows, cols;
private final int nonPits;
public Maze(int[][] matrix) {
rows = matrix.length;
cols = matrix[0].length;
nodes = new Node[rows][cols];
int nonPits = 0;
for (int i = 0; i < rows; i++)
for (int j = 0; j < cols; j++) {
nodes[i][j] = new Node(matrix[i][j], "" + (char) ('A' + i) + (char) ('0' + j));
if (!nodes[i][j].isPit())
nonPits++;
}
this.nonPits = nonPits;
}
public void findPaths(int x, int y, StringBuilder path, List<String> paths) {
if (x < 0 || x >= cols) return;
if (y < 0 || y >= rows) return;
Node node = nodes[y][x];
if (node.isTraversed() || node.isPit()) return;
node.setTraversed(true);
int length = path.length();
path.append(node.getName());
if (node.isExit()) {
if (path.length() == nonPits*2) {
paths.add(path.toString());
if (paths.size() % 10000 == 0)
System.out.printf("... found %,d paths%n", paths.size());
}
} else {
findPaths(x - 1, y, path, paths);
findPaths(x + 1, y, path, paths);
findPaths(x, y - 1, path, paths);
findPaths(x, y + 1, path, paths);
}
path.setLength(length);
node.setTraversed(false);
}
}
打印
Paths found 40,616
Took 81.1 seconds
<小时/>
很多时间都花在记录路径上。如果它只是计算路径(需要相同数量的递归)
import java.util.ArrayList;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Set;
public class PathFinder {
public static void main(String[] args) {
int[][] set1 = new int[][]{{2, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 0, 0, 0},
{0, 0, 0, 0, 3, 1, 1}};
long start = System.nanoTime();
Maze maze = new Maze(set1);
long[] count = {0L};
maze.findPaths(0, 0, 1, count);
System.out.printf("Paths found %,d%n", count[0]);
long time = System.nanoTime() - start;
System.out.printf("Took %,.1f seconds%n", time / 1e9);
}
}
class Node {
final int num;
private boolean traversed;
private final String name;
Node(int num, String name) {
this.num = num;
this.name = name;
}
public boolean isTraversed() {
return traversed;
}
public void setTraversed(boolean traversed) {
this.traversed = traversed;
}
public String getName() {
return name;
}
public boolean isExit() {
return num == 3;
}
public boolean isPit() {
return num == 1;
}
}
class Maze {
private final Node[][] nodes;
private final int rows, cols;
private final int nonPits;
public Maze(int[][] matrix) {
rows = matrix.length;
cols = matrix[0].length;
nodes = new Node[rows][cols];
int nonPits = 0;
for (int i = 0; i < rows; i++)
for (int j = 0; j < cols; j++) {
nodes[i][j] = new Node(matrix[i][j], "" + (char) ('A' + i) + (char) ('0' + j));
if (!nodes[i][j].isPit())
nonPits++;
}
this.nonPits = nonPits;
}
public void findPaths(int x, int y, int depth, long[] count) {
if (x < 0 || x >= cols) return;
if (y < 0 || y >= rows) return;
Node node = nodes[y][x];
if (node.isTraversed() || node.isPit()) return;
node.setTraversed(true);
if (node.isExit()) {
if (depth == nonPits) {
count[0]++;
if (count[0] % 10000 == 0)
System.out.printf("... found %,d paths%n", count[0]);
}
} else {
findPaths(x - 1, y, depth + 1, count);
findPaths(x + 1, y, depth + 1, count);
findPaths(x, y - 1, depth + 1, count);
findPaths(x, y + 1, depth + 1, count);
}
node.setTraversed(false);
}
}
打印
Paths found 40,616
Took 58.5 seconds
关于c# - 为什么java中的递归比.net慢?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11232565/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!