草庐IT

自定义镜像上传阿里云

南山 有谷堆 2023-11-19 原文

目录

目标:

一、alpine Linux简介

二、alpine制作jdk8镜像

jdkv.1.0的制作

jdkv.2.0的制作

jdkv.3.0的制作

三、Doker镜像上传至阿里云


目标:

  1. alpine制作jdk镜像
  2. Alpine制作jre镜像(瘦身)
  3. Docker镜像上传至阿里云

一、alpine Linux简介

        1.Alpine Linux是一个轻型Linux发行版,它不同于通常的Linux发行版,Alpine采用了musl libc 和 BusyBox以减少系统的体积和运行时的资源消耗。


        2.Alpine Linux提供了自己的包管理工具:apk(注意:ubuntu中是apt-get),我们可以通过https://pkgs.alpinelinux.org/packages 查询包信息


        3.Alpine Docker镜像继承了Alpine Linux发行版的这些优势,相比于其他Linux Docker镜像,它的体积非常小
             对比常用的、没有压缩过的基础镜像(查看当前的:latest标签):
             Alpine - 4.8MB
             centos - 124.8 MB
             Debian - 125.1MB
             Centos - 196MB
        4.建议使用Alpine Linux 3.10.0版本,这也是 v3.10 稳定系列的首个版本
             alpine:3.10 

二、alpine制作jdk8镜像

docker search jdk               搜索镜像

发现官方下载最多的java那些都不是我们想要的,我们得自己做一个

jdkv.1.0的制作

1.创建并编辑dockerfile文件

2.在Dockerfile添加追踪jdk镜像的命令

        指定基础镜像        centos:7

        在centos7中添加jdk的安装包

        解压安装包

        配置path

        java -version

1.创建编辑dockerfile文件 

vi Dockerfile        创建Dockerfile文件并进行编辑

将下面内容复制进去就行        里面的 javaxl 可以改成自己喜欢的名字

#1.指定基础镜像,并且必须是第一条指令

RROM centos:7

#2.指明该镜像的作者和其电子邮件

MAINTAINER zs "zs@qq.com"

#3.在构建镜像时,指定镜像的工作目录,之后的命令都是基于此工作目录,如果不存在,则会创建目录

WORKDIR /javaxl_docker/jdk

#4.一个复制命令,把jdk安装文件复制到镜像中,语法:ADD <src>... <dest>,注意:jdk*.tar.gz使用的是相对路径

ADD jdk-8u221-linux-x64.tar.gz /javaxl_docker/jdk/

#5.配置环境变量

ENV JAVA_HOME=/javaxl_docker/jdk/jdk1.8.0_221

ENV CLASSPATH=.:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar

ENV PATH=$JAVA_HOME/bin:$PATH

#容器启动时需要执行的命令

#CMD ["java","-version"]

我们仔细看里面的内容的话是可以发现里面是需要添加压缩包的

所以我们将这几个压缩包放到dockerfile同级

 

2.执行dockerfile创建镜像

docker build -t jdk8:v1.0 .

注1:-t  jdk8:v1.0  给新构建的镜像取名为 jdk8, 并设定版本为 v1.0

注2:注意最后有个点,代表使用当前路径的 Dockerfile 进行构建

3.查看镜像 

docker images        查看镜像

4.创建并启动容器

 docker run -it jdk8:v1.0 /bin/bash

5.测试jdk

java -version

 dockerfile实践经验

   1. 精简镜像用途:尽量让每个镜像的用途都比较集中单一,避免构造大而复杂、多功能的镜像

   2. 选用合适的基础镜像:容器的核心是应用,选择过大的父镜像(如Ubuntu系统镜像)会造成最终生成应用镜像的膝肿,

       推荐选用瘦身过的应用镜像或者较为小巧的系统镜像(alpine)

jdkv.2.0的制作

因为1.0版本还是比较大所用到的是jdk,jdk所包含的有编译和运行环境,而且里面还有很多东西是我们不要用到的,我们可以使用本身就不大的jre,所以我们还可以优化,步骤跟1.0的差不多

第二个版本采用的是我们的基础镜像 aIpine,所要用到的是glibc的一个依赖包,glibc安装包如果从网络下载速度实在是太慢了,先提前下载复制到镜像中

这里博主所用的是2.29的版本,将其粘贴到dockerfile同级文件夹中即可

 

 然后又是vi Dockerfile        将下面的内容复制进去即可(里面的javaxl是可以改的,Dockerfile首字母要大写         创建的时候不要搞错了!)

