Excel VBA - For/While 循环节点路径问题

Excel VBA - For/While Loop in a Node-Path Problem

所以,

我有一组字符串 (Connector_String),其中包含显示所有可能连接的字符串(代表类似网络的节点连接)。 Connector_String 具有以下格式(我认为这对我有帮助,但如果需要我可以更改它):

例如, -RANDIAC*RANDACBD-RANDV*RANDIF-...-RANDA*RANDACAC- 这意味着 RANDIACRANDACBD 等连接。还要注意 RANDIAC 可以与另一个节点连接。

我正在尝试列出给定起点和终点的节点之间的所有可能路径。为此,我有两个字符串,其中包括所有起始节点 (Start_String) 和结束节点 (End_String)。格式如下: -RAND26RD-RAND06RD-...-RAND12RD-

我开始编写一个 for 循环代码来遍历 Connector_String 但我很快意识到我必须多次编写相同的循环(我不知道如何定义多少次).然后我写了一个 Do While 循环代码(我第一次使用它),结果根本没有 运行ning(我不明白为什么)。然后,我尝试用我在 Sub 上使用的相同 for 循环编写一个 Function,然后 运行 在 Sub 中使用 Function并在 'Function' 中(希望它能完成与 Do While 循环相同的工作)。

None 我的代码有效,但我添加了我的最后一次尝试,因为建议将它放在问题上(尽管我怀疑有经验的人是否会阅读它,因为它不太好书面 - 加上不起作用)。

Public Function Str_Search(a As String) As String
    
    Dim i As Long
    Debug.Print "Func " & a
    If InStr(End_Str, Split(a, "-")(UBound(Split(a, "-")))) > 0 Then
        Str_Search = a
        Exit Function
    End If
    
    For i = 1 To UBound(Split(Connector_String, "-")) - 1
        If Split(a, "-")(UBound(Split(a, "-"))) = Split(Split(Connector_String, "-")(i), "*")(0) Then
            a = a & "-" & Split(Split(Connector_String, "-")(i), "*")(1)
            Str_Search (a)
        End If
    Next i
    
End Function

Sub test_V4()
    Dim a As String
    Dim i As Long
    
    a = ""
    
    For i = 1 To UBound(Split(Connector_String, "-")) - 1
        If InStr(Start_String, Split(Split(Connector_String, "-")(i), "*")(0)) > 0 Then
            a = Replace(Split(Connector_String, "-")(i), "*", "-")
        ElseIf a <> "" Then
            Str_Search (a)
        ElseIf InStr(End_String, Split(a, "-")(UBound(Split(a, "-")))) > 0 Then
            Exit Sub
        End If
    Next

End Sub

最后,我的节点的另一个棘手问题是有些节点是双向的(因此,我可能有 String1*String2String2*String1),这会造成创建无限循环的问题(我没有不要尝试在我的代码中解决这个问题,因为我什至无法获得一些路径)。

见下面的字符串:

Start_String

-RAND_VW_E-RAND_VG_E-RAND_VG_F-RAND_M_2C-RAND_M_3A-RAND_VW_D-RAND_VW_V-RAND_M_1E-RAND_M_4E-RAND_VG_V-RAND_M_2D-RAND_M_3B-RAND_VW_C-RAND_VG_D-RAND_M_1F-RAND_M_4F-RAND_I_LINE-RAND_M_1E & RAND_M_4E-RAND_M_1F & RAND_M_4F

End_String

-RAND26RD-RAND06RD-RAND08RD-RAND12RD-RAND06RD-RAND02RD-RAND07RD-RAND01RD-RAND05RD-RAND03RD-RAND09RD-RAND04RD-RAND10RD-RAND20RD-RAND21RD-RAND22RD-RAND23RD-

Connector_String

