很久之前就用過,一直沒有好好寫篇博客,今天就總結一下
創建一個 SpringBoot項目
創建一個SpringBoot項目並打成jar包,結構如圖
編寫 Dockerfile文件
FROM java:8
VOLUME /tmp
ADD springboot-1.0-SNAPSHOT.jar boot.jar
EXPOSE 8080
ENTRYPOINT ["java","-Djava.security.egd=file:/dev/./urandom","-jar","/boot.jar"]
解釋一下
-
FROM
:表示基礎鏡像,即運行環境 -
VOLUME
: /tmp創建/tmp目錄並持久化到Docker數據文件夾,因為Spring Boot使用的內嵌Tomcat容器默認使用/tmp作為工作目錄 -
ADD
:拷貝文件並且重命名(ADD springboot-1.0-SNAPSHOT.jar boot.jar 將應用jar包復制到/boot.jar) -
EXPOSE
:並不是真正的發布端口,這個只是容器部署人員與建立image的人員之間的交流,即建立image的人員告訴容器布署人員容器應該映射哪個端口給外界 -
ENTRYPOINT
:容器啟動時運行的命令,相當於我們在命令行中輸入java -jar xxxx.jar,為了縮短 Tomcat 的啟動時間,添加java.security.egd的系統屬性指向/dev/urandom作為 ENTRYPOINT
構建容器
我們將 Dockerfile 和 我們的項目 jar包 上傳到 Linux 服務器上去 ,如何執行下面命令
docker build -t springboot .
解釋
docker build
: 就是 我們編譯 當前目錄下的 Dockerfile文件
-t
: 我的個人理解就是 在進程中運行 ,而那個 點 代碼當前目錄
[root@spiritmark]# docker build -t springboot .
Sending build context to Docker daemon 14.43 MB
Step 1/5 : FROM java:8
Trying to pull repository docker.io/library/java ...
8: Pulling from docker.io/library/java
5040bd298390: Pull complete
fce5728aad85: Pull complete
76610ec20bf5: Pull complete
60170fec2151: Pull complete
e98f73de8f0d: Pull complete
11f7af24ed9c: Pull complete
49e2d6393f32: Pull complete
bb9cdec9c7f3: Pull complete
Digest: sha256:c1ff613e8ba25833d2e1940da0940c3824f03f802c449f3d1815a66b7f8c0e9d
Status: Downloaded newer image for docker.io/java:8
---> d23bdf5b1b1b
Step 2/5 : VOLUME /tmp
---> Running in 0aec2dc2f98c
---> a52e844f25d4
Removing intermediate container 0aec2dc2f98c
Step 3/5 : ADD springboot-1.0-SNAPSHOT.jar boot .jar
---> 3ba2f4fdddda
Removing intermediate container 860a0f748a23
Step 4/5 : EXPOSE 8080
---> Running in 1d3331cc2be6
---> e9ac33d26ce0
Removing intermediate container 1d3331cc2be6
Step 5/5 : ENTRYPOINT java -Djava.security.egd=file:/dev/./urandom -jar /boot.jar
---> Running in d354f8ee2af5
---> 8937e1ade6c7
Removing intermediate container d354f8ee2af5
Successfully built 8937e1ade6c7
運行容器
docker run --rm -d --name 容器名稱 -p 8080:8080 鏡像名稱
- 其中-d表示后台運行容器,這也就自然地解決的Spring Boot不支持后台運行應用程序的問題。
- -p 8080:8080表示將容器內部的8080端口映射到宿主機器的8080端口,這樣就可以通過宿主機器直接訪問應
用。 –name
給容器取一個容易記住的名字方便日后管理。--rm
在Docker容器退出時,默認容器內部的文件系統仍然被保留,以方便調試並保留用戶數據
查看容器運行日志
docker logs -f --tail=100 容器名稱
[root@spiritmark] # docker logs -f --tail=100 04d6b2c34795
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \ ( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / / =========|_|==============|___/=/_/_/_/ :: Spring Boot :: (v1.5.4.RELEASE) 2019-12-29 07:42:58.982 INFO 1 --- [ main] c.b.ElkExampleSpringBootApplication : Starting ElkExampleSpringBootApplication v1.0-SNAPSHOT on 04d6b2c34795 with PID 1 (/elk.jar started by root in /) 2019-12-29 07:42:58.999 INFO 1 --- [ main] c.b.ElkExampleSpringBootApplication : No active profile set, falling back to default profiles: default 2019-12-29 07:42:59.243 INFO 1 --- [ main] ationConfigEmbeddedWebApplicationContext : Refreshing org.springframework.boot.context.embedded.AnnotationConfigEmbeddedWebApplicationContext@5a2e4553: startup date [Sun Dec 29 07:42:59 UTC 2019]; root of context hierarchy 2019-12-29 07:43:03.652 INFO 1 --- [ main] s.b.c.e.t.TomcatEmbeddedServletContainer : Tomcat initialized with port(s): 8080 (http) 2019-12-29 07:43:03.699 INFO 1 --- [ main] o.apache.catalina.core.StandardService : Starting service [Tomcat] 2019-12-29 07:43:03.714 INFO 1 --- [ main] org.apache.catalina.core.StandardEngine : Starting Servlet Engine: Apache Tomcat/8.5.15 2019-12-29 07:43:04.012 INFO 1 --- [ost-startStop-1] o.a.c.c.C.[Tomcat].[localhost].[/] : Initializing Spring embedded WebApplicationContext 2019-12-29 07:43:04.012 INFO 1 --- [ost-startStop-1] o.s.web.context.ContextLoader : Root WebApplicationContext: initialization completed in 4772 ms 2019-12-29 07:43:04.449 INFO 1 --- [ost-startStop-1] o.s.b.w.servlet.ServletRegistrationBean : Mapping servlet: 'dispatcherServlet' to [/] 2019-12-29 07:43:04.470 INFO 1 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean : Mapping filter: 'characterEncodingFilter' to: [/*] 2019-12-29 07:43:04.470 INFO 1 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean : Mapping filter: 'hiddenHttpMethodFilter' to: [/*] 2019-12-29 07:43:04.471 INFO 1 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean : Mapping filter: 'httpPutFormContentFilter' to: [/*] 2019-12-29 07:43:04.471 INFO 1 --- [ost-startStop-1] o.s.b.w.servlet.FilterRegistrationBean : Mapping filter: 'requestContextFilter' to: [/*]
2019-12-29 07:43:05.534 INFO 1 --- [ main] s.w.s.m.m.a.RequestMappingHandlerAdapter : Looking for @ControllerAdvice: org.springframework.boot.context.embedded.AnnotationConfigEmbeddedWebApplicationContext@5a2e4553: startup date [Sun Dec 29 07:42:59 UTC 2019]; root of context hierarchy
2019-12-29 07:43:05.765 INFO 1 --- [ main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/exception]}" onto public java.lang.String com.bruceliu.controller.ELKController.exception()
2019-12-29 07:43:05.766 INFO 1 --- [ main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/elkdemo]}" onto public java.lang.String com.bruceliu.controller.ELKController.helloWorld()
2019-12-29 07:43:05.772 INFO 1 --- [ main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/error],produces=[text/html]}" onto public org.springframework.web.servlet.ModelAndView org.springframework.boot.autoconfigure.web.BasicErrorController.errorHtml(javax.servlet.http.HttpServletRequest,javax.servlet.http.HttpServletResponse)
2019-12-29 07:43:05.780 INFO 1 --- [ main] s.w.s.m.m.a.RequestMappingHandlerMapping : Mapped "{[/error]}" onto public org.springframework.http.ResponseEntity<java.util.Map<java.lang.String, java.lang.Object>> org.springframework.boot.autoconfigure.web.BasicErrorController.error(javax.servlet.http.HttpServletRequest)
2019-12-29 07:43:05.869 INFO 1 --- [ main] o.s.w.s.handler.SimpleUrlHandlerMapping : Mapped URL path [/webjars/**] onto handler of type [class org.springframework.web.servlet.resource.ResourceHttpRequestHandler]
2019-12-29 07:43:05.869 INFO 1 --- [ main] o.s.w.s.handler.SimpleUrlHandlerMapping : Mapped URL path [/**] onto handler of type [class org.springframework.web.servlet.resource.ResourceHttpRequestHandler]
2019-12-29 07:43:05.984 INFO 1 --- [ main] o.s.w.s.handler.SimpleUrlHandlerMapping : Mapped URL path [/**/favicon.ico] onto handler of type [class org.springframework.web.servlet.resource.ResourceHttpRequestHandler]
2019-12-29 07:43:06.387 INFO 1 --- [ main] o.s.j.e.a.AnnotationMBeanExporter : Registering beans for JMX exposure on startup
2019-12-29 07:43:06.537 INFO 1 --- [ main] s.b.c.e.t.TomcatEmbeddedServletContainer : Tomcat started on port(s): 8080 (http)
2019-12-29 07:43:06.562 INFO 1 --- [ main] c.b.ElkExampleSpringBootApplication : Started ElkExampleSpringBootApplication in 8.771 seconds (JVM running for 9.832)
訪問測試
這樣我們項目 已經部署在 阿里雲服務器上了
idea集成docker實現鏡像打包一鍵部署
1、Docker開啟遠程訪問
使用 idea去連接docker ,就必須開放端口,默認 docker是不支持 idea遠程 連接 docker的,所以我們要開放,
首先編輯我們服務器上的docker文件,進入修改docker的地方
#修改該Docker服務文件
vi /usr/lib/systemd/system/docker.service
修改以ExecStart開頭的行(centos 7):添加
#修改ExecStart這行
ExecStart=/usr/bin/dockerd -H tcp://0.0.0.0:2375 -H unix:///var/run/docker.sock
將文件內的 ExecStart注釋。 新增如上行。
#ExecStart=/usr/bin/dockerd -H fd:// --containerd=/run/containerd/containerd.sock
ExecStart=/usr/bin/dockerd -H tcp://0.0.0.0:2375-H unix:///var/run/docker.sock
修改后保存文件,然后重啟docker
systemctl daemon-reload
service docker restart
重啟之后測試遠程連接是否正常,這里的2375是之前配置的端口
curl http://服務ip:2375/version
然后開啟端口,或者關閉防火牆,二者選其一即可
firewall-cmd --zone=public --add-port=2375/tcp --permanent
chkconfig iptables off
然后打開瀏覽器測試將之前的localhost修改為你的ip
使用idea連接到docker
首先下載docker插件,idea2019自帶了docker插件。如果沒有插件可以選擇安裝docker插件
打開Idea,從File->Settings->Plugins->Install JetBrains plugin進入插件安裝界面,
在搜索框中輸入docker,可以看到Docker integration,點擊右邊的Install按鈕進行安裝。
安裝后重啟Idea。
IDEA配置docker
配置docker,連接到遠程docker服務。
從File->Settings->Build,Execution,Deployment->Docker打開配置界面
配置完成鏈接之后,出現了框中的內容即可.
鏈接成功之后會列出容器和鏡像!
配置阿里雲鏡像加速器:
加速器 地址 : —》 大家可以從 阿里雲官網點擊服務鏡像加速 ,去獲取
docker-maven-plugin
傳統過程中,打包、部署、等。
而在持續集成過程中,項目工程一般使用 Maven 編譯打包,然后生成鏡像,通過鏡像上線,能夠大大提供上線效
率,同時能夠快速動態擴容,快速回滾,着實很方便。docker-maven-plugin 插件就是為了幫助我們在Maven工程
中,通過簡單的配置,自動生成鏡像並推送到倉庫中。
<build>
<finalName>${project.artifactId}</finalName>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<version>1.5.3.RELEASE</version>
</plugin>
<!--使用docker-maven-plugin插件-->
<plugin>
<groupId>com.spotify</groupId>
<artifactId>docker-maven-plugin</artifactId>
<version>1.0.0</version>
<configuration>
<!--指定生成的鏡像名-->
<imageName>${docker.image.prefix}/template</imageName>
<!--指定標簽-->
<imageTags>
<imageTag>latest</imageTag>
</imageTags>
<!--指定基礎鏡像jdk1.8-->
<baseImage>java</baseImage>
<!--鏡像制作人本人信息-->
<maintainer>2509919428@email.com</maintainer>
<!--切換到ROOT目錄-->
<workdir>/ROOT</workdir>
<cmd>["java", "-version"]</cmd>
<entryPoint>["java", "-jar", "/${project.build.finalName}.jar"]</entryPoint>
<!--指定遠程 docker api地址-->
<dockerHost>http://39.97.241.130:2375</dockerHost>
<!-- 這里是復制 jar 包到 docker 容器指定目錄配置 -->
<resources>
<resource>
<targetPath>/ROOT</targetPath>
<!--jar 包所在的路徑 此處配置的 即對應 target 目錄-->
<directory>${project.build.directory}</directory>
<!--用於指定需要復制的文件 需要包含的 jar包 ,這里對應的是 Dockerfile中添加的文件名 -->
<include>${project.build.finalName}.jar</include>
</resource>
</resources>
</configuration>
<!---當執行mvn package 時,執行: mvn clean package docker:build-->
<executions>
<execution>
<id>build-image</id>
<phase>package</phase>
<goals>
<goal>build</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
Dockerfile
上面的配置 就和 下面的 效果一樣 ,只不過變成了自動化
FROM java:8
VOLUME /tmp
ADD springboot-1.0-SNAPSHOT.jar boot.jar
EXPOSE 8080
ENTRYPOINT ["java","-Djava.security.egd=file:/dev/./urandom","-jar","/boot.jar"]
執行命令
mvn clean package docker:build
或者
完成后你就會發現多出一個鏡像