#1.指定基础镜像,并且必须是第一条指令
FROM alpine:latest
#FROM alpine:3.10

#2.指明该镜像的作者和其电子邮件
MAINTAINER xyz "xyz@qq.com"

#3.在构建镜像时,指定镜像的工作目录,之后的命令都是基于此工作目录,如果不存在,则会创建目录
WORKDIR /javaxl_docker/jdk

#4.将一些安装包复制到镜像中,语法:ADD/COPY <src>... <dest>
## ADD与COPY的区别:ADD复制并解压,COPY仅复制
ADD jdk-8u221-linux-x64.tar.gz /javaxl_docker/jdk/
## glibc安装包如果从网络下载速度实在是太慢了,先提前下载复制到镜像中
COPY glibc-2.29-r0.apk /javaxl_docker/jdk/
COPY glibc-bin-2.29-r0.apk /javaxl_docker/jdk/
COPY glibc-i18n-2.29-r0.apk /javaxl_docker/jdk/

#5.更新Alpine的软件源为阿里云,因为从默认官源拉取实在太慢了
RUN echo http://mirrors.aliyun.com/alpine/v3.10/main/ > /etc/apk/repositories && \
    echo http://mirrors.aliyun.com/alpine/v3.10/community/ >> /etc/apk/repositories
RUN apk update && apk upgrade

