制作Go程序的Docker容器(以及容器和主机的网络问题)
今天突然遇到需要将 Go 程序制作成 Docker 的需求,所以进行了一些研究。方法很简单,但是官方文档和教程有些需要注意的地方,所以写本文进行记录。
源程序
首先介绍一下示例程序,示例程序是一个 HTTP 服务器,会显示sin(r)/r
的图像,如下:
新建一个目录draw-surface
,然后在里面新建一个draw-surface.go
文件,内容为:
// display Animated Lissajous in a browser and can set arguments in queries.
package mainimport ("errors""fmt""io""log""math""net/http""strconv""sync"
)var mu sync.Mutex
var count intfunc main() {http.HandleFunc("/", handler) log.Fatal(http.ListenAndServe("localhost:8000", nil))
}func handler(w http.ResponseWriter, r *http.Request) {w.Header().Set("Content-Type", "image/svg+xml")func2svg(w, r)
}var width, height int = 500, 400
var cells int = 200
var xyrange int = 50.0
var xyscale int = width / 2.0 / xyrange
var zscale float64 = float64(height) * 0.4
var angle float64 = math.Pi / 9 var sin, cos float64 = math.Sin(angle), math.Cos(angle)func func2svg(out io.Writer, r *http.Request) {fmt.Fprintf(out, "<svg xmlns='http://www.w3.org/2000/svg' "+"style='stroke: grey; fill: white; stroke-width: 0.7' "+"width='%d' height='%d'>", width, height)for i := 0; i < cells; i++ {for j := 0; j < cells; j++ {ax, ay, error1 := corner(i+1, j)bx, by, error2 := corner(i, j)cx, cy, error3 := corner(i, j+1)dx, dy, error4 := corner(i+1, j+1)if error1 == nil || error2 == nil || error3 == nil || error4 == nil {fmt.Fprintf(out, "<polygon points='%g,%g %g,%g %g,%g %g,%g'/>\n",ax, ay, bx, by, cx, cy, dx, dy)}}}fmt.Fprintf(out, "</svg>")
}func corner(i, j int) (float64, float64, error) {x := float64(xyrange) * (float64(i)/float64(cells) - 0.5)y := float64(xyrange) * (float64(j)/float64(cells) - 0.5)z := f(x, y)if math.IsInf(z, 1) {return math.NaN(), math.NaN(), errors.New("Result of f is non-finite")} else {sx := float64(width/2) + (x-y)*cos*float64(xyscale)sy := float64(height/2) + (x+y)*sin*float64(xyscale) - z*zscalereturn sx, sy, nil}
}func f(x, y float64) float64 {r := math.Hypot(x, y)return math.Sin(r) / r
}
然后是用以下命令初始化模块:
$ go mod init
这个程序来自于《The Go Programming language》,本文重点看main
函数即可。由于是用来演示构建映像和容器,所以删除了 URL 参数部分。
不在容器里运行(本机)
这个程序可以直接在本机运行,此时main
函数内容如下,地址设置为了localhost:8000
:
func main() {http.HandleFunc("/", handler)log.Fatal(http.ListenAndServe("localhost:8000", nil))
}
可以直接在终端运行:
go run draw-surface.go
然后在浏览器中使用localhost:8000
就可以看到前面的示例图的内容。
在容器中运行
构建Docker映像
官方文档很奇怪,Containerize your application 中介绍说用docker init
来新建所需的文件,但是 Mac 上的 Docker(20.10.23)没有这个命令:
$ docker init
docker: 'init' is not a docker command.
See 'docker --help'
但是在介绍构建 Go 映像的教程 Build your Go image 里介绍的是手动新建。
所以这里使用手动新建 Docker 映像所需的配置文件Dockerfile
:
$ touch Dockerfile
然后以此输入下面的内容,解释请看注释:
# syntax=docker/dockerfile:1
# 继承自 Go 1.20 的映像,这样就可以使用Go的编译器等。把容器必做一台设备的话,就是在这个设备上安装了 Go
FROM golang:1.20# 工作目录为/app,这里的/是容器内部的根目录
WORKDIR /app
#将当前目录下的go.mod复制到容器内的工作目录,也就是/app下
COPY go.mod ./
# 使用命令安装所需的模块
RUN go mod download
#将当前目录下所有的go源代码文件复制到容器内的工作目录,也就是/app下
COPY *.go ./# 构建 Go 程序 draw-surface到根目录下
RUN CGO_ENABLED=0 GOOS=linux go build -o /draw-surface# 将该映像当做容器启动之后,执行该程序
CMD [ "/draw-surface" ]
然后需要修改一下 Go 源代码文件draw-surface.go
中的一个地方,将地址中的localhost
删除,只留下:8000
。如下:
func main() {http.HandleFunc("/", handler)log.Fatal(http.ListenAndServe(":8000", nil))
}
为什么要这样修改呢?要解释这个问题有点偏题和涉及后面的内容了,所以放在最后的“题外话”部分。
接下来就可以进行构建映像了,命令如下:
$ docker build --tag draw-surface .
[+] Building 16.6s (15/15) FINISHED => [internal] load build definition from Dockerfile 0.0s=> => transferring dockerfile: 220B 0.0s=> [internal] load .dockerignore 0.0s=> => transferring context: 2B 0.0s=> resolve image config for docker.io/docker/dockerfile:1 2.2s=> CACHED docker-image://docker.io/docker/dockerfile:1@sha256:ac85f380a6 0.0s=> [internal] load build definition from Dockerfile 0.0s=> [internal] load metadata for docker.io/library/golang:1.20 1.6s=> [internal] load .dockerignore 0.0s=> [1/6] FROM docker.io/library/golang:1.20@sha256:77e4e426190723821471a 0.0s=> [internal] load build context 0.0s=> => transferring context: 3.50kB 0.0s=> CACHED [2/6] WORKDIR /app 0.0s=> CACHED [3/6] COPY go.mod ./ 0.0s=> CACHED [4/6] RUN go mod download 0.0s=> [5/6] COPY *.go ./ 0.0s=> [6/6] RUN CGO_ENABLED=0 GOOS=linux go build -o /draw-surface 12.0s=> exporting to image 0.5s=> => exporting layers 0.5s=> => writing image sha256:d3e0c9b2bc8fb859f189ef7a51f7a478cf61f9d8a3e0c 0.0s=> => naming to docker.io/library/draw-surface 0.0s
使用映像运行容器
然后就可以使用映像运行一个容器了,使用以下命令运行:
docker run --publish 8000:8000 draw-surface
这里的8000:8000
是容器内外端口的映射,和代码中一样即可。如果你想使用 Docker Desktop 运行容器,那么端口映射需要写到 Dockerfile 中,然后才可以在第一次运行该映像的时候设置对应端口。
使用的话在浏览器中输入:http://localhost:8000
即可看到示例图:
除了http://localhost:8000
,还可以使用http://HostIp的内容:8000
和http://0.0.0.0:8000
,但是不能使用http://IPAddress:8000
来获取服务器返回的图像。关于如何获取容器HostIp
和IPAddress
,在下面的“如何获取 Docker 容器的 IP 地址”有详细说明。
题外话
如何获取 Docker 容器的 IP 地址和主机地址
获取 Docker 容器的 IP 地址很简单。
Docker Desktop
如果使用 Docker Desktop,那么在容器部分查看详细信息,然后最下面就是。如下:
Docker CLI
如果是在终端中使用 Docker CLI,那么首先使用docker ps
找到你想查询的容器的 ID,然后使用以下命令查看这个容器的详细信息:
$ docker inspect 容器ID
这里列出了很多信息,但是本文中我们需要的是HostIp
,而不是IPAddress
部分。你可以使用http://HostIp:8000
、http://0.0.0.0:8000
和http://localhost:8000
,唯独不能使用http://IPAddress:8000
获取服务器返回的图像(强调,只是本文的情况是这样,其他的项目需要根据情况而定)。
所以如果你想获取HostIp
,那么可以将上面的命令修改成:
$ docker inspect 容器ID | grep "HostIp""HostIp": "","HostIp": "0.0.0.0",
如果你想获取IPAddress
,那么使用下面的命令比较方便:
$ docker inspect 容器ID | grep "IPAddress""SecondaryIPAddresses": null,"IPAddress": "172.17.0.2","IPAddress": "172.17.0.2",
为什么使用:8000
格式作为地址(使用静态IP行不行)
Docker 是一种虚拟技术,致力于用最小系统环境模拟单台设备。如果在本地网络上的一台设备上使用了localhost:8000
这样的地址,而这个端口也对外开放。那么在本地网络上的其他设备中,在浏览器中使用http://服务器IP:8000
这样的 URL 也无法看到图。只能在代码设置地址的时候,使用:8000
说明端口或服务器IP:8000
说明网络中的地址,然后访问时使用服务器IP:8000
这样的 URL 就可以看到图了。不过一般考虑到移植问题,后者使用的少。
但是 Docker 容器虽然工作起来像这样,但不是完全符合,或者说默认情况下不是这样的。
Docker 容器不能使用服务器IP:8000
声明和访问。因为在运行映像的时候,使用的8000:8000
表示的是将主机的8000
端口和容器的8000
端口映射,但是容器并不是在主机的网络上,而是在 Docker 网络上(也就是在主机内部,有一个守护进程为各个容器分配 IP)。也就是说,主机就是一个Host
,这些容器把端口映射到Host
的端口上了。
这也解释了为什么这里无法使用容器的 IP 地址获取图像,因为主机的网络端口根本找不到他。但却可以使用HostIp
的地址来获取图像,因为Host
就是主机,是一台机器。
此外,不光主机本地地址http://localhost:8000
可以,你还可以通过主机的 IP 来使用,也就是某个网络接口的 IP 地址,比如无线接口的IP:http://169.252.1.4:8000
。
参考阅读
IP address, Network address, and Host address Explained
Servers - Go
Networking overview - Docker Docs
Dockerfile reference - Docker Docs
How to get a Docker container’s IP address from the host - stackoverflow
希望能帮到有需要的人~
相关文章:

制作Go程序的Docker容器(以及容器和主机的网络问题)
今天突然遇到需要将 Go 程序制作成 Docker 的需求,所以进行了一些研究。方法很简单,但是官方文档和教程有些需要注意的地方,所以写本文进行记录。 源程序 首先介绍一下示例程序,示例程序是一个 HTTP 服务器,会显示si…...

mysql清除数据痕迹_MySQL使用痕迹清理~/.mysql_history - milantgh
mysql会给出我们最近执行的SQL命令和脚本;同linux command保存在~/.bash_history一样,你用mysql连接MySQL server的所有操作也会被记录到~/.mysql_history文件中,这样就会有很大的安全风险了,如添加MySQL用户的sql也同样会被明文记…...

PDF控件Spire.PDF for .NET【转换】演示:自定义宽度、高度将 PDF 转 SVG
我们在上一篇文章中演示了如何将 PDF 页面转换为 SVG 文件格式。本指南向您展示如何使用最新版本的 Spire.PDF 以及 C# 和 VB.NET 指定输出文件的宽度和高度。 Spire.Doc 是一款专门对 Word 文档进行操作的 类库。在于帮助开发人员无需安装 Microsoft Word情况下,轻…...

01背包 P1507 NASA的食物计划
P1507 NASA的食物计划 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 普通01背包状态表示:f(i, j)表示前i件物品放入一个容量为j的背包可以获得的最大价值。 本题类似,f(i, j, k)表示前i件物品放入一个限制为j,且另一个限制为k的背包中可以…...

