是什么
Docker-Compose是Docker官方的开源项目,
负责实现对Docker容器集群的快速编排。
能干嘛
去哪下
官网
https://docs.docker.com/compose/compose-file/compose-file-v3/
官网下载
https://docs.docker.com/compose/install/
curl -L "https://github.com/docker/compose/releases/download/1.29.2/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose chmod +x /usr/local/bin/docker-compose docker-compose --version卸载步骤 Compose核心概念 一文件
docker-compose.yml
两要素 服务(service)
一个个应用容器实例,比如订单微服务、库存微服务、mysql容器、nginx容器或者redis容器
工程(project)
由一组关联的应用容器组成的一个完整业务单元,在 docker-compose.yml 文件中定义。
编写Dockerfile定义各个微服务应用并构建出对应的镜像文件
使用 docker-compose.yml 定义一个完整业务单元,安排好整体应用中的各个容器服务。
最后,执行docker-compose up命令 来启动并运行整个应用程序,完成一键部署上线
Compose常用命令 docker-compose -h # 查看帮助 docker-compose up # 启动所有docker-compose服务 docker-compose up -d # 启动所有docker-compose服务并后台运行 docker-compose down # 停止并删除容器、网络、卷、镜像。 docker-compose exec yml里面的服务id # 进入容器实例内部 docker-compose exec docker-compose.yml文件中写的服务id /bin/bash docker-compose ps # 展示当前docker-compose编排过的运行的所有容器 docker-compose top # 展示当前docker-compose编排过的容器进程 docker-compose logs yml里面的服务id # 查看容器输出日志 docker-compose config # 检查配置 docker-compose config -q # 检查配置,有问题才有输出 docker-compose restart # 重启服务 docker-compose start # 启动服务 docker-compose stop # 停止服务Compose编排微服务 改造升级微服务工程docker_boot
以前的基础版
SQL建表建库
CREATE TABLE `t_user` ( `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`username` varchar(50) NOT NULL DEFAULT '' COMMENT '用户名',
`password` varchar(50) NOT NULL DEFAULT '' COMMENT '密码',
`sex` tinyint(4) NOT NULL DEFAULT '0' COMMENT '性别 0=女 1=男 ',
`deleted` tinyint(4) unsigned NOT NULL DEFAULT '0' COMMENT '删除标志,默认0不删除,1删除',
`update_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新时间',
`create_time` timestamp NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '创建时间', PRIMARY KEY (`id`)) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COMMENT='用户表'
一键生成说明
改POM
4.0.0 org.springframework.boot spring-boot-starter-parent 2.5.6 com.bowenxu docker_boot 1.0 docker_boot Demo project for Spring Boot and DockerCompose 1.8 org.springframework.boot spring-boot-starter-web org.springframework.boot spring-boot-starter-actuator com.google.guava guava 23.0 io.springfox springfox-swagger2 2.9.2 io.springfox springfox-swagger-ui 2.9.2 org.springframework.boot spring-boot-starter-data-redis org.springframework.boot spring-boot-starter-cache org.apache.commons commons-pool2 redis.clients jedis 3.1.0 mysql mysql-connector-java 5.1.49 com.alibaba druid-spring-boot-starter 1.1.22 com.alibaba druid 1.1.16 org.mybatis.spring.boot mybatis-spring-boot-starter 2.2.1 commons-codec commons-codec 1.10 cn.hutool hutool-all 4.6.3 log4j log4j 1.2.17 org.projectlombok lombok org.springframework.boot spring-boot-starter-test test io.projectreactor reactor-test test org.springframework.boot spring-boot-maven-plugin
写YML
server.port=6001
# ========================alibaba.druid相关配置=====================
spring.datasource.type=com.alibaba.druid.pool.DruidDataSource
spring.datasource.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.url=jdbc:mysql://192.168.111.169:3306/db2021?useUnicode=true&characterEncoding=utf-8&useSSL=false
spring.datasource.username=root
spring.datasource.password=123456
spring.datasource.druid.test-while-idle=false
# ========================redis相关配置=====================
spring.redis.database=0
spring.redis.host=192.168.111.169
spring.redis.port=6379
spring.redis.password=
spring.redis.lettuce.pool.max-active=8
spring.redis.lettuce.pool.max-wait=-1ms
spring.redis.lettuce.pool.max-idle=8
spring.redis.lettuce.pool.min-idle=0
# ========================mybatis相关配置===================
mybatis.mapper-locations=classpath:mapper
@Configuration
@Slf4j
public class RedisConfig{
@Bean
public RedisTemplate redisTemplate(LettuceConnectionFactory lettuceConnectionFactory) {
RedisTemplate redisTemplate = new RedisTemplate<>();
redisTemplate.setConnectionFactory(lettuceConnectionFactory);
//设置key序列化方式string
redisTemplate.setKeySerializer(new StringRedisSerializer());
//设置value的序列化方式json
redisTemplate.setValueSerializer(new GenericJackson2JsonRedisSerializer());
redisTemplate.setHashKeySerializer(new StringRedisSerializer());
redisTemplate.setHashValueSerializer(new GenericJackson2JsonRedisSerializer());
redisTemplate.afterPropertiesSet(); return redisTemplate;
}
}
SwaggerConfig
package com.atguigu.docker.config;import org.springframework.beans.factory.annotation.Value;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Configuration;import springfox.documentation.builders.ApiInfoBuilder;import springfox.documentation.builders.PathSelectors;import springfox.documentation.builders.RequestHandlerSelectors;import springfox.documentation.service.ApiInfo;import springfox.documentation.spi.documentationType;import springfox.documentation.spring.web.plugins.Docket;import springfox.documentation.swagger2.annotations.EnableSwagger2;import java.text.SimpleDateFormat;import java.util.Date;
@Configuration
@EnableSwagger2
public class SwaggerConfig{
@Value("${spring.swagger2.enabled}")
private Boolean enabled;
@Bean
public Docket createRestApi() {
return new Docket(documentationType.SWAGGER_2)
.apiInfo(apiInfo())
.enable(enabled)
.select()
.apis(RequestHandlerSelectors.basePackage("com.bowenxu.docker_boot"))
//你自己的package
.paths(PathSelectors.any())
.build();
}
public ApiInfo apiInfo() {
return new ApiInfoBuilder()
.title("title"+"t"+new SimpleDateFormat("yyyy-MM-dd").format(new Date()))
.description("docker-compose")
.version("1.0")
.termsOfServiceUrl("https://www.url.com/")
.build();
}
}
新建entity
User
package com.bowenxu.docker_boot.entity;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import org.springframework.data.annotation.Id;
import java.util.Date;
@Data
@AllArgsConstructor
@NoArgsConstructor
public class User {
private Integer id;
private String username;
private String password;
private Byte sex;
//0女1男,默认0
private Byte deleted;
//删除标志,默认0不删除,1删除
private Date updateTime;
private Date createTime;
}
UserDTO
package com.bowenxu.docker_boot.entity;
import io.swagger.annotations.ApiModel;
import io.swagger.annotations.ApiModelProperty;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.io.Serializable;
import java.util.Date;
@NoArgsConstructor
@AllArgsConstructor
@Data
@ApiModel(value = "用户信息")
public class UserDTO implements Serializable {
@ApiModelProperty(value = "用户ID")
private Integer id;
@ApiModelProperty(value = "用户名")
private String username;
@ApiModelProperty(value = "密码")
private String password;
@ApiModelProperty(value = "性别 0=女 1=男 ")
private Byte sex;
@ApiModelProperty(value = "删除标志,默认0不删除,1删除")
private Byte deleted;
@ApiModelProperty(value = "更新时间")
private Date updateTime;
@ApiModelProperty(value = "创建时间")
private Date createTime;
}
新建mapper
新建接口UserMapper
package com.bowenxu.docker_boot.dao;
import com.bowenxu.docker_boot.entity.CommonResult;
import com.bowenxu.docker_boot.entity.User;
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param;
@Mapper
public interface UserMapper {
User getUserById(@Param("id")Long id);
Integer createUser(User user);
}
srcmainresources路径下新建mapper文件夹并新增UserMapper.xml
insert into t_user (username,password) values (#{username},#{password});
新建service
package com.bowenxu.docker_boot.service;
import com.bowenxu.docker_boot.dao.UserMapper;
import com.bowenxu.docker_boot.entity.CommonResult;
import com.bowenxu.docker_boot.entity.User;
import lombok.extern.slf4j.Slf4j;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
@Service
@Slf4j
public class UserServiceImpl implements UserService{
public static final String CACHE_KEY_USER = "user:";
@Resource
private UserMapper userMapper;
@Resource
private RedisTemplate redisTemplate;
@Override
public Integer testAddUser(User user) {
return userMapper.createUser(user);
}
@Override
public Integer testAddUserCache(User user){
log.info("====>插入之前的user"+user);
//先插入数据库
Integer i = userMapper.createUser(user);
if (i <= 0){
return i;
}
log.info("====>插入之后的user"+user);
//插入成功,放入缓存
//需要查询一下刚才的user把它重新老回来了吗
user = userMapper.getUserById(user.getId());
log.info("====>mysql回写的user"+user);
String key = CACHE_KEY_USER+user.getId();
redisTemplate.opsForValue().set(key,user);
log.info("====>缓存写入成功"+user);
return 1;
}
@Override
public User findUserById(Long id) {
User user = null;
String key = CACHE_KEY_USER+id;
//1 先从redis里面查询,如果有直接返回结果,如果没有再去查询mysql
user = (User) redisTemplate.opsForValue().get(key);
log.info("===========>redis里面查询的user"+user);
if(user != null)
{
return user;
}
//2 redis里面无,从mysql里查
user = userMapper.getUserById(id);
if (user!= null){
//3 mysql有,需要将数据写回redis,保证下一次的缓存命中率
redisTemplate.opsForValue().set(key,user);
}
log.info("===========>从mysql里面查到的user"+user);
return user;
}
}
新建controller
package com.bowenxu.docker_boot.controller;
import cn.hutool.core.bean.BeanUtil;
import com.bowenxu.docker_boot.entity.User;
import com.bowenxu.docker_boot.entity.UserDTO;
import com.bowenxu.docker_boot.service.UserService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.*;
import javax.annotation.Resource;
@RestController
@Slf4j
public class TestController {
@Resource
private UserService userService;
@GetMapping("/testadd")
public String testAddUser( UserDTO userDTO){
log.info("======>接受到的对象"+userDTO);
User user = new User();
//映射copy接收的对象到entity对象
BeanUtil.copyProperties(userDTO,user);
log.info("entity的对象====>"+user);
log.info("=====>开始操作数据库,新增记录");
Integer integer = userService.testAddUser(user);
if (integer>0)
{
return "新增成功";
}else {
return "新增失败";
}
}
@GetMapping("/testcache")
public String testCache(){
User user = new User("123","123456");
Integer integer = userService.testAddUserCache(user);
if (integer>0){
return "创建成功";
}
return "创建失败";
}
@GetMapping("/testqueryuser/{id}")
public String testFindUserById(@PathVariable("id")Long id){
User user = userService.findUserById(id);
return user.toString();
}
}
mvn package命令将微服务形成新的jar包
并上传到Linux服务器/mydocker目录下
编写Dockerfile
# 基础镜像使用java FROM java:8 # 作者MAINTAINER bowenxu # VOLUME 指定临时文件目录为/tmp,在主机/var/lib/docker目录下创建了一个临时文件并链接到容器的/tmp VOLUME /var/lib/docker/tmp:/tmp # 将jar包添加到容器中并更名为test_docker.jar ADD docker_boot-1.0.jar test_docker.jar # 运行jar包 RUN bash -c 'touch /test_docker.jar' ENTRYPOINT ["java","-jar","/test_docker.jar"] #暴露6001端口作为微服务 EXPOSE 6001
构建镜像
docker build -t test_docker:2.0 .
不用Compose
单独的mysql容器实例
新建mysql容器实例
进入mysql容器实例并新建库db2021+新建表t_user
单独的redis容器实例
微服务工程
上面三个容器实例依次顺序启动成功
swagger测试
http://localhost:你的微服务端口/swagger-ui.html#/
上面成功了,有哪些问题?
先后顺序要求固定,先mysql+redis才能微服务访问成功
多个run命令…
容器间的启停或宕机,有可能导致IP地址对应的容器实例变化,映射出错,
要么生产IP写死(可以但是不推荐),要么通过服务调用
使用Compose
服务编排,一套带走,安排
编写docker-compose.yml文件
version: "3"
services:
microService:
image: test_docker:2.0
container_name: ms01
ports:
- "6001:6001"
volumes:
- /app/microService:/data
networks:
- bowen_net
depends_on:
- redis
- mysql
redis:
image: redis:6.0.8
ports:
- "6379:6379"
volumes:
- /app/redis/redis.conf:/etc/redis/redis.conf
- /app/redis/data:/data
networks:
- bowen_net
command:
redis-server /etc/redis/redis.conf
mysql:
image: mysql:5.7
environment:
MYSQL_ROOT_PASSWORD: '123456'
MYSQL_ALLOW_EMPTY_PASSWORD: 'no'
MYSQL_DATAbase: 'db2021'
MYSQL_USER: 'bowenxu'
MYSQL_PASSWORD: '123456'
ports:
- "3306:3306"
volumes:
- /app/mysql/db:/var/lib/mysql
- /app/mysql/conf/my.cnf:/etc/my.cnf
- /app/mysql/init:/docker-entrypoint-initdb.d
networks:
- bowen_net
command:
--default-authentication-plugin=mysql_native_password
#解决外部无法访问
networks:
bowen_net:
第二次修改微服务工程docker_boot
写YML
通过docker服务名(也就是容器名)访问,IP无关,也不是docker网名
mvn package命令将微服务形成新的jar包
并上传到Linux服务器/mydocker目录下
编写Dockerfile
构建镜像
docker build -t test_docker:2.0 .
执行 docker-compose up
或者
执行 docker-compose up -d
进入mysql容器实例并新建库db2021+新建表t_user
测试通过
Compose常用命令
关停



