bootstrap网页模板源码_Go Web 编程--超详细的模板库应用指南

8fcf12ebdc03fb9e6f6d29e3d5312bfc.png

如果你有过Web编程的经验,那么或多或少都听说过或者使用过模板。简而言之,模板是可用于创建动态内容的文本文件。例如,你有一个网站导航栏的模板,其中动态内容的一部分可能是根据当前用户是否登录显示登录还是退出按钮。

Go提供了两个模板库text/templatehtml/template。这两个模板库的使用方式是相同的,但是html/template包在渲染页面模板时会在后台进行一些编码以帮助防止造成代码注入(XSS 攻击)。

因为两个模板库都使用相同的接口,因此本文中介绍的所有内容均可用于这两个程序包,但是大多数时候我们都会使用html/template程序包来生成HTML代码段。

Go Web 编程系列的每篇文章的源代码都打了对应版本的软件包,供大家参考。公众号中回复gohttp07获取本文源代码

模板文件的后缀名

模板文件可以使用.html或任何其他扩展名。但是通常我们将使用.gohtml扩展名来命名模板文件,因为编辑器通常使用它来表示你想要高亮Go HTML模板语法。 AtomSublime Text等编辑器都具有Go插件,来默认识别此扩展名。

模板语法

我们先来创建一个简单的模板文件test.gohtml:

<!DOCTYPE html>
<html><head><meta http-equiv="Content-Type" content="text/html; charset=utf-8"><title>Go Web</title></head><body>{{ . }}</body>
</html>

{{ 和 }} 中间的半角句号 . 它代表模板对象执行Execute(w, data)传入模板的数据,它是顶级作用域范围内的,根据传入的数据不同渲染不同的内容。. 可以代表Go语言中的任何类型,如结构体、Map等。

在写模板的时候,会经常用到.。比如{{.}}{{len .}}{{.Name}}{{$x.Name}}

{{ 和 }} 包裹的内容统称为 action,分为两种类型:

  • 数据求值(data evaluations)
  • 控制结构(control structures)

action求值的结果会直接复制到模板中,控制结构和我们写Go程序差不多,也是条件语句、循环语句、变量、函数调用等等...模板中的 action 并不多,我们一个一个看。

注释

{{/* comment */}}

裁剪空字符

注意裁剪的是替换内容前面或者后面的空字符,你可以理解成模板中{{前面或}}后面的空字符(包括换行符、制表符、空格等)。

// 裁剪 content 前后的空字符
{{- content -}}// 裁剪 content 前面的空字符
{{- content }}// 裁剪 content 后面的空字符
{{ content -}}

文本输出

{{ pipeline }}

pipeline代表的数据会产生与调用 fmt.Print 函数类似的输出,例如整数类型的 3 会转换成字符串 "3" 输出。

条件语句

{{ if pipeline }} T1 {{ end }}{{ if pipeline }} T1 {{ else }} T0 {{ end }}{{ if pipeline }} T1 {{ else if pipeline }} T0 {{ end }}// 上面的语法其实是下面的简写
{{ if pipeline }} T1 {{ else }}{{ if pipeline }} T0 { {end }}{{ end }}{{ if pipeline }} T1 {{ else if pipeline }} T2 {{ else }} T0 {{ end }}

如果 pipeline 的值为空,不会输出 T1,除此之外 T1 都会被输出。

空值有false0nil空字符串 ""(长度为 0 的字符串)。

循环语句

{{ range pipeline }} T1 {{ end }}// 这个 else 比较有意思,如果 pipeline 的长度为 0 则输出 else 中的内容
{{ range pipeline }} T1 {{ else }} T0 {{ end }}// 获取容器的下标
{{ range $index, $value := pipeline }} T1 {{ end }}

循环语句中的pipeline 的值必须是数组、切片、字典和通道中的一种,即可迭代类型的值,根据值的长度输出多个 T1。

define

{{ define "name" }} T {{ end }}

定义命名为 name 的模板。

template

{{ template "name" }}{{ template "name" pipeline }}

