以alpine作為基礎鏡像構建Golang可執行程序操作
Alpine介紹
Alpine 操作系統是一個面向安全的輕型 Linux 發行版。它不同於通常 Linux 發行版,Alpine 采用瞭 musl libc 和 busybox 以減小系統的體積和運行時資源消耗,但功能上比 busybox 又完善的多,因此得到開源社區越來越多的青睞。在保持瘦身的同時,Alpine 還提供瞭自己的包管理工具 apk,可以通過 https://pkgs.alpinelinux.org/packages 網站上查詢包信息,也可以直接通過 apk 命令直接查詢和安裝各種軟件。
Alpine 由非商業組織維護的,支持廣泛場景的 Linux發行版,它特別為資深/重度Linux用戶而優化,關註安全,性能和資源效能。Alpine 鏡像可以適用於更多常用場景,並且是一個優秀的可以適用於生產的基礎系統/環境。
Alpine Docker 鏡像也繼承瞭 Alpine Linux 發行版的這些優勢。相比於其他 Docker 鏡像,它的容量非常小,僅僅隻有 5 MB 左右(對比 Ubuntu 系列鏡像接近 200 MB),且擁有非常友好的包管理機制。官方鏡像來自 docker-alpine 項目。
目前 Docker 官方已開始推薦使用 Alpine 替代之前的 Ubuntu 做為基礎鏡像環境。這樣會帶來多個好處。包括鏡像下載速度加快,鏡像安全性提高,主機之間的切換更方便,占用更少磁盤空間等。
下表是官方鏡像的大小比較:
REPOSITORY TAG IMAGE ID VIRTUAL SIZE alpine latest 4e38e38c8ce0 4.799 MB debian latest 4d6ce913b130 84.98 MB ubuntu latest b39b81afc8ca 188.3 MB centos latest 8efe422e6104 210 MB
Alpine和其他通用Linux發行版對於Golang編譯出來的可執行文件要求有所不同,Alpine要求可執行文件必須是靜態鏈接的可執行文件。
所以在編譯Golang時需要添加 -tags netgo ,來生成靜態鏈接的可執行文件。
示例:
代碼示例:每三秒控制臺輸出一次消息,輸出100次
package main import ( "fmt" "time" ) func main() { for i := 0 ; i< 100 ; i++ { fmt.Println("hello,",i) time.Sleep(time.Second*3) } }
編譯成靜態鏈接的可執行文件:
go build -tags netgo -o test
Dockerfile的參考內容:
FROM alpine:latest ADD test /usr/local/bin/ # Alpine Linux doesn't use pam, which means that there is no /etc/nsswitch.conf, # but Golang relies on /etc/nsswitch.conf to check the order of DNS resolving # (see https://github.com/golang/go/commit/9dee7771f561cf6aee081c0af6658cc81fac3918) # To fix this we just create /etc/nsswitch.conf and add the following line: RUN echo 'hosts: files mdns4_minimal [NOTFOUND=return] dns mdns4' >> /etc/nsswitch.conf WORKDIR /usr/local/bin/ CMD ["test"]
構建鏡像、運行容器即可
補充:編譯後的go程序無法在alpine基礎鏡像創建的容器運行問題
問題描述:
我要使用容器創建一個使用golang的http的服務,為瞭不讓鏡像太大,所以選擇瞭基礎鏡像alpine,才幾m,但把go的程序編譯之後無法在容器中運行,解決辦法是在編譯go程序時,加上參數-tags netgo。
具體做法如下:
1、首先下載apline基礎鏡像:
docker pull docker.io/alpine
2、編譯go程序:(我的程序是sensor.go,編譯之後放在sensors目錄下)
go build -tags netgo -o sensors/sensor sensor.go
3、dockerfile文件,保存為sensor:
FROM docker.io/alpine:latest
MAINTAINER weinengl
# 拷貝相關文件。
COPY sensors/sensor /tmp/
RUN chmod +x /tmp/sensor
ENTRYPOINT [“/tmp/sensor”]
4、鏡像制作命令:
docker build -t reg.netlab.com/temperature:v1 -f sensor .
制作完成之後運行即可:
docker run -itd -p 8881:8881 –name test10 reg.netlab.com/temperature:v1
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。
推薦閱讀:
- 使用Dockerfile腳本定制鏡像的方法
- 使用Docker鏡像構建Go應用的實現方法
- 利用Dockerfile文件部署php項目的全過程
- 優化Docker鏡像安全性的12個技巧總結
- 多階段構建優化Go 程序Docker鏡像