#6.运行指定的命令
## Alpine linux为了精简本身并没有安装太多的常用软件,apk类似于ubuntu的apt-get,
## 用来安装一些常用软V件,其语法如下:apk add bash wget curl git make vim docker
## wget是linux下的ftp/http传输工具,没安装会报错“/bin/sh:   wget: not found”,网上例子少安装wget
## ca-certificates证书服务,是安装glibc前置依赖
RUN apk --no-cache add ca-certificates wget \
    && wget -q -O /etc/apk/keys/sgerrand.rsa.pub https://alpine-pkgs.sgerrand.com/sgerrand.rsa.pub \
    && apk add glibc-2.29-r0.apk glibc-bin-2.29-r0.apk glibc-i18n-2.29-r0.apk \
    && rm -rf /var/cache/apk/* glibc-2.29-r0.apk glibc-bin-2.29-r0.apk glibc-i18n-2.29-r0.apk

#7.配置环境变量
ENV JAVA_HOME=/javaxl_docker/jdk/jdk1.8.0_221
ENV CLASSPATH=.:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar
ENV PATH=$JAVA_HOME/bin:$PATH

#容器启动时需要执行的命令
#CMD ["java","-version"]

执行dockerfile创建镜像

docker build -t jdk8:v2.0 .

docker images        查看镜像

 

docker run -it --name 别名 镜像ID  /bin/sh 

 

 制作完成后的镜像大小为400M左右

jdkv.3.0的制作

这个时候我们还是觉得太大了 

 这时候我们就要进行用jre来制作一个更小的jdk

 我们先将其扔到镜像里去,在解压jre的时候我们发现,官方提供的这个jre文件夹里面

有很多官方提供的一些文档描述都是没有用的,于是我们得将其删除减少空间

tar -zxvf jre-8u221-linux-x64.tar.gz        解压jre压缩包

du -sh jre1.8.0_221                查看jre大小(瘦身前230M)

进入jre目录,并执行瘦身命令

#删除文本文件
rm -rf COPYRIGHT LICENSE README release THIRDPARTYLICENSEREADME-JAVAFX.txtTHIRDPARTYLICENSEREADME.txt Welcome.html \
#删除其他无用文件
rm -rf lib/plugin.jar \
lib/ext/jfxrt.jar \
bin/javaws \
lib/javaws.jar \
lib/desktop \
plugin \
lib/deploy* \
lib/*javafx* \
lib/*jfx* \
lib/amd64/libdecora_sse.so \
lib/amd64/libprism_*.so \
lib/amd64/libfxplugins.so \
lib/amd64/libglass.so \
lib/amd64/libgstreamer-lite.so \
lib/amd64/libjavafx*.so \
lib/amd64/libjfx*.so

 执行完后再次查看文件大小:

 接着我们将这个文件再压缩成压缩包(记得注意目录)

tar -zcvf jre1.8.0_221.tar.gz jre1.8.0_221                重新打包

将Dockerfile进行备份

接着删除该文件重新创建并编写内容

#1.指定基础镜像,并且必须是第一条指令
FROM alpine:latest
#FROM alpine:3.10

#2.指明该镜像的作者和其电子邮件
MAINTAINER xyz "xyz@qq.com"

#3.在构建镜像时,指定镜像的工作目录,之后的命令都是基于此工作目录,如果不存在,则会创建目录
WORKDIR /javaxl_docker/jdk

#4.将一些安装包复制到镜像中,语法:ADD/COPY <src>... <dest>
## ADD与COPY的区别:ADD复制并解压,COPY仅复制
## 注意~~~上传的瘦身后的jre
ADD jre1.8.0_221.tar.gz /javaxl_docker/jdk/
## glibc安装包如果从网络下载速度实在是太慢了,先提前下载复制到镜像中
COPY glibc-2.29-r0.apk /javaxl_docker/jdk/
COPY glibc-bin-2.29-r0.apk /javaxl_docker/jdk/
COPY glibc-i18n-2.29-r0.apk /javaxl_docker/jdk/

#5.更新Alpine的软件源为阿里云,因为从默认官源拉取实在太慢了
RUN echo http://mirrors.aliyun.com/alpine/v3.10/main/ > /etc/apk/repositories && \
    echo http://mirrors.aliyun.com/alpine/v3.10/community/ >> /etc/apk/repositories
RUN apk update && apk upgrade

#6.运行指定的命令
## Alpine linux为了精简本身并没有安装太多的常用软件,apk类似于ubuntu的apt-get,
## 用来安装一些常用软V件,其语法如下:apk add bash wget curl git make vim docker
## wget是linux下的ftp/http传输工具,没安装会报错“/bin/sh:   wget: not found”,网上例子少安装wget
## ca-certificates证书服务,是安装glibc前置依赖
RUN apk --no-cache add ca-certificates wget \
    && wget -q -O /etc/apk/keys/sgerrand.rsa.pub https://alpine-pkgs.sgerrand.com/sgerrand.rsa.pub \
    && apk add glibc-2.29-r0.apk glibc-bin-2.29-r0.apk glibc-i18n-2.29-r0.apk \
    && rm -rf /var/cache/apk/* glibc-2.29-r0.apk glibc-bin-2.29-r0.apk glibc-i18n-2.29-r0.apk

#7.配置环境变量
## 注意~~~没有jdk啦,直接指向jre
ENV JAVA_HOME=/javaxl_docker/jdk/jre1.8.0_221
ENV CLASSPATH=.:$JAVA_HOME/lib/dt.jar:$JAVA_HOME/lib/tools.jar
ENV PATH=$JAVA_HOME/bin:$PATH

#容器启动时需要执行的命令
#CMD ["java","-version"]

 docker build -t jdk8:v3.0 .        重新创建一个新的镜像

创建完成后用  docker images   查看大小:

三、Doker镜像上传至阿里云

阿里云官方网站链接:开放云原生应用-云原生(Cloud Native)-云原生介绍 - 阿里云

进去后搜索容器镜像

 如果是第一次进去的小伙伴,可能列表就只是实例列表,这时候我们点个人实例就行了

进去后我们得先绑定我们的代码源: 

就是让你设置镜像仓库的密码 ,这里博主绑定的是Codeup

 进去后点新建代码库

输入代码库名称新建就行

 然后点我们的个人设置

 进入个人访问令牌

 

然后复制令牌 

 然后回到阿里云进行代码源的绑定

点击确定,然后我们返回到镜像仓库 

接着我们创建一下镜像仓库

相对应的项目是指codeup里面创建好的代码库 

如何把自己的项目拉到镜像里面呢? 

我们点击仓库名称,进入到基本信息

 我们将操作指南的步骤一的代码copy进去不要$符号

 然后就是将镜像推送到Registry        [ImageId]是指镜像id

docker tag [ImageId] registry.cn-hangzhou.aliyuncs.com/javayuyan/yuyan:[镜像版本号]

镜像版本号可以自己命名方便认出就行 

docker push registry.cn-hangzhou.aliyuncs.com/javayuyan/yuyan:[镜像版本号]

这时候我们的镜像版本里面就有我们刚下好的版本了

  

 拉取后进行删除原先的镜像:

docker rmi -f [镜像id] 

然后再从刚刚拉取的镜像仓库那里下载(注意后面的版本号得和你之前下载好的那个一样)

docker pull registry.cn-hangzhou.aliyuncs.com/javayuyan/yuyan:v5.0

最后喜欢该内容的点点关注和赞,咱们下期再见!

有关自定义镜像上传阿里云的更多相关文章

  1. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  2. ruby-on-rails - Rails 3.2.1 中 ActionMailer 中的未定义方法 'default_content_type=' - 2

    我在我的项目中添加了一个系统来重置用户密码并通过电子邮件将密码发送给他,以防他忘记密码。昨天它运行良好(当我实现它时)。当我今天尝试启动服务器时,出现以下错误。=>BootingWEBrick=>Rails3.2.1applicationstartingindevelopmentonhttp://0.0.0.0:3000=>Callwith-dtodetach=>Ctrl-CtoshutdownserverExiting/Users/vinayshenoy/.rvm/gems/ruby-1.9.3-p0/gems/actionmailer-3.2.1/lib/action_mailer

  3. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

  4. ruby - 主要 :Object when running build from sublime 的未定义方法 `require_relative' - 2

    我已经从我的命令行中获得了一切,所以我可以运行rubymyfile并且它可以正常工作。但是当我尝试从sublime中运行它时,我得到了undefinedmethod`require_relative'formain:Object有人知道我的sublime设置中缺少什么吗?我正在使用OSX并安装了rvm。 最佳答案 或者,您可以只使用“require”,它应该可以正常工作。我认为“require_relative”仅适用于ruby​​1.9+ 关于ruby-主要:Objectwhenrun

  5. ruby - 我可以使用 aws-sdk-ruby 在 AWS S3 上使用事务性文件删除/上传吗? - 2

    我发现ActiveRecord::Base.transaction在复杂方法中非常有效。我想知道是否可以在如下事务中从AWSS3上传/删除文件:S3Object.transactiondo#writeintofiles#raiseanexceptionend引发异常后,每个操作都应在S3上回滚。S3Object这可能吗?? 最佳答案 虽然S3API具有批量删除功能,但它不支持事务,因为每个删除操作都可以独立于其他操作成功/失败。该API不提供任何批量上传功能(通过PUT或POST),因此每个上传操作都是通过一个独立的API调用完成的

  6. ruby - 在 Ruby 中有条件地定义函数 - 2

    我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin

  7. ruby - 定义方法参数的条件 - 2

    我有一个只接受一个参数的方法:defmy_method(number)end如果使用number调用方法,我该如何引发错误??通常,我如何定义方法参数的条件?比如我想在调用的时候报错:my_method(1) 最佳答案 您可以添加guard在函数的开头,如果参数无效则引发异常。例如:defmy_method(number)failArgumentError,"Inputshouldbegreaterthanorequalto2"ifnumbereputse.messageend#=>Inputshouldbegreaterthano

  8. ruby - 如何在 Grape 中定义哈希数组? - 2

    我使用Ember作为我的前端和GrapeAPI来为我的API提供服务。前端发送类似:{"service"=>{"name"=>"Name","duration"=>"30","user"=>nil,"organization"=>"org","category"=>nil,"description"=>"description","disabled"=>true,"color"=>nil,"availabilities"=>[{"day"=>"Saturday","enabled"=>false,"timeSlots"=>[{"startAt"=>"09:00AM","endAt"=>

  9. ruby - 获取模块中定义的所有常量的值 - 2

    我想获取模块中定义的所有常量的值:moduleLettersA='apple'.freezeB='boy'.freezeendconstants给了我常量的名字:Letters.constants(false)#=>[:A,:B]如何获取它们的值的数组,即["apple","boy"]? 最佳答案 为了做到这一点,请使用mapLetters.constants(false).map&Letters.method(:const_get)这将返回["a","b"]第二种方式:Letters.constants(false).map{|c

  10. ruby - 这两个 Ruby 类初始化定义有什么区别? - 2

    我正在阅读一本关于Ruby的书,作者在编写类初始化定义时使用的形式与他在本书前几节中使用的形式略有不同。它看起来像这样:classTicketattr_accessor:venue,:datedefinitialize(venue,date)self.venue=venueself.date=dateendend在本书的前几节中,它的定义如下:classTicketattr_accessor:venue,:datedefinitialize(venue,date)@venue=venue@date=dateendend在第一个示例中使用setter方法与在第二个示例中使用实例变量之间是

随机推荐