第一种是直接执行名为name的模板,模板的全局数据对象.设置为nil。第二种是点.设置为pipeline的值,并执行名为name的模板。

block

{{ block "name" pipeline }} T1 {{ end }}

block 的语义是如果有命名为 name 的模板,就引用过来执行,如果没有命名为 name 的模板,就是执行自己定义的内容。换句话说,block可以认为是设置一个默认模板。

with

{{ with pipeline }} T1 {{ end }}// 如果 pipeline 是空值则输出 T0
{{ with pipeline }} T1 {{ else }} T0 {{ end }}{{ with arg }}. // 此时 . 就是 arg
{{ end }}

with 创建一个新的上下文环境,在此环境中的 . 与外面的 . 无关。

对于第一种格式,当pipeline不为0值的时候,点.设置为pipeline运算的值,否则跳过。对于第二种格式,当pipeline为0值时,执行else语句块,否则.设置为pipeline运算的值,并执行T1。

例如:

{{with .Person}}{{ .Name}}{{end}}

在这个 with 块中.Name实际上引用的是全局数据对象的.Person.Name

实践练习:课程花名册页面

了解完模板语法后,接下来让我们再http_demo项目中结合BootStrap创建一个简单的模板,来展示服务器如何把数据传递给模板、渲染HTML页面,把页面响应返回给客户端。

我们创建一个用来展示大学物理课程的花名册(授课老师和上课学生)

创建页面模板

首先在我们的项目添加一个views目录用于存放模板文件,在创建三个模板文件分别是:

layout.gohtml 用于存放页面的整体布局。

<html lang="en">
<head><meta charset="utf-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1"><meta name="description" content=""><meta name="author" content=""><title>Bootstrap Template Page for Go Web Programming</title><!-- Bootstrap core CSS --><link href="//cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet">
</head><body>{{ template "nav" .}}<div class="container">{{template "content" .}}
</div> <script src="//cdn.bootcss.com/bootstrap/3.3.7/js/bootstrap.min.js"></script>
</body>
</html>

nav.gohtml是网页头部区域的页面模板。

{{define "nav"}}
<nav class="navbar navbar-inverse navbar-fixed-top"><div class="container"><div class="navbar-header"><a class="navbar-brand" href="#">Person general infor</a></div></div>
</nav><div class="jumbotron"><div class="container"><h1>Hello, Professor {{.Teacher.Name}}</h1><ul><li>Name   : {{.Teacher.Name}}<p><li>Subject : {{.Teacher.Subject}}</ul><p><a class="btn btn-primary btn-lg" href="#" role="button">More &raquo;</a></p></div>
</div>
{{end}}

content.gohtml是网页主体内容部分的页面模板。

{{define "content"}}
{{range .Students}}
<div class="row"><div class="col-md-4"><h2>Name</h2><p>Name has the value of : {{.Name}} </p><p><a class="btn btn-default" href="#" role="button">More &raquo;</a></p></div><div class="col-md-4"><h2>Id</h2><p>Id has the value of : {{.Id}} </p><p><a class="btn btn-default" href="#" role="button">More &raquo;</a></p></div><div class="col-md-4"><h2>Country</h2><p>Country has the value of : {{.Country}} </p><p><a class="btn btn-default" href="#" role="button">More &raquo;</a></p></div>
</div>
{{end}}
{{end}}

layout.gohtml中我们引用了另外的两个模板:

{{ template "nav" .}}
{{template "content" .}}

这样不同的页面变化的部分就只是content部分,针对不同的页面我们只需要定义多个content模板,每次根据不同请求使用不同的content模板就行了。当然这里的例子有点简陋,大家理解意思就行了。

注意模板名称后面的.,我们把layout.gohtml的全局数据对象传给了另外两个模板这样,在子模板里也能访问传给模板的数据了。如果页面模板中使用的数据字段和循环语句有点疑惑可以先不用管,继续往下看,等看过传给页面模板的数据后自然就理解了。

创建响应 页面请求的Handler

