dpp66953 2018-10-18 15:55
浏览 8

xml.unmarshal不遍历XML树

I am attempting to marshal XML over to a set of custom structs and it isn't working as I would expect. I would expect that the xml.StartElement for each of my "UnmarshalXML" functions would change for each object that it is processing (e.g. TrainingCenterDatabase vs Activities). That doesn't appear to be happening. My xml.StartElement is always the root element of the entire XML File.

In addition, my final decoded Struct is empty when the Unmarshaling is completed. I hypothesize that because the wrong "startElement" is being passed in then it isn't finding any values to parse.

What am I doing wrong?

Structs and Functions:

func (v *TrainingCenterDatabase) UnmarshalXML(d *xml.Decoder, startElement xml.StartElement) error {
    decodeError := d.DecodeElement(&v.Activities, &startElement)
    fmt.Println("TrainingCenterDatabase StartELement Name: " + startElement.Name.Local)
    if decodeError != nil {
        logging.LogError("tcxhelper - XMLTpxFile.UnmarshalXML Error - "+decodeError.Error(), "")
        return decodeError
    }
    return nil
}

func (v *Activities) UnmarshalXML(d *xml.Decoder, startElement xml.StartElement) error {
    decodeError := d.DecodeElement(&v.Activity, &startElement)
    fmt.Println("Activities StartELement Name: " + startElement.Name.Local)
    if decodeError != nil {
        logging.LogError("tcxhelper - XMLTpxFile.UnmarshalXML Error - "+decodeError.Error(), "")
        return decodeError
    }
    return nil
}

func (v *Activity) UnmarshalXML(d *xml.Decoder, startElement xml.StartElement) error {
    for _, attr := range startElement.Attr {
        fmt.Println("Attr Name: " + attr.Name.Local)
        fmt.Println("Attr Value: " + attr.Value)
    }
    fmt.Println("Activity StartELement Name: " + startElement.Name.Local)

    decodeError := d.DecodeElement(&v.Lap, &startElement)
    if decodeError != nil {
        logging.LogError("tcxhelper - XMLTpxFile.UnmarshalXML Error - "+decodeError.Error(), "")
        return decodeError
    }
    return nil
}

type TpxFile struct {
    TpxFile TrainingCenterDatabase `xml:"TrainingCenterDatabase"`
}

type TrainingCenterDatabase struct {
    Activities Activities `xml:"Activities"`
}

type Activities struct {
    Activity Activity `xml:"Activity"`
}

type Activity struct {
    Sport string    `xml:"Sport"`
    ID    time.Time `xml:"Id"`
    Lap   Lap       `xml:"Lap"`
}

type Lap struct {
    Cadence       string `xml:"Cadence"`
    TriggerMethod string `xml:"TriggerMethod"`
    Track         Track  `xml:"Track"`
}

type Track struct {
    Trackpoint          Trackpoint          `xml:"Trackpoint"`
    MaximumSpeed        string              `xml:"MaximumSpeed"`
    MaximumHeartRateBpm MaximumHeartRateBpm `xml:"MaximumHeartRateBpm"`
    Intensity           string              `xml:"Intensity"`
    Calories            string              `xml:"Calories"`
    AverageHeartRateBpm AverageHeartRateBpm `xml:"AverageHeartRateBpm"`
    StartTime           time.Time           `xml:"StartTime"`
    TotalTimeSeconds    string              `xml:"TotalTimeSeconds"`
    DistanceMeters      string              `xml:"DistanceMeters"`
}

type MaximumHeartRateBpm struct {
    Value string `xml:"Value"`
}

type AverageHeartRateBpm struct {
    Value string `xml:"Value"`
}

type Trackpoint []struct {
    Position       Position     `xml:"Position"`
    AltitudeMeters string       `xml:"AltitudeMeters"`
    DistanceMeters string       `xml:"DistanceMeters"`
    HeartRateBpm   HeartRateBpm `xml:"HeartRateBpm"`
    Cadence        string       `xml:"Cadence"`
    Extensions     Extensions   `xml:"Extensions"`
    Time           time.Time    `xml:"Time"`
}

type Extensions struct {
    TPX TPX `xml:"TPX"`
}

type TPX struct {
    Xmlns string `xml:"xmlns"`
    Speed string `xml:"Speed"`
    Watts string `xml:"Watts"`
}
type HeartRateBpm struct {
    Value string `xml:"Value"`
}

