当前位置:首页 > 文章列表 > Golang > Go问答 > 更新KML文件节点的Golang实现

更新KML文件节点的Golang实现

来源:stackoverflow 2024-02-25 13:42:25 0浏览 收藏

小伙伴们有没有觉得学习Golang很有意思?有意思就对了!今天就给大家带来《更新KML文件节点的Golang实现》,以下内容将会涉及到,若是在学习中对其中部分知识点有疑问,或许看了本文就能帮到你!

问题内容

我正在使用一个 kml 文件,我用它来在 google earth 中绘制 linestring。我正在从 usb 适配器接收 gps 数据并将坐标馈送到 go 通道。我正在尝试读取通道并更新 kml 文件中的节点以添加到 linestring(从而绘制我的运动)。

这是 kml 结构:




    foo
    1
    
        bar
        
            
            
                1
                
                    1.23411166666667,10.12345678901234,0 
                    1.23421166666667,10.12345678901234,0 
                    1.23431166666667,10.12345678901234,0 
                    1.23431166666667,10.32345678901234,0 
                
            
        
    

我正在寻找附加到坐标节点。

我正在考虑两种方法之一。首先解析文件并使用正则表达式查找 并在其前面插入数据。其次,解析 xml 并查找更新节点中的值。后者似乎是更明智的选择,但我在 google 上搜索到的所有内容都向我展示了如何向 xml 树添加新节点,而不是附加到现有条目。

到目前为止,我所做的尝试感觉就像一团糟,每次从通道读取时都低效地打开文件,并且最终不起作用。

type LineString struct {
    coordinates string `xml:"coordinates"`
}    

func plotLocation(c chan data.GpsPos) {
    /*
        continuously read from the channel
        use the location data to plot a breadcrumb trail
    */

    defer wg.Done()

    for currentCoords := range c {

        file, err := os.Open("/Users/me/foo.kml")
        if err != nil {
            log.Fatal(err)
        }
        defer file.Close()

        var buf bytes.Buffer
        decoder := xml.NewDecoder(file)
        encoder := xml.NewEncoder(&buf)

        for {
            token, err := decoder.Token()
            if err == io.EOF {
                break
            }
            if err != nil {
                log.Printf("error getting token: %v\n", err)
                break
            }
            switch v := token.(type) {
            case xml.StartElement:
                if v.Name.Local == "LineString" {

                    var coords LineString
                    if err = decoder.DecodeElement(&coords, &v); err != nil {
                        log.Fatal(err)
                    }
                    coords.coordinates += fmt.Sprintf("%f,%f,%d\n", currentCoords.Lat, currentCoords.Long, 0)
                    if err = encoder.EncodeElement(coords, v); err != nil {
                        log.Fatal(err)
                    }
                    continue
                }
            }
            if err := encoder.EncodeToken(xml.CopyToken(token)); err != nil {
                log.Fatal(err)
            }
        }
    }
}

我是否做了一些明显错误的事情,是否有更好的方法将此数据写入文件(大约每秒发生一次)?


正确答案


文件是否在您的应用程序之外发生更改?如果没有,那么您可以在循环之前解析文件一次,维护坐标列表,并在每次更改时将其写出,以便外部应用程序可以看到中间结果。如果您计划进行更多转换,或者如果您想从头开始生成整个文件,这也将很有用。

首先,您需要一个具有适当标签的结构(请参阅 xml.Unmarshal)。我通常从在线生成器开始处理这些事情:

// type definition adapted from https://www.onlinetool.io/xmltogo/

type kml struct {
    xmlname xml.name `xml:"kml"`
    text    string   `xml:",chardata"`
    xmlns   string   `xml:"xmlns,attr"`
    gx      string   `xml:"gx,attr"`
    kml     string   `xml:"kml,attr"`
    atom    string   `xml:"atom,attr"`
    folder  struct {
        text     string `xml:",chardata"`
        name     string `xml:"name"`
        open     string `xml:"open"`
        document struct {
            text      string `xml:",chardata"`
            name      string `xml:"name"`
            placemark struct {
                text  string `xml:",chardata"`
                style struct {
                    text      string `xml:",chardata"`
                    linestyle struct {
                        text  string `xml:",chardata"`
                        color string `xml:"color"`
                        width string `xml:"width"`
                    } `xml:"linestyle"`
                    polystyle struct {
                        text    string `xml:",chardata"`
                        color   string `xml:"color"`
                        fill    string `xml:"fill"`
                        outline string `xml:"outline"`
                    } `xml:"polystyle"`
                } `xml:"style"`
                linestring struct {
                    text        string `xml:",chardata"`
                    tessellate  string `xml:"tessellate"`
                    coordinates string `xml:"coordinates"`
                } `xml:"linestring"`
            } `xml:"placemark"`
        } `xml:"document"`
    } `xml:"folder"`
} 