接下来创建一个伺服页面请求的Handler

package handlerimport ("fmt""html/template""net/http"
)type Teacher struct {Name    stringSubject string
}
type Student struct {Id      intName    stringCountry string
}type Rooster struct {Teacher TeacherStudents []Student
}func ShowIndexView(response http.ResponseWriter, request *http.Request) {teacher := Teacher{Name:    "Alex",Subject: "Physics",}students := []Student{{Id: 1001, Name: "Peter", Country: "China"},{Id: 1002, Name: "Jeniffer", Country: "Sweden"},}rooster := Rooster{Teacher:  teacher,Students: students,}tmpl, err := template.ParseFiles("./views/layout.gohtml", "./views/nav.gohtml", "./views/content.gohtml")if err != nil {fmt.Println("Error " +  err.Error())}tmpl.Execute(response, rooster)
}

使用template.ParseFiles加载这个页面要使用的全部三个模板(如果加载少了,访问页面时会发生panic),然后使用模板对象的 Execute方法把我们存储了花名册信息的数据对象传给模板: tmpl.Execute(response, rooster) 渲染页面并写到响应里去(http.ResponseWriter对象)。

注册页面路由

处理程序写完后,为其注册路由,在我们项目的路由模块添加如下路由:

package routerimport ("example.com/http_demo/middleware""github.com/gorilla/mux""example.com/http_demo/handler"
)func RegisterRoutes(r *mux.Router) {r.Use(middleware.Logging())
...viewRouter := r.PathPrefix("/view").Subrouter()viewRouter.HandleFunc("/index", handler.ShowIndexView)
}

访问页面

现在所有步骤都完成了,重启我们的服务器后就可以访问到新写的页面了。

如果是在本地电脑里,用Ctrl+C结束服务器进程后再次执行go run main.go。如果是使用我们之前文章里的Docker开发环境的话(公众号回复:go-docker 获取Docker环境的安装指南)需要在docker-compose.yml所在的目录里用docker-compose restart重启服务。

打开浏览器输入http://localhost:8000/view/index就能访问到我们刚才写的页面了。

7f60587995f771435c74192d06905a27.png

总结

今天的文章讲解了Go模板最常使用的几个功能的使用方法,使用html/template模板库结合BootStrap做页面模板,还是比较简单的BootStrap帮我们解决了很多前端的样式问题。模板库还有很多更高级的用法,比如在模板中调用函数、定义变量等功能,可以看下文末给出的参考链接了解这些内容。在前后端分离架构流行的今天我觉得作为用Go开发的后端工程师了解文章中列出的这些功能就够了。

今天的例子中是通过 CDN 引用的BootStrap静态资源,到目前我们的服务器还无法伺服静态资源,这个我们下篇文章再讲。公众号回复gohttp07即可获取今天文章中示例代码的下载链接。如果觉得我的文章有收获,请帮忙分享给更多人。

参考

Go 语言标准库 text/template 包深入浅出

An Introduction to Templates in Go

前文回顾

深入学习用Go编写HTTP服务器

设置HTTP服务器的路由

Go Web编程--应用ORM

Go Web编程--深入学习解析HTTP请求

41b7edd13a9ea445fb6e4df32b918be5.png
扫码关注公众号:网管叨bi叨

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/559512.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python matpoltlib绘制动态图_使用Python、Geopandas和Matplotlib制作gif动态

原标题&#xff1a;使用Python、Geopandas和Matplotlib制作gif动态 不需要Photoshop&#xff1a;仅使用Python和命令行制作动画图表。作为一种编程语言&#xff0c;Python非常灵活。这使得有时可以在几行代码中进行大量不同的可视化成为可能。但是现在有了很多不同的图表网站和…

dst发育筛查有意义吗_儿童视力筛查,都筛些啥?

感谢眼科小伙伴们的早出晚归辛勤忙碌&#xff0c;感谢幼儿园、学校及小朋友们的积极配合&#xff0c;本学期的校园视力筛查也已进入尾声。相信爸爸妈妈们都已拿到了这样的筛查结果单&#xff1a;&#xff08;学龄前儿童仅行电脑验光筛查&#xff0c;学龄儿童还需行视力检查&…

