2

我正在尝试提出一种树遍历的算法,但我被卡住了。

这是一个相当困难的问题(与我问过的其他问题相比),所以我可能需要自己继续思考。但我想我会把它扔在这里。

我有以下类结构:

public class Transition
{
    // The state we are moving from.
    public String From { get; set; }
    // All the To states for this from
    public List<String>To { get; set; }
}

List<Transition> currentTransistions;

当 currentTransistions 完全填写时,它看起来像这样(对我来说):

<?xml version="1.0" encoding="utf-8"?>
<ArrayOfTransition xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema">
  <Transition>
    <From />
    <To>
      <string>Not Done</string>
    </To>
  </Transition>
  <Transition>
    <From>Not Done</From>
    <To>
      <string>In Progress</string>
      <string>Deleted</string>
    </To>
  </Transition>
  <Transition>
    <From>Deleted</From>
    <To>
      <string>Not Done</string>
    </To>
  </Transition>
  <Transition>
    <From>In Progress</From>
    <To>
      <string>Done</string>
      <string>Ready For Test</string>
      <string>Deleted</string>
    </To>
  </Transition>
  <Transition>
    <From>Done</From>
    <To>
      <string>In Progress</string>
    </To>
  </Transition>
  <Transition>
    <From>Ready For Test</From>
    <To>
      <string>In Progress</string>
      <string>Done</string>
      <string>Deleted</string>
    </To>
  </Transition>
</ArrayOfTransition>

这里的想法是我已经映射了 TFS 工作项的状态转换。我现在需要的是一种说“给定当前状态,我如何到达另一个状态”的方式。

理想情况下,它看起来像这样:

 foreach (string state in GetToFinalState(finalState, currentState, currentTransistions)
 {
     // Save the workitem at the state so we can get to the final state.
 }

GetToFinalState 必须有一种方法来计算最短路径并使用 C# 的 yield 功能为 foreach 语句一次提供一个。

我以前用过产量一,所以我想我可以弄清楚。但我不确定如何在找到最短路径的同时做到这一点(无需在函数中的每次重新计算)?

如果你已经读到这里,谢谢。如果您提供答案,请再次感谢。

4

2 回答 2

6

yield如果不计算最短路径并在整个过程完成后 ing 每个路径段,您将无法有效地做到这一点。最短路径问题的性质不适用于有效计算此类部分解决方案的算法。

由于转移图没有加权,您可以简单地在其上运行BFS来计算最短路径。你需要做这样的事情(我不确定 TFS 对象的属性,所以这只是一个伪代码):

IEnumerable<string> ShortestPath(string fromState, string toState, Transition[] currentTransitions) {
    var map = new Dictionary<string, string>();
    var edges = currentTransitions.ToDictionary(i => i.From, i => i.To);
    var q = new Queue<string>(); 
    map.Add(fromState, null);
    q.Enqueue(fromState);
    while (q.Count > 0) {
        var current = q.Dequeue();
        foreach (var s in edges[current]) {
            if (!map.ContainsKey(s)) {
                map.Add(s, current);
                if (s == toState) {
                    var result = new Stack<string>();
                    var thisNode = s;
                    do {
                        result.Push(thisNode);
                        thisNode = map[thisNode];
                    } while (thisNode != fromState);
                    while (result.Count > 0)
                        yield return result.Pop();
                    yield break;
                }
                q.Enqueue(s);
            }
        }
    }
    // no path exists
}
于 2010-02-10T23:53:47.187 回答
4

如果您需要在无环树中找到从节点到后代节点的最短路径,那么 Mehrdad 的解决方案是一个不错的解决方案。即先进行广度优先搜索,直到找到目的节点,然后计算出从起始节点到目的节点的路径。

如果您的图不是(非循环)树,而是任意加权图,那么朴素的广度优先搜索不起作用。它要么进入无限循环(如果你不聪明地跟踪你已经看到一个节点的时间),要么不能保证找到最小权重路径。

如果您处于这种情况,那么一个好的算法就是著名的“A*”算法。我在这里有一些关于如何在 C# 中实现 A* 的注释:

http://blogs.msdn.com/ericlippert/archive/tags/AStar/default.aspx

如果您有一个“估计功能”可以猜测最短路径上最有可能的下一个节点是什么,这将特别有用。

于 2010-02-11T00:29:46.373 回答