type Position struct {
    LatitudeDegrees  string `xml:"LatitudeDegrees"`
    LongitudeDegrees string `xml:"LongitudeDegrees"`
} 

Code to Marshal

func TestMarshalTPX(t *testing.T) {

    fileByteArray, myErr := ioutil.ReadFile("/my/local/directory/Lunch_Ride.tcx")
    if myErr != nil {
        t.Error("Error Reading File: " + myErr.Error())
    }
    tpxFile := TrainingCenterDatabase{}

    marshalError := xml.Unmarshal(fileByteArray, &tpxFile)
    if marshalError != nil {
        t.Error("Marshal Error: " + marshalError.Error())
    }
}

Sample XML:

<?xml version="1.0" encoding="UTF-8"?>
<TrainingCenterDatabase
  xsi:schemaLocation="http://www.garmin.com/xmlschemas/TrainingCenterDatabase/v2 http://www.garmin.com/xmlschemas/TrainingCenterDatabasev2.xsd"
  xmlns:ns5="http://www.garmin.com/xmlschemas/ActivityGoals/v1"
  xmlns:ns3="http://www.garmin.com/xmlschemas/ActivityExtension/v2"
  xmlns:ns2="http://www.garmin.com/xmlschemas/UserProfile/v2"
  xmlns="http://www.garmin.com/xmlschemas/TrainingCenterDatabase/v2"
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
 <Activities>
  <Activity Sport="Biking">
   <Id>2018-10-16T18:41:12Z</Id>
   <Lap StartTime="2018-10-16T18:41:12Z">
    <TotalTimeSeconds>3732</TotalTimeSeconds>
    <DistanceMeters>26451.5</DistanceMeters>
    <MaximumSpeed>52560.0</MaximumSpeed>
    <Calories>0</Calories>
    <AverageHeartRateBpm>
     <Value>148</Value>
    </AverageHeartRateBpm>
    <MaximumHeartRateBpm>
     <Value>188</Value>
    </MaximumHeartRateBpm>
    <Intensity>Active</Intensity>
    <Cadence>79</Cadence>
    <TriggerMethod>Manual</TriggerMethod>
    <Track>
     <Trackpoint>
      <Time>2018-10-16T18:41:12Z</Time>
      <Position>
       <LatitudeDegrees>39.7711140</LatitudeDegrees>
       <LongitudeDegrees>-105.0392580</LongitudeDegrees>
      </Position>
      <AltitudeMeters>1646.2</AltitudeMeters>
      <DistanceMeters>0.0</DistanceMeters>
      <HeartRateBpm>
       <Value>97</Value>
      </HeartRateBpm>
      <Cadence>69</Cadence>
      <Extensions>
       <TPX xmlns="http://www.garmin.com/xmlschemas/ActivityExtension/v2">
        <Speed>0.0</Speed>
        <Watts>320</Watts>
       </TPX>
      </Extensions>
     </Trackpoint>
     <Trackpoint>
      <Time>2018-10-16T18:41:13Z</Time>
      <Position>
       <LatitudeDegrees>39.7711120</LatitudeDegrees>
       <LongitudeDegrees>-105.0392590</LongitudeDegrees>
      </Position>
      <AltitudeMeters>1646.2</AltitudeMeters>
      <DistanceMeters>0.2</DistanceMeters>
      <HeartRateBpm>
       <Value>97</Value>
      </HeartRateBpm>
      <Cadence>73</Cadence>
      <Extensions>
       <TPX xmlns="http://www.garmin.com/xmlschemas/ActivityExtension/v2">
        <Speed>0.2</Speed>
        <Watts>195</Watts>
       </TPX>
      </Extensions>
     </Trackpoint>
    </Track>
   </Lap>
  </Activity>
 </Activities>
</TrainingCenterDatabase>
  • 写回答

0条回答 默认 最新

    报告相同问题?

    悬赏问题

    • ¥15 执行 virtuoso 命令后,界面没有,cadence 启动不起来
    • ¥50 comfyui下连接animatediff节点生成视频质量非常差的原因
    • ¥20 有关区间dp的问题求解
    • ¥15 多电路系统共用电源的串扰问题
    • ¥15 slam rangenet++配置
    • ¥15 有没有研究水声通信方面的帮我改俩matlab代码
    • ¥15 ubuntu子系统密码忘记
    • ¥15 保护模式-系统加载-段寄存器
    • ¥15 电脑桌面设定一个区域禁止鼠标操作
    • ¥15 求NPF226060磁芯的详细资料