美文网首页
Docker 入门(二)

Docker 入门(二)

作者: appea | 来源:发表于2016-03-30 14:20 被阅读244次

    引言

    Docker 是一个非常有趣的项目。它自己宣称可以减轻部署服务器的难度,当然我相信里面有炒作的成分。但是实际使用后,我觉得 Docker 的表现还是可圈可点的。
    Docker最大的作用就是隔绝了操作系统环境,类似于虚拟机,但是相对于虚拟机,他又拥有绝对的高效率、和通用性

    1. docker有着比虚拟机更少的抽象层
    2. docker利用的是宿主机的内核

    所以本节为了能使大家对Docker有一个直观的认识,这里引用一个网友的实际程序来给大家讲解

    开始之前,我们先要引入一个叫做 Image 的东西,不过这里不是图片的意思,他代表的是镜像
      Docker最核心的一个概念就是镜像,他类似于虚拟机当中的虚拟机文件,但是又完全不同,最直观的感觉就是运行一个Docker Image是非常迅速的,通常只需要几秒钟!你可以从Docker的镜像库中下载到很多很多的镜像,上一章的“Hello-Docker”大家还记得么,那就是其中的一个,还有很多,比如说Ubuntu,RabbitMQ等等。

    这里有一个镜像库,感兴趣的同学可以去看看,阿里镜像库

    阿里镜像库

    </br>
    </br>
    </br>


    开始

    好了,这里先不说那么多复杂的概念,咱们上手试试吧!
    今天,我们要在使用Jetty + Docker快速实现和部署一个能显示随机正态分布的页面,非常简单,最终效果如下:
    我们要在使用Jetty + Docker快速实现和部署一个能显示随机正态分布的页面,非常简单,最终效果如下:

    运行结果(图中的钟型曲线是使用Java的Random.nextGaussian产生的。用户可以在页面上输入限制最大值、最小值、数学期望值、标准差和样本总量并由这些参数生成正态分布图。)

    核心目标:

    1. 理解Docker两条基本命令(打包运行)。
    2. 了解Docker基本使用方式。

    开发环境

    先看看开发环境是否准备好了,嗯~想一下我们需要什么,我们需要一个房子和一些零部件,房子就是我们Docker环境,零部件就是JDK环境,然后为了使应用看起来更简单,我们引入Groovy语言来代替java,接着就是一个好的编辑器,vim、emacs,当然哪个顺手就用哪个。

    Jetty服务器

    我们需要写一个Groovy脚本来跑Jetty,首先新建一个文件夹,把这个文件夹作为这个原型的根目录。

    然后在原型根目录中新建app.groovy作为程序入口。这个脚本主要的任务就是为我们启动Jetty服务器,内容如下:

    @Grab('org.eclipse.jetty.aggregate:jetty-server')
    @Grab('org.eclipse.jetty.aggregate:jetty-servlet')
    @Grab('javax.servlet:javax.servlet-api')
    import groovy.servlet.GroovyServlet
    import org.eclipse.jetty.server.Server
    import org.eclipse.jetty.servlet.DefaultServlet
    import org.eclipse.jetty.servlet.ServletContextHandler
    
    def server = new Server(8080)
    def context = new ServletContextHandler(server, '/', ServletContextHandler.SESSIONS)
    
    context.with { 
        resourceBase = 'webroot' // 使用webroot文件夹作为根目录 
        addServlet(DefaultServlet, '/') // 挂入DefaultServlet为*.groovy脚本以外的文件提供访问 
        addServlet(GroovyServlet, '*.groovy') // Groovy脚本用的Servlet     
        welcomeFiles = ['index.groovy'] // welcome文件设置为index.groovy,此处Optional
    }
    server.start()
    

    简单说明一下这个脚本的作用。

    首先我们用@Grab抓取需要的jar包,完了之后导入一些Servlet和Server,然后把这些Servlet挂到端口为8080的Jetty Server上,最后启动Jetty。

    服务器有了,接下来我们需要写正态分布的页面了。

    正态分布

    在原型根目录下新建webroot,在其中新建一个脚本,名字随意取,在这里我们取名叫test1.groovy。我不准备在这里贴这个脚本的全部代码了,你可以在这个项目里面看到源代码。
      更详细的程序说明也可以查看这个博客,注意!这里的代码不是重点,我们只是引用了这位博主的代码,大家不愿意看代码的同学可以直接下载源码。
      这里用到了几个名词需要简单说一下

    1. java.util.Random

    Java的java.util.Random中自带一个能生成数学期望(u)是0、标准差(a)是1的近似随机标准正态分布的随机方法,叫做nextGaussian,我们用它来生成随机数。
      由标准正态分布的特性可知,99%以上的数值落在(u-2.58a, u+2.58a)区间中,由于标准正态分布u = 0, a = 1所以我们可以假定nextGaussian方法生成的双精度值范围在-2.58 ~ +2.58之间,我们要做的只是对这个结果值做一下线性平移,然后收集结果即可。

    1. GroovyServlet

    如果你查看groovy.servlet.ServletBinding
    ,可以看到在GroovyServlet中,已经默认绑定了以下几个变量供我们调遣
    Eager variables

    • “request” : the HttpServletRequest object
    • “response” : the HttpServletRequest object
    • “context” : the ServletContext object
    • “application” : same as context
    • “session” : shorthand for request.getSession(false) - can be null!
    • “params” : map of all form parameters - can be empty
    • “headers” : map of all request header fields

    Lazy variables

    • “out” : response.getWriter()
    • “sout” : response.getOutputStream()
    • “html” : new MarkupBuilder(response.getWriter())
    • “json” : new JsonBuilder()

    Methods

    • “forward(String path)” : request.getRequestDispatcher(path).forward(request, response)
    • “include(String path)” : request.getRequestDispatcher(path).include(request, response)
    • “redirect(String location)” : response.sendRedirect(location)
    1. chart.js

    一个开源图表控件,参考这里的中文文档

    详细代码如下:

    def random = new Random();
    def map = [:]
    
    int MIN = params.distMin? params.distMin.toInteger():0 // 最大值
    int MAX = params.distMax? params.distMax.toInteger():100 // 最小值
    int mean = params.distMean? params.distMean.toInteger():50 // 数学期望值
    int sd = params.distSD? params.distSD.toInteger():2 // 标注差
    int nums = params.distTimes? params.distTimes.toInteger():100
    MAX = Math.max(MAX, MIN)
    MIN = Math.min(MAX, MIN)
    
    if(MAX == MIN) {
        MAX = MIN + 100
    }
    
    if(nums <=0) {
        nums = 100
    }
    
    // System.out.println "MIN:$MIN"
    // System.out.println "MAX:$MAX"
    // System.out.println "mean:$mean"
    // System.out.println "sd:$sd"
    // System.out.println "nums:$nums"
    
    nums.times {
        def gaussian = MIN - 1
        while(gaussian < MIN || gaussian > MAX) {
            gaussian = random.nextGaussian(); // mean 0.0, standard deviation 1.0// transform
            gaussian = (int)(sd * gaussian) + mean
        }
        // System.out.println "runs[$it]=$gaussian"
        if(map[gaussian]) {
            map[gaussian] += 1
        } else {
            map[gaussian] = 1
        }
    }
    
    // System.out.println map
    def keys = map.keySet().sort()
    def dataLabels = (keys[0]..keys[-1]).collect { it.toString() }
    def dataList = (keys[0]..keys[-1]).collect { map[it]?:0 }
    
    //这里动态生成html页面
    html.html {
        head {
            title 'Gaussian Distribution Test'
            script src:'//cdn.bootcss.com/Chart.js/1.0.2/Chart.min.js'
        }
        body {
            h1 '(伪)正态分布研究'
            canvas id:'myChart', width: 800, height: 500
            form (method:'POST') {
                label ('最小值MIN') { input(type:'number', name:'distMin', placeholder:'最小值', value:"$MIN", required:'required') }
                label ('最大值MAX') { input(type:'number', name:'distMax', placeholder:'最大值', value:"$MAX", required:'required') }
                label ('数学期望值') { input(type:'number', name:'distMean', placeholder:'数学期望', value:"$mean", required:'required') }
                label ('标准差') { input(type:'number', name:'distSD', placeholder:'标准差', value:"$sd", required:'required') }
                label ('样本总量') { input(type:'number', name:'distTimes', placeholder:'样本总量', value:"$nums", required:'required') }
                button(type:'submit', '提交')
            }
            script {
                mkp.yield 'var data ='
            json {
                labels dataLabels
                datasets ([[
                    fillColor : "rgba(151,187,205,0.5)",
                    strokeColor : "rgba(151,187,205,1)",
                    pointColor : "rgba(151,187,205,1)",
                    pointStrokeColor : "#fff",
                    data : dataList
                ]])
            }
        }
        script {
            mkp.yieldUnescaped '''
            var options = {
                //String - Colour of the scale line
                scaleLineColor : "rgba(0,0,0,.5)",
                //String - Colour of the grid lines
                scaleGridLineColor : "rgba(0,0,0,.1)",
                //Boolean - Whether to show a dot for each point
                pointDot : false,
             };
             var ctx = document.getElementById("myChart").getContext("2d");
             new Chart(ctx).Line(data, options);
             '''
            }
        }
    }
    

    运行

    重头戏现在才开始,首先让我们运行一下上面的代码,先检查一下工程文件是否像下图这样:

    目录1

    webroot中包含test1.groovy文件

    目录2

    好啦,让我们在主目录尝试运行如下命令:

    $ groovy app.groovy

    完成!浏览器访问http://localhost:8080/test1.groovy

    是不是看到运行的效果了?如果不对的话再检查检查,相信不会有什么问题!

    部署

    这里我们先回忆一下我们需要部署什么东西,首先就是java环境,很多人可能要问,为什么要部署java环境,本机上的java不是已经安装了么?其实,用上面的房子理论来解释就是,docker他默认是一个空房子,并不包含任何运行环境,为了让我们的工具可以运行起来,我们首先是要配置好房子里的工具,也就是运行环境,其次才是打包我们的代码、数据库文件等。
      接下来就是要部署Groovy的运行环境,配置完成以后还需要将相应的工具配置到环境变量,就和本机配置一样。
      Jetty呢?这个不用在容器中配置,因为它是通过Groovy动态引入的,这也是Groovy的一个魅力所在!我们可以用过它简化很多操作,并实现一些Java不容易实现的功能。这个我在后面会单独开一个Groovy的文章详细说明。
    总结一下:

    都用到了如下这些工具:

    1. JAVA
      openjdk-8FROM java:openjdk-8-jdk
    2. Groovy
      Install groovyADD http://dl.bintray.com/groovy/maven/apache-groovy-binary-${GROOVY_VERSION}.zip /tmp/ RUN unzip -d /opt/ /tmp/apache-groovy-binary-${GROOVY_VERSION}.zip \ && rm /tmp/apache-groovy-binary-${GROOVY_VERSION}.zip

    Dockerfile代码如下:

    # 使用openjdk-8
    FROM java:openjdk-8-jdk
    
    # 安装wget和unzip
    RUN apt-get update && \
        apt-get -y install wget unzip && \
        apt-get clean
    
    # 设定环境变量
    ENV GROOVY_VERSION=2.4.5
    ENV JAVA_HOME=/usr/lib/jvm/java-8-openjdk-amd64 \
        GROOVY_HOME=/opt/groovy-${GROOVY_VERSION}
    
    ENV PATH=$GROOVY_HOME/bin/:$JAVA_HOME/bin:$PATH
    
    # Install groovy
    ADD http://dl.bintray.com/groovy/maven/apache-groovy-binary-${GROOVY_VERSION}.zip /tmp/
    
    RUN unzip -d /opt/ /tmp/apache-groovy-binary-${GROOVY_VERSION}.zip \
      && rm /tmp/apache-groovy-binary-${GROOVY_VERSION}.zip
    
    # 复制代码
    ADD ./src/ /groovyApp
    
    EXPOSE 8080
    
    WORKDIR /groovyApp
    
    # 运行groovy
    ENTRYPOINT ["groovy", "app.groovy"]
    

    OK,然后在docker环境中运行:(如果你是docker-machine的话,就是mac或者windows的docker,就是在Docker Quickstart Terminal
    中,对就是头上有个金鱼
    鲸鱼的那个终端)

    这里需要注意下目录:

    目录3

    src中就是刚才的项目文件夹,在这个目录中我们打开Terminal,运行

    docker build -t gaussianranddemo .

    需要注意新版本的Docker规定命名必须是全部 小写
    首次运行Groovy需要下载依赖包, 可能需要点时间, 请耐心等候....
    等待Docker下载并构建成功后运行

    docker run -d -p 8080:8080 GaussianRandDemo

    -d为是否后台运行,如果想查错误,可以把-d去掉,有异常可以调试

    这个时候我们去查看http://localhost:8080/test1.groovy
    如果页面正常显示,证明部署,运行都已经成功了!
    最后再看一眼实际运行的效果

    最终效果

    排查问题

    • 如果没有成功的话,可以去查看下端口占用情况

    $ netstat -ap|grep 8080

    • 容器运行情况

    $ sudo docker ps -a


    容器实例运行状态
    • 镜像状态

    $ sudo docker images


    镜像状态

    相关文章

      网友评论

          本文标题:Docker 入门(二)

          本文链接:https://www.haomeiwen.com/subject/dobjlttx.html