我会为此做一些帮手:

func readkml(filename string) (*kml, error) {
  f, err := os.open(filename)
  if err != nil {
    return nil, fmt.errorf("opening kml file: %w", err) // contains filename
  }
  defer f.close() // reading, ignoring error is acceptable
  var kml kml
  if err := xml.newdecoder(f).decode(&kml); err != nil {
    return nil, fmt.errorf("decoding xml from %q as kml: %w", filename, err)
  }
  return &kml, nil
}

func writekml(filename string, kml *kml) error {
  f, err := os.create(filename)
  if err != nil {
    return fmt.errorf("creating kml file: %w", err) // contains filename
  }
  defer f.close() // double close is ok for *os.file
  enc := xml.newencoder(f)
  enc.indent("", "    ")
  if err := enc.encode(kml); err != nil {
    return nil, fmt.errorf("encoding kml to %q: %w", filename, err)
  }
  return nil
}

然后你的循环可能看起来像这样:

kml, err := readKML(filename)
if err != nil {
  return err // contains context
}

coordinates := strings.Fields(kml.Folder.Document.Placemark.LineString.Coordinates)

for coord := range incoming {
  line := fmt.Sprintf("%f,%f,%d\n", coord.Lat, coord.Long, 0)
  coordinates = append(coordinates, coord)
  
  kml.Folder.Document.Placemark.LineString.Coordinates = strings.Join(coordinates, "\n")
  if err := writeKML(filename, kml); err != nil {
    log.Printf("Warning: failed to update %q: %s", filename, err)
  }
}

当查看您的代码时,我怀疑问题在于您推迟了文件关闭,该关闭将在函数返回时执行,而不是在循环继续时执行。您也许也可以使这种方法发挥作用,为此,我建议您将逻辑分解为函数,以便可以独立测试每个部分,这也可能意味着您的 defer 现在在函数内正确确定了范围。

好了,本文到此结束,带大家了解了《更新KML文件节点的Golang实现》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
取消速率限制下的令牌恢复取消速率限制下的令牌恢复
上一篇
取消速率限制下的令牌恢复
检查 float64 转换为 float32 时的精度损失方法
下一篇
检查 float64 转换为 float32 时的精度损失方法
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 笔灵AI生成答辩PPT:高效制作学术与职场PPT的利器
    笔灵AI生成答辩PPT
    探索笔灵AI生成答辩PPT的强大功能,快速制作高质量答辩PPT。精准内容提取、多样模板匹配、数据可视化、配套自述稿生成,让您的学术和职场展示更加专业与高效。
    24次使用
  • 知网AIGC检测服务系统:精准识别学术文本中的AI生成内容
    知网AIGC检测服务系统
    知网AIGC检测服务系统,专注于检测学术文本中的疑似AI生成内容。依托知网海量高质量文献资源,结合先进的“知识增强AIGC检测技术”,系统能够从语言模式和语义逻辑两方面精准识别AI生成内容,适用于学术研究、教育和企业领域,确保文本的真实性和原创性。
    40次使用
  • AIGC检测服务:AIbiye助力确保论文原创性
    AIGC检测-Aibiye
    AIbiye官网推出的AIGC检测服务,专注于检测ChatGPT、Gemini、Claude等AIGC工具生成的文本,帮助用户确保论文的原创性和学术规范。支持txt和doc(x)格式,检测范围为论文正文,提供高准确性和便捷的用户体验。
    38次使用
  • 易笔AI论文平台:快速生成高质量学术论文的利器
    易笔AI论文
    易笔AI论文平台提供自动写作、格式校对、查重检测等功能,支持多种学术领域的论文生成。价格优惠,界面友好,操作简便,适用于学术研究者、学生及论文辅导机构。
    50次使用
  • 笔启AI论文写作平台:多类型论文生成与多语言支持
    笔启AI论文写作平台
    笔启AI论文写作平台提供多类型论文生成服务,支持多语言写作,满足学术研究者、学生和职场人士的需求。平台采用AI 4.0版本,确保论文质量和原创性,并提供查重保障和隐私保护。
    41次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码