-RANDIAC*RANDACBD-RANDV*RANDIF-RANDV*RANDIBD-RANDBD*RAND26RD-RANDACBD*RANDBD-RAND67F*RAND06RD-RAND89AC*RAND08RD-RANDACAC*RAND89AC-RANDA*RANDACAC-RAND_VW_E*RANDE-RAND_VG_E*RANDE-RAND_VG_F*RANDF-RAND_M_2C*RANDC-RAND_M_3A*RANDA-RANDEBD*RANDBD-RANDE*RANDEBD-RANDI*RANDIBD-RANDIBD*RANDBD-RANDF*RANDFNTH-RANDACAC*RANDACBD-RAND_VW_D*RANDD-RANDFSTH*RAND12F-RAND12F*RAND12RD-RANDIAC*RAND67AC-RAND67AC*RAND06RD-RANDFSTH*RAND02F-RAND02F*RAND02RD-RAND_VW_V*RANDV-RANDE*RANDEF-RAND_M_1E*RANDE-RAND_M_4E*RANDE-RANDEF*RANDFSTH-RAND_VG_V*RANDV-RANDV*RANDIAC-RANDFSTH*RAND67F-RAND67F*RAND07RD-RANDFNTH*RAND01RD-RANDIF*RANDFSTH-RANDB*RANDBD-RAND_M_2D*RANDD-RAND_M_3B*RANDB-RANDI*RANDIF-RANDIF*RANDFNTH-RANDFNTH*RAND05RD-RANDC*RANDACAC-RAND_VW_C*RANDC-RANDACAC*RAND67AC-RAND67AC*RAND07RD-RAND_VG_D*RANDD-RANDD*RANDBD-RAND_M_1F*RANDF-RAND_M_4F*RANDF-RANDFSTH*RAND03F-RAND03F*RAND03RD-RANDI*RANDIAC-RAND_I_LINE*RANDI-RANDIAC*RAND89AC-RAND89AC*RAND09RD-RANDF*RANDFSTH-RANDFSTH*RAND0410-RAND0410*RAND04RD-RAND0410*RAND10RD-RANDBD*RAND26BD-RANDFSTH*RANDFWST-RANDFWST*RANDFX-RAND20X*RAND20RD-RAND21X*RAND21RD-RANDFX*RAND21X-RANDFX*RAND20X-RANDEF*RANDFNTH-RANDACAC*RANDJET-RAND22Y*RAND22RD-RAND23Y*RAND23RD-RANDACY*RAND23Y-RANDJET*RANDACY-RANDACY*RAND22Y-RAND23Y*RAND23BD-RAND22Y*RAND22BD-RAND22Y*RAND23BD-RAND26BD*RAND22BD-RAND26BD*RAND23BD-RAND23BD*RAND26BD-RAND22BD*RAND26BD-RAND23BD*RAND23RD-RAND22BD*RAND22RD-RAND26BD*RAND26RD-RANDJET*RANDACX-RANDACX*RAND20X-RANDACX*RAND21X-RANDACX*RANDFX-RANDFX*RANDFWST-RANDFWST*RANDFSTH-RANDFSTH*RANDFNTH-

希望有人能帮助我。

将连接复制到名为 Connector.txt 的文本文件,并保存在与工作簿相同的文件夹中。连接写入 Sheet1,路由写入 Sheet2。使用从连接器文件构建的字典 dict 跟踪路由。 route 数组存储沿路径递归的节点。端点以黄色突出显示。

Option Explicit 
Dim dictEnd As Object 
Dim dict As Object