java 堆_JAVA学习笔记 07——堆、栈和垃圾回收机制

程序执行的内存分析过程&#xff1a;虚拟机栈(简称&#xff1a;栈)的特点如下&#xff1a;1. 栈描述的是方法执行的内存模型。每个方法被调用都会创建一个栈帧(存储局部变量、操作数、方法出口等)。2. JVM为每个线程创建一个栈&#xff0c;用于存放该线程执行方法的信息(实际参…

dubbo 自定义路由_Dubbo分层架构概述

本节我们从整体上来看看 Dubbo 的分层架构设计&#xff0c;架构分层是一个比较经典的模式&#xff0c;比如网络中的7层协议&#xff0c;每层执行固定的功能&#xff0c;上层依赖下层提供的功能&#xff0c;下层对上层提供功能&#xff0c;下层的改变对上层不可见&#xff0c;并…

python类方法继承_python类的继承

一、概述 面向对象编程 (OOP) 语言的一个主要功能就是“继承”。继承是指这样一种能力&#xff1a;它可以使用现有类的所有功能&#xff0c;并在无需重新编写原来的类的情况下对这些功能进行扩展。 通过继承创建的新类称为“子类”或“派生类”&#xff0c;被继承的类称为“基类…

添加编译宏_软件开发——编译链接

对于平常的应用程序开发&#xff0c;我们很少需要关注编译和链接过程。我们平常Xcode开发就是集成的的开发环境(IDE),这样的IDE一般都将编译和链接的过程一步完成&#xff0c;通常将这种编译和链接合并在一起的过程称为构建&#xff0c;即使使用命令行来编译一个源代码文件,简单…

shell打开wifi命令_远程执行命令的填坑记录

前言本文主要记录 bash 四种模式的细节&#xff0c;以便于遇到问题时查阅。远程执行出错了最近使用 ansible 比较多&#xff0c;在某次使用 shell 模块远程执行命令的时候老是报 ‘command not found’。但是手动登录到远端机器执行命令是成功的&#xff0c;于是开始思考里面的…

nodejs express use 传值_再也不怕面试官问你express和koa的区别了

前言用了那么多年的express.js&#xff0c;终于有时间来深入学习express&#xff0c;然后顺便再和koa2的实现方式对比一下。老实说&#xff0c;还没看express.js源码之前&#xff0c;一直觉得express.js还是很不错的&#xff0c;无论从api设计&#xff0c;还是使用上都是可以的…

没有工作经验找it_校招和社招有什么区别?没有工作经验,如何找工作?

我们的招聘一般分为社招和校招。社招就是社会上招聘&#xff0c;绝大多数都要有相关的工作经验。而校招则是针对在校大学生的&#xff0c;所以应届毕业生的身份很重要&#xff0c;失去了这个身份&#xff0c;就要进入社招的渠道&#xff0c;面临的压力和要求都要高得多。那么&a…

python输入个人所得税计算_个税起征点上调至5000,用Python算一算少交多少税

原标题&#xff1a;个税起征点上调至5000&#xff0c;用Python算一算少交多少税今天出了一个重磅消息&#xff0c;个税起征点从3500上调到5000啦&#xff01; 广大IT农民工的生活压力又减轻了一些&#xff0c;有没有 晚上加一个鸡腿&#xff0c;要不要~ 开心归开心&#xff0c;…

hive 行转列和列转行的方法_面试常考!SQL行转列和列转行

关注上方“数据挖掘工程师”&#xff0c;选择星标&#xff0c;关键时间&#xff0c;第一时间送达&#xff01;行转列&#xff0c;列转行是我们在开发过程中经常碰到的问题。行转列一般通过CASE WHEN 语句来实现&#xff0c;也可以通过 SQL SERVER 的运算符PIVOT来实现。用传统的…

