Go 语言 Web 开发很简单:使用模板将视图与逻辑分离

简介: 我们的 Web 服务器由两部分构成:为在用户浏览器中运行的 HTML 和 JavaScript 前端代码提供服务;接受 Web socket 连接以允许客户端进行通信。

使用 HTML 文件来改进我们的代码

改进 main.go 代码,如下:

package main
import (
  "log"
  "net/http"
)
func main() {
  http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
    w.Write([]byte(`
    <html>
      <head>
        <title>Hello</title>
      </head>
      <body>
        Hello, World!
      </body>
    </html>
    `))
  })
  // web 服务器开始
  if err := http.ListenAndServe(":8800", nil); err != nil {
    log.Fatal("ListenAndServe:", err)
  }
}


然后,像上一篇文章一样,使用命令 go run main.go 运行该代码。


打开浏览器,输入地址 http://localhost:8800/ ,就可以看到如下的进阶版 HelloWorld。


image.png


像这样将 HTML 代码嵌入到我们的 Go 代码中虽然是可行的,但它非常不合理、耦合度太高。


而且随着我们项目的增长,只会变得更糟。 接下来,我们将看看 HTML 模板如何帮助我们解决这个问题。

使用模板将视图与逻辑分离

模板允许我们将通用文本与特定文本混合,例如,加入用户名并对用户说 Hello。考虑以下模板:

Hello, {{userName}}!

然后,我们可以将 {{name}} 文本替换为真正的用户名。如果是用户“张三”登录了这个系统,就可能看到如下内容:

Hello, 张三!


Go 标准库有两个主要的模板包:一个称为 text/template 用于文本,另一个称为 html/template 用于 HTML。


html/template 包的作用与文本版本相同,只是它了解将数据注入模板的上下文。 这很有用,因为它避免了脚本注入攻击并解决了常见问题,例如必须为 URL 编码特殊字符。


最初,我们只想将 HTML 代码从 Go 代码中移动到它自己的文件中,但目前还不会混合任何文本。 template 包使加载外部文件变得非常容易,因此对我们来说是一个不错的选择。


在当前项目中新建一个 templates 文件夹,然后创建 hello.html 文件,把上一段 main.go 代码中 HTML 部分写入,但是需要做一点小小的改动才能生效:

<html>
    <head>
        <title>Hello</title>
    </head>
    <body>
        Hello, 张三!
    </body>
</html>


我们将编写自己的结构类型,负责加载、编译和交付我们的模板。 我们将定义一个接受文件名字符串的新类型,编译一次模板(使用 sync.Once 类型),保留对已编译模板的引用,然后响应 HTTP 请求。 您需要导入文本/模板、路径/文件路径和同步包以构建您的代码。sync.Once 类型保证我们作为参数传递的函数只会执行一次。


代码结构如下图:


image.png


templateHandler 结构是一个有效的 http.Handler 类型,因此我们可以将它直接传递给 http.Handle 函数并要求它处理与指定模式匹配的请求。 在前面的代码中,我们创建了一个 templateHandler 类型的新对象,将文件名指定为 hello.html,然后我们将其地址(使用运算符的 & 地址)传递给 http.Handle 函数。 我们不存储对新创建的 templateHandler 类型的引用,但这没关系,因为我们不需要再次引用它。


改进后的整体代码如下:

package main
import (
  "html/template"
  "log"
  "net/http"
  "path/filepath"
  "sync"
)
type templateHandler struct {
  once     sync.Once
  filename string
  templ    *template.Template
}
func (t *templateHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
  t.once.Do(func() {
    t.templ = template.Must(template.ParseFiles(filepath.Join("templates", t.filename)))
  })
  t.templ.Execute(w, nil)
}
func main() {
  http.Handle("/", &templateHandler{filename: "hello.html"})
  // web 服务器开始
  if err := http.ListenAndServe(":8800", nil); err != nil {
    log.Fatal("ListenAndServe:", err)
  }
}


使用 go run main.go 运行上面代码,然后在浏览器中输入网址,得到如下结果:


image.png


此时,我们的代码就很清爽了,也实现了同样的功能。

总结

又到了该做总结的时候了,从长期来看,开发就是需要做到高内聚低耦合,Go 代码和 HTML 模板的分离,其实就是代码逻辑与模板视图分开,方便项目的维护。

相关文章
|
2天前
|
存储 Go 索引
go语言使用for循环遍历
go语言使用for循环遍历
16 7
|
5天前
|
存储 Go
go语言 遍历映射(map)
go语言 遍历映射(map)
18 2
|
6天前
|
Go 调度 开发者
Go语言中的并发编程:深入理解goroutines和channels####
本文旨在探讨Go语言中并发编程的核心概念——goroutines和channels。通过分析它们的工作原理、使用场景以及最佳实践,帮助开发者更好地理解和运用这两种强大的工具来构建高效、可扩展的应用程序。文章还将涵盖一些常见的陷阱和解决方案,以确保在实际应用中能够避免潜在的问题。 ####
|
6天前
|
测试技术 Go 索引
go语言使用 range 关键字遍历
go语言使用 range 关键字遍历
14 3
|
6天前
|
测试技术 Go 索引
go语言通过 for 循环遍历
go语言通过 for 循环遍历
16 3
|
8天前
|
安全 Go 数据处理
Go语言中的并发编程:掌握goroutine和channel的艺术####
本文深入探讨了Go语言在并发编程领域的核心概念——goroutine与channel。不同于传统的单线程执行模式,Go通过轻量级的goroutine实现了高效的并发处理,而channel作为goroutines之间通信的桥梁,确保了数据传递的安全性与高效性。文章首先简述了goroutine的基本特性及其创建方法,随后详细解析了channel的类型、操作以及它们如何协同工作以构建健壮的并发应用。此外,还介绍了select语句在多路复用中的应用,以及如何利用WaitGroup等待一组goroutine完成。最后,通过一个实际案例展示了如何在Go中设计并实现一个简单的并发程序,旨在帮助读者理解并掌
|
7天前
|
Go 索引
go语言按字符(Rune)遍历
go语言按字符(Rune)遍历
21 3
|
8天前
|
开发框架 JavaScript 前端开发
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势
TypeScript 是一种静态类型的编程语言,它扩展了 JavaScript,为 Web 开发带来了强大的类型系统、组件化开发支持、与主流框架的无缝集成、大型项目管理能力和提升开发体验等多方面优势。通过明确的类型定义,TypeScript 能够在编码阶段发现潜在错误,提高代码质量;支持组件的清晰定义与复用,增强代码的可维护性;与 React、Vue 等框架结合,提供更佳的开发体验;适用于大型项目,优化代码结构和性能。随着 Web 技术的发展,TypeScript 的应用前景广阔,将继续引领 Web 开发的新趋势。
25 2
|
10天前
|
Go API 数据库
Go 语言中常用的 ORM 框架,如 GORM、XORM 和 BeeORM,分析了它们的特点、优势及不足,并从功能特性、性能表现、易用性和社区活跃度等方面进行了比较,旨在帮助开发者根据项目需求选择合适的 ORM 框架。
本文介绍了 Go 语言中常用的 ORM 框架,如 GORM、XORM 和 BeeORM,分析了它们的特点、优势及不足,并从功能特性、性能表现、易用性和社区活跃度等方面进行了比较,旨在帮助开发者根据项目需求选择合适的 ORM 框架。
34 4
|
10天前
|
缓存 监控 前端开发
在 Go 语言中实现 WebSocket 实时通信的应用,包括 WebSocket 的简介、Go 语言的优势、基本实现步骤、应用案例、注意事项及性能优化策略,旨在帮助开发者构建高效稳定的实时通信系统
本文深入探讨了在 Go 语言中实现 WebSocket 实时通信的应用,包括 WebSocket 的简介、Go 语言的优势、基本实现步骤、应用案例、注意事项及性能优化策略,旨在帮助开发者构建高效稳定的实时通信系统。
45 1