Search code examples
xmlvb.net.net-corexmldocument

Traversing through XML


I have the following XML file:

<Tournament TeamPlayers="1">
    <Teams>
        <Team>
            <TeamID>0</TeamID>
            <TeamName>Sample</TeamName>
            <Status>10</Status>
            <Memo>Sample Team</Memo>
            <ByeRounds>0</ByeRounds>
            <Players>
                <Player>
                    <MemberNumber>1</MemberNumber>
                    <MemberName>Dummy</MemberName>
                    <PlayerFirstName>Test</PlayerFirstName>
                    <PlayerLastName>User</PlayerLastName>
                    <SeatOrder>A</SeatOrder>
                </Player>
                <Player>
                    <MemberNumber></MemberNumber>
                    <MemberName></MemberName>
                    <PlayerFirstName></PlayerFirstName>
                    <PlayerLastName></PlayerLastName>
                    <SeatOrder>B</SeatOrder>
                </Player>
                <Player>
                    <MemberNumber></MemberNumber>
                    <MemberName></MemberName>
                    <PlayerFirstName></PlayerFirstName>
                    <PlayerLastName></PlayerLastName>
                    <SeatOrder>C</SeatOrder>
                </Player>
            </Players>
        </Team>
    </Teams>
</Tournament>

The idea is that I loop through the <Players> block and add the data there to a holder of class data. However, when I run the following code, which looks like it is yielding the contents of the Inner Players field, VS stops warning that it is empty:

    Dim teamNodes As XmlNodeList = xmlDoc.DocumentElement.SelectNodes("/Tournament/Teams/Team")
    Dim lstOutputTeams As New List(Of TournamentTeam)

    ' Check the atrribute of "TeamPlayers" in the root. 
    'Dim intExpectedPlayers As Integer = xmlDoc.Attributes("/").Value
    Dim intExpectedPlayers As Integer = xmlDoc.SelectSingleNode("/Tournament").Attributes("TeamPlayers").Value

    For Each node As XmlNode In teamNodes

        Dim tpPlayerA As New PlayerInfo
        Dim tpPlayerB As New PlayerInfo
        Dim tpPlayerC As New PlayerInfo

        Dim playerNodes As XmlNodeList = node.SelectNodes("Players")

        For Each innerNode As XmlNode In playerNodes
            Dim tpPlayer As PlayerInfo = New PlayerInfo With {
                .strMembershipName = innerNode.SelectSingleNode("MemberName").InnerText,
                .strMembershipNo = innerNode.SelectSingleNode("MemberNumber").InnerText,
                .strPlayerFirstName = innerNode.SelectSingleNode("PlayerFirstName").InnerText,
                .strPlayerLastName = innerNode.SelectSingleNode("PlayerLastName").InnerText,
                .strSeatOrder = innerNode.SelectSingleNode("SeatOrder").InnerText
            }

            Select Case tpPlayer.strSeatOrder
                Case "A"
                    tpPlayerA = tpPlayer
                Case "B"
                    tpPlayerB = tpPlayerB
                Case "C"
                    tpPlayerC = tpPlayerC
            End Select
        Next

        lstOutputTeams.Add(New TournamentTeam() With {
                           .strTeamName = node.SelectSingleNode("TeamName").InnerText,
                           .intTeamID = node.SelectSingleNode("TeamID").InnerText,
                           .intByeRounds = node.SelectSingleNode("ByeRounds").InnerText,
                           .intStatus = node.SelectSingleNode("Status").InnerText,
                           .strMemo = node.SelectSingleNode("Memo").InnerText,
                           .tpPlayerA = tpPlayerA,
                           .tpPlayerB = tpPlayerB,
                           .tpPlayerC = tpPlayerC}
                           )
    Next

In particular, the error is: System.NullReferenceException: 'Object reference not set to an instance of an object.'. I'm not sure where the null reference is taking place. (The breakpoint places it at the Dim tpPlayer As Player Info line.