平衡二叉树c语言版
一、定义二叉树结点结构体 /*** 定义平衡二叉树结点 */ struct avlbinarytree { //数据域NodeData* data;///树高int h;struct avlbinarytree* left;struct avlbinarytree* right; }; typedef struct avlbinarytree AVLNode; 二、声明函数的操作 /*** 创建结点 */ AV…...

初始环境配置
目录 一、JDK1、简介2、配置步骤 二、Redis1、简介2、配置步骤 三、MySQL1、简介2、配置步骤 四、Git1、简介2、配置步骤 五、NodeJS1、简介2、配置步骤 六、Maven1、简介2、配置步骤 七、Tomcat1、简介2、配置步骤 一、JDK 1、简介 JDK 是 Oracle 提供的 Java 开发工具包&…...

记GitLab服务器迁移后SSH访问无法生效的问题解决过程
公司IT心血来潮对GitLab服务器进行安全升级,升级后无法启动。。。只得启用备用服务器,具体的备份机制不祥,只知道原数据都在,但文件系统是否完全一样不清楚。切换为备用服务器后使用SSH下载代码死活不成功,反复提示需要…...

【NGINX--2】高性能负载均衡
1、HTTP 负载均衡 将负载分发到两台或多台 HTTP 服务器。 在 NGINX 的 HTTP 模块内使用 upstream 代码块对 HTTP 服务器实施负载均衡: upstream backend {server 10.10.12.45:80 weight1;server app.example.com:80 weight2;server spare.example.com:80 backup; …...

Android studio run 手机或者模拟器安装失败,但是生成了debug.apk
错误信息如下:Error Installation did not succeed. The application could not be installed:List of apks 出现中文乱码; 我首先尝试了打包,能正常安装,再次尝试了debug的安装包,也正常安装࿱…...

【面试经典150 | 数学】加一
文章目录 写在前面Tag题目来源解题思路方法一:加一 其他语言python3 写在最后 写在前面 本专栏专注于分析与讲解【面试经典150】算法,两到三天更新一篇文章,欢迎催更…… 专栏内容以分析题目为主,并附带一些对于本题涉及到的数据结…...

Rust unix domain socket
先用起来再说 use std::io::prelude::*; use std::os::unix::net::UnixStream;fn main() {let mut stream: UnixStream;let mut buffer vec![0u8; 4096];match UnixStream::connect("/tmp/hello.world.serv") {Ok(handle) > {stream handle;match stream.write_…...

初识分布式键值对存储etcd
欢迎大家到我的博客浏览。胤凯 (oyto.github.io)大家好,今天我带大家来学习一下 etcd。 一、什么是 etcd etcd 是一个开源的分布式键值存储系统,主要用于构建分布式系统中那点服务发现、配置管理、分布式锁等场景。它采用 Raft 一致性算法来确保所有节…...

docker swarm集群部署
文章目录 前言一、安装docker1.1 解压1.2 配置docker 存储目录和dns1.3 添加docker.service文件1.4 docker 启动验证 二、docker swarm 集群配置2.1 关闭selinux2.2 设置主机名称并加入/etc/hosts2.3 修改各个服务器名称(uname -a 进行验证)2.4 初始化sw…...

MySQL进阶_9.事务基础知识
文章目录 第一节、数据库事务概述1.1、基本概念1.2、事务的ACID特性 第二节、如何使用事务 第一节、数据库事务概述 1.1、基本概念 事务 一组逻辑操作单元,使数据从一种状态变换到另一种状态。事务处理的原则 保证所有事务都作为 一个工作单元 来执行,…...

IDEA调用接口超时,但Postman可成功调用接口
📢专注于分享软件测试干货内容,欢迎点赞 👍 收藏 ⭐留言 📝 如有错误敬请指正!📢交流讨论:欢迎加入我们一起学习!📢资源分享:耗时200小时精选的「软件测试」资…...

TableUtilCache:针对CSV表格进行的缓存
TableUtilCache:针对CSV表格进行的缓存 文件结构 首先来看下CSV文件的结构,如下图: 第一行是字段类型,第二行是字段名字;再往下是数据。每个元素之间都是使用逗号分隔。 看一下缓存里面存储所有表数据的字段 如下图ÿ…...

java源码-工程讲解
说明: 源码工程目录讲解部分,讲解过程会让大家对后端源码工程有一个大致的了解,能让大家在此改造,就可以衍生出一些新的功能,需要对java技术深入了解,需要看后续java技术讲解部分,源码也是以前很…...

K8S基础笔记
1、namespace 名称空间用来对集群资源进行隔离划分,默认只隔离资源,不隔离网络k8s默认的名称空间为default 查看k8s的所有命名空间 kubectl get namespace 或者 kubectl get ns 创建名称空间 kubectl create ns 名称 或使用yaml方式 编写yamlkub…...

十一、统一网关GateWay(搭建网关、过滤器、跨越解决)
目录 一、网关技术的实现 在SpringCloud中网关的实现包括两种: 作用: 二、搭建网关服务 1、新建模块,并添加依赖 2、新建Gateway包,并编写启动类 3、编写yml文件 4、启动服务,并在网页内测试 5、步骤 三、路由断言工厂 …...

C语言--每日五道选择题--Day20
第一题 1. 在如下结构定义中,不正确的是( )。 A: struct student { int no; char name[10]; float score; }; B: struct stud[20] { int no; char name[10]; float score; }; C: struct stu…...
Fourier分析导论——第6章——R^d 上的Fourier变换(E.M. Stein R. Shakarchi)
第6章 上的 Fourier 变换 It occurred to me that in order to improve treatment planning one had to know the distribution of the at- tenuation coefficient of tissues in the body. This in- formation would be useful for diagnostic purposes and would con…...

音视频技术在手机上的应用与挑战
// 编者按:随着手机相机功能日益强大,4k,8k,各类特色短视频的拍摄,编辑、播放需求日益增长,短视频应用的火爆也对当前的手机音视频技术提出了更高的要求,如何更好地提高用户体验成为了行业共同…...

三十分钟学会SCALA
SCALA Scala 是一种运行在 JVM上的函数式的面向对象语言。 Scala 是兼容的:兼容 Java,可以访问庞大的 Java 类库;Scala 是精简的:Scala 表达能力强,一行代码抵得上多行 Java 代码,开发速度快。可以让程序…...

leetcode做题笔记242. 有效的字母异位词
给定两个字符串 s 和 t ,编写一个函数来判断 t 是否是 s 的字母异位词。 注意:若 s 和 t 中每个字符出现的次数都相同,则称 s 和 t 互为字母异位词。 示例 1: 输入: s "anagram", t "nagaram" 输出: true示例 2: 输…...

沸点 | Ultipa 图数据库金融应用场景优秀案例首批入选,金融街论坛年会发布
为推进图数据库在金融行业的创新应用试点,近日,在2023金融街论坛年会“全球金融科技中心网络年会暨ZIBS北京论坛”上,北京前沿金融监管科技研究院发布了基于国际标准组织——国际关联数据基准委员会(LDBC)的《图数据库…...

GaussDB SQL基础语法示例-GOTO语句
目录 一、前言 二、在GaussDB数据库中的概念及语法 1、基本概念 2、语法 三、在GaussDB数据库中的基础示例和限制场景说明 1、基础示例 2、限制场景说明 四、小结 一、前言 SQL是用于访问和处理数据库的标准计算机语言。GaussDB支持SQL标准(默认支持SQL2、…...

ClickHouse 物化视图
ClickHouse的物化视图是一种查询结果的持久化,它确实是给我们带来了查询效率的提升。用户查起来跟表没有区别,它就是一张表,它也像是一张时刻在预计算的表,创建的过程它是用了一个特殊引擎,加上后来 as select…...

原理Redis-ZipList
ZipList 1) ZipList的组成2) ZipList的连锁更新问题3) 总结 1) ZipList的组成 ZipList 是一种特殊的“双端链表” ,由一系列特殊编码的连续内存块组成。可以在任意一端进行压入/弹出操作, 并且该操作的时间复杂度为 O(1)。 ZipListEntry: ZipList 中的Entry并不像…...

小迪安全笔记——Web架构篇语言中间件数据库系统源码获取
1、信息搜集搜集哪些东西? 架构信息收集,主要包括:操作系统、开发语言、中间件容器、数据库类型、第三方软件等; web源码信息收集,CMS开源?闭源?售卖?自主研发? 进行web…...

Linux从 全栈开发 centOS 7 到 运维
Linux从 全栈开发centOS 7 到 运维 一 Linux 入门概述1.1 操作系统1.2 Linux 简介1.3 Linux 系统组成1.4 Linux 发行版1.4 Linux 应用领域1.5 Linux vs Windows 二 环境搭建【狂神说Java】服务器购买及宝塔部署环境说明为什么程序员都需要一个自己的服务器服务器如何购买买完服…...