Sub Str_Search()

    Const CONFILE = "Connector.txt"
    
    ' dictionaries
    Set dictEnd = CreateObject("Scripting.Dictionary")
    Call EndNodes(dictEnd)
    'MsgBox Join(dictEnd.keys, vbLf)

    Set dict = CreateObject("Scripting.Dictionary")
    Call ConnectedNodes(dict, ThisWorkbook.Path & "\" & CONFILE)
    ' dump source to check
    Call DumpConnected(Sheet1, dict)
    
    ' trace routes to sheet2
    Const STEPS = 20
    Dim route(1 To STEPS) As String, arStart, k
    Dim n As Long, r As Long
    r = 2
    arStart = StartNodes()
    With Sheet2
        .Cells.Clear
        .Cells(1, 1) = "Start Node"
        For n = 0 To UBound(arStart)
            k = arStart(n)
            If dict.exists(k) Then
                route(1) = k
                Call TraceRoute(route, 1, r, Sheet2)
                r = r + 1
            ElseIf Len(k) > 0 Then
               MsgBox k & " not found", vbCritical
            End If
        Next
        .Columns.AutoFit
    End With
    MsgBox "Done", vbInformation
    
End Sub

Sub TraceRoute(ByRef route, ByRef i As Long, ByRef r As Long, ws As Worksheet)

    'Debug.Print r, i, route(i)
    Dim node As String, dest As String
    Dim n As Long, j As Long, msg As String
    
    ' current node
    node = route(i)
    ws.Cells(r, i) = node
    
    ' is end node
    If dictEnd.exists(node) Then
        ws.Cells(r, i).Interior.Color = RGB(255, 255, 0)
    End If
        
    ' check not infinite loop
    For j = 1 To i - 1
        If route(j) = node Then
            msg = "Inf Loop "
            ws.Cells(r, i + 1) = msg
            r = r + 1
            Exit Sub
        End If
    Next
    
    ' end of route ?
    If Not dict.exists(node) Then
        r = r + 1
        Exit Sub
    End If
    
    msg = ""
    For n = 1 To dict(node).Count
        dest = dict(node).Item(n)
        
        ' recurse
        If dict.exists(dest) Then
            
            i = i + 1
            route(i) = dest
            Call TraceRoute(route, i, r, ws)
            i = i - 1
        Else
            ws.Cells(r, i + 1) = dest
            If dictEnd.exists(dest) Then
                ws.Cells(r, i + 1).Interior.Color = RGB(255, 255, 0)
            End If
            r = r + 1
        End If
    Next
    
End Sub

Function StartNodes() As Variant
    StartNodes = Split("-RAND_VW_E-RAND_VG_E-RAND_VG_F-RAND_M_2C-RAND_M_3A-RAND_VW_D-RAND_VW_V" & _
    "-RAND_M_1E-RAND_M_4E-RAND_VG_V-RAND_M_2D-RAND_M_3B-RAND_VW_C-RAND_VG_D" & _
    "-RAND_M_1F-RAND_M_4F-RAND_I_LINE-RAND_M_1E-RAND_M_4E-RAND_M_1F-RAND_M_4F", "-")
End Function

Sub EndNodes(ByRef d)
    Dim k
    For Each k In Split("-RAND26RD-RAND06RD-RAND08RD-RAND12RD-RAND06RD-RAND02RD-RAND07RD-RAND01RD" & _
    "-RAND05RD-RAND03RD-RAND09RD-RAND04RD-RAND10RD-RAND20RD-RAND21RD-RAND22RD-RAND23RD-", "-")
        If Len(Trim(k)) > 0 Then d(Trim(k)) = 1
    Next
    MsgBox d.Count & " End Nodes"
End Sub

Sub ConnectedNodes(ByRef d, filename As String)

    ' read connection file
    Dim FSO As Object, ts As Object, sTxt As String
    Set FSO = CreateObject("Scripting.FilesystemObject")
    Set ts = FSO.OpenTextFile(filename)
    sTxt = ts.readAll
    ts.Close
    
    ' regular expression
    Dim regex As Object, m As Object, node As Object
    Dim n As Long, k
    
    Set regex = CreateObject("vbscript.regexp")
    With regex
        .Global = True
        .MultiLine = True
        .IgnoreCase = True
        .Pattern = "(?:-([^*]+)\*([^-]+))"
    End With
    
    ' parse file
    If regex.test(sTxt) Then
        Set m = regex.Execute(sTxt) '
        For n = 1 To m.Count
            Set node = m.Item(n - 1).submatches
            k = Trim(node(0))
            If Not dict.exists(k) And Len(k) > 0 Then
                dict.Add k, New Collection
            End If
            dict(k).Add Trim(node(1))
        Next
    End If
    MsgBox d.Count & " Connectd Nodes"
End Sub

Sub DumpConnected(ws As Worksheet, dict)

    Dim k, r As Long, n As Long
    r = 1
    With ws
        .Cells.Clear
        .Cells(r, 1) = "Start Node"
        For Each k In dict
            r = r + 1
            .Cells(r, 1) = k
            For n = 1 To dict(k).Count
                .Cells(r, n + 1) = dict(k).Item(n)
            Next
        Next
        .Columns.AutoFit
        
        .Sort.SortFields.Clear
        .Sort.SortFields.Add Key:=.Range("A1"), SortOn:=xlSortOnValues, _
            Order:=xlAscending, DataOption:=xlSortNormal
        With .Sort
            .SetRange ws.UsedRange
            .Header = xlYes
            .MatchCase = False
            .Orientation = xlTopToBottom
            .SortMethod = xlPinYin
            .Apply
        End With
    End With

End Sub