Solution

  • It may be causing you grief that your Select Case is assigning variables to themselves:

    Select Case tpPlayer.strSeatOrder
        Case "A"
            tpPlayerA = tpPlayer
        Case "B"
            tpPlayerB = tpPlayerB
        Case "C"
            tpPlayerC = tpPlayerC
    End Select
    

    The RHS of each should just be tpPlayer.

    You caused this bug by doing this:

        Dim tpPlayerA As New PlayerInfo
        Dim tpPlayerB As New PlayerInfo
        Dim tpPlayerC As New PlayerInfo
    

    You should have done this instead:

        Dim tpPlayerA As PlayerInfo
        Dim tpPlayerB As PlayerInfo
        Dim tpPlayerC As PlayerInfo
    

    Here's what your code would look like with XDocument. It's much cleaner.

        Dim intExpectedPlayers As Integer = CInt(xd.Root.Attribute("TeamPlayers"))
        
        Dim lstOutputTeams As List(Of TournamentTeam) = _
        ( _
            From team In xd.Root.Element("Teams").Elements("Team") _
            Let players = team.Element("Players").Elements("Player").Select(Function(player) _
                New PlayerInfo() With _
                {
                    .strMembershipName = player.Element("MemberName").Value,
                    .strMembershipNo = player.Element("MemberNumber").Value,
                    .strPlayerFirstName = player.Element("PlayerFirstName").Value,
                    .strPlayerLastName = player.Element("PlayerLastName").Value,
                    .strSeatOrder = player.Element("SeatOrder").Value
                }).ToDictionary(Function(x) x.strSeatOrder) _
            Select New TournamentTeam() With _
            {
                .intTeamID = CInt(team.Element("TeamID")),
                .strTeamName = CStr(team.Element("TeamName")),
                .intStatus = CInt(team.Element("Status")),
                .strMemo = CStr(team.Element("Memo")),
                .intByeRounds = CInt(team.Element("ByeRounds")),
                .tpPlayerA = players("A"),
                .tpPlayerB = players("B"),
                .tpPlayerC = players("C")
            } _
        ).ToList()
    

    Here's the creation of the xd that I used for testing:

        Dim xd = XDocument.Parse("<Tournament TeamPlayers=""1"">
        <Teams>
            <Team>
                <TeamID>0</TeamID>
                <TeamName>Sample</TeamName>
                <Status>10</Status>
                <Memo>Sample Team</Memo>
                <ByeRounds>0</ByeRounds>
                <Players>
                    <Player>
                        <MemberNumber>1</MemberNumber>
                        <MemberName>Dummy</MemberName>
                        <PlayerFirstName>Test</PlayerFirstName>
                        <PlayerLastName>User</PlayerLastName>
                        <SeatOrder>A</SeatOrder>
                    </Player>
                    <Player>
                        <MemberNumber></MemberNumber>
                        <MemberName></MemberName>
                        <PlayerFirstName></PlayerFirstName>
                        <PlayerLastName></PlayerLastName>
                        <SeatOrder>B</SeatOrder>
                    </Player>
                    <Player>
                        <MemberNumber></MemberNumber>
                        <MemberName></MemberName>
                        <PlayerFirstName></PlayerFirstName>
                        <PlayerLastName></PlayerLastName>
                        <SeatOrder>C</SeatOrder>
                    </Player>
                </Players>
            </Team>
        </Teams>
    </Tournament>")
    

    I'd also suggest changing your classes to use the standard naming conventions for .NET.

    Public Class TournamentTeam
        Public Name As String
        Public ID As Integer
        Public ByeRounds As Integer
        Public Status As Integer
        Public Memo As String
        Public PlayerA As PlayerInfo
        Public PlayerB As PlayerInfo
        Public PlayerC As PlayerInfo
    End Class
    
    Public Class PlayerInfo
        Public MembershipName As String
        Public MembershipNo As String
        Public FirstName As String
        Public LastName As String
        Public SeatOrder As String
    End Class
    

    Finally, I'd also look at removing the three Player properties and put a dictionary in instead.

    If it's always A, B, and C then use an Enum as the key, otherwise use a String.

    Public Class TournamentTeam
        Public Name As String
        Public ID As Integer
        Public ByeRounds As Integer
        Public Status As Integer
        Public Memo As String
        Public Players As Dictionary(Of PlayerSeat, PlayerInfo)
    End Class
    
    Public Enum PlayerSeat
        A
        B
        C
    End Enum
    
    Public Class PlayerInfo
        Public MembershipName As String
        Public MembershipNo As String
        Public FirstName As String
        Public LastName As String
        Public SeatOrder As String
    End Class
    

    Then your code would be:

        Dim lstOutputTeams As List(Of TournamentTeam) = _
        ( _
            From team In xd.Root.Element("Teams").Elements("Team") _
            Select New TournamentTeam() With _
            {
                .ID = CInt(team.Element("TeamID")),
                .Name = CStr(team.Element("TeamName")),
                .Status = CInt(team.Element("Status")),
                .Memo = CStr(team.Element("Memo")),
                .ByeRounds = CInt(team.Element("ByeRounds")),
                .Players = team.Element("Players").Elements("Player").Select(Function(player) _
                New PlayerInfo() With _
                {
                    .MembershipName = player.Element("MemberName").Value,
                    .MembershipNo = player.Element("MemberNumber").Value,
                    .FirstName = player.Element("PlayerFirstName").Value,
                    .LastName = player.Element("PlayerLastName").Value,
                    .SeatOrder = player.Element("SeatOrder").Value
                }).ToDictionary(Function(x) CType([Enum].Parse(GetType(PlayerSeat), x.SeatOrder), PlayerSeat))
            } _
        ).ToList()