.net core 装了没生效_王者荣耀:辅助装出现惊天bug?辅助光环对自己无效?

大家好&#xff0c;我是老刘备。今天说的这个bug我已经发现许久&#xff0c;从新版本以来就一直在出现&#xff0c;起初我以为是辅助装的光环改动了&#xff0c;但是最近和我的小伙伴开黑之后也遇到这个bug&#xff0c;才发现不是如此。本期我就来带大家一起看看。辅助光环技我…

python sns绘制回归线_Python数分实战:员工流失情况预测

在很久之前&#xff0c;我有写一个Excel数据分析的实战项目&#xff0c;不晓得大家还记不记得&#xff0c;感兴趣的童鞋可以回看&#xff1a;A九姑娘&#xff1a;Excel数分实战&#xff1a;员工流失率分析​zhuanlan.zhihu.com本次的项目数据依旧是这个&#xff0c;但是我们这次…

go var 一个整数_Go语言学习基础-值、变量、常量

Hello world第一个程序是打印经典的“hello world”信息。保存为hello-world.go文件&#xff0c;在命令行中使用go run命令运行程序。若要构建为二进制文件&#xff0c;则使用go build来实现&#xff0c;然后直接执行二进制文件。结果&#xff1a;$ go run hello-world.gohello…

怎么取消自动格式化_iPhone自动扣费怎么取消?App Store、微信、支付宝关闭自动扣费教程...

经常有网友反馈&#xff0c;自己的手机每月会被不知不觉的被扣钱&#xff0c;但又不知道如何关闭&#xff0c;烦恼不已。其实&#xff0c;现在很多视频、音乐、购物等APP都有会员充值功能&#xff0c;其中不少还包含“自动续费”功能&#xff0c;一旦开启会员&#xff0c;今后就…

c# selenium chrome 文件下载_Selenium 框架原理以及环境安装

Selenium 框架原理图片来自转载 蓝色的图Selenium Language Binding就是我们平时使用的编码语言&#xff0c;包括java、python、C#等。可以看到这些语言并没有直接与Browser Drivers进行通信&#xff0c;而是通过了JSON WireProtocol&#xff0c;这就是webdriver协议 &#xff…

怎么打包图片_怎么将许多张照片打包发到邮箱?

怎么将许多张照片打包发到邮箱&#xff1f;所谓的打包&#xff0c;就是需要你把大量照片文件变成一个压缩文件&#xff0c;类似于放到一个文件夹内&#xff0c;也可以理解为把多张纸质照片装订成一个相册&#xff0c;而在装订过程中需要相应工具才能完成。使用压缩包发送文件&a…

dram sram利用 利用_使用量子力学技术的新型超低功耗存储器或将取代DRAM和Flash...

雷锋网按&#xff1a;业界普遍认为未来从数据中将能挖掘出最大的价值&#xff0c;但要挖掘数据的价值除了需要很强的计算能力之外&#xff0c;数据的存储也非常关键。目前&#xff0c;新型存储器也是领先的企业非常关注的一个方向&#xff0c;兰开斯特大学(Lancaster Universit…

mac地址转换_mac电脑格式转化工具

软件下载【软件名称】&#xff1a;Permute【系统支持】&#xff1a;MAC电脑【位数支持】&#xff1a;32位/64位软件下载地址&#xff1a;链接&#xff1a;https://pan.baidu.com/s/1OAEpnvmkQgqd9MMq3Rucow提取码&#xff1a;tgt7注&#xff1a;阿狸提供用户下载的所有内容均来…

电脑怎么打出冒号符号_Mac小技巧:教你如何在Mac电脑打出command?、option?等特殊符号...

很多Mac新用户不知道该如何打出command⌘、option⌥等特殊符号&#xff0c;而网上好多关于打出Mac特殊符号的教程已经过期&#xff0c;或者繁琐看不懂。今天为大家分享一份简单&#xff0c;一看就懂的Mac电脑怎么打出command⌘、option⌥等特殊符号教程。1.我们打开系统偏好设置…