【微服务】springboot 构建docker镜像多模式使用详解

目录

一、前言

二、微服务常用的镜像构建方案

3.1 使用Dockerfile

3.2 使用docker plugin插件

3.3 使用docker compose 编排文件

三、环境准备

3.1 服务器

3.2 安装JDK环境

3.2.1 创建目录

3.2.2 下载安装包

3.2.3 配置环境变量

2.2.4 查看java版本

3.3 安装maven

3.3.1 下载maven安装包并解压

3.3.2 配置setting文件

3.3.3 配置maven的环境变量

3.3.4 验证maven是否配置成功

3.4 安装git

3.5 准备docker环境和docker compose环境

四、搭建springboot工程

4.1 搭建步骤

4.1.1 添加依赖

4.1.2 添加测试接口

4.1.3 接口测试

五、使用Dockerfile构建docker镜像

5.1 添加Dockerfile文件

5.2 推送代码到仓库

5.3 下载代码

5.4 使用mvn命令打包

5.5 docker-compose构建镜像

5.6 启动镜像

5.7 访问服务

5.8 优化改进

5.8.1 利用jenkins进行任务构建

5.8.2 使用shell脚本

六、使用docker maven plugin插件构建镜像

6.1 docker maven plugin介绍

6.1.1 docker maven plugin 配置语法说明

6.1.2 Docker Maven 插件常用执行命令

6.2 使用Docker Maven Plugin插件构建镜像

6.2.1 添加插件配置信息

6.2.2 添加一个测试接口

6.2.3 推送代码到gitee

6.2.4 再次拉取代码

6.2.5 打包构建镜像

6.2.6 镜像检查

5.2.7 启动镜像

5.2.8 访问接口

5.2.9 打包阶段触发构建镜像

七、使用dockerfile maven plugin插件构建镜像

7.1 dockerfile maven plugin概述

7.2 使用dockerfile maven plugin制作镜像

7.2.1 配置Dockerfile文件

7.2.2 配置maven plugin信息

7.2.3 代码推送与拉代码

7.2.4 镜像打包

7.2.5 镜像检查

7.2.6 启动镜像服务

7.2.7 接口测试

7.3 dockerfile maven plugin补充说明

7.3.1 跳过插件目标

7.3.2 跳过某一个goal

八、写在文末


一、前言

容器化部署方式让微服务从打包到部署更加方便,以springboot开发的微服务为例来说,随着服务的数量越来越多,微服务部署需要占用的服务器资源开销也越来越大,这对于很多项目团队来说,服务器资源是一笔不可忽略的成本计算,而容器化部署之后,一定程度上可以缓解服务部署带来的服务器资源过度紧张的问题,接下来将详细聊聊如何基于springboot微服务快速完成镜像打包,部署的常用方式。

二、微服务常用的镜像构建方案

常用的服务构建镜像方式主要有下面几种

3.1 使用Dockerfile

这是一种经过很多中小团队实践之后,容易上手,技术成本也比较低的方式,对于springboot开发的小型应用服务来说,是一个不错的选择,网上关于这方面的资料也比价多。

3.2 使用docker plugin插件

docker plugin插件,可以结合maven管理工具,以插件的方式引入到工程中,再通过参数配置的方式,即可实现在项目打包的时候完成镜像的构建工作。

3.3 使用docker compose 编排文件

这种方式更适合于那种基于微服务框架开发的应用,通过docker compose 编排文件,最后完成微服务启动时所依赖的所有环境,难度较大,但是可以基于这种方式,后续再在服务接入k8s时提供基础和便利。

接下来将详细介绍几种常用的镜像构建方式。

三、环境准备

3.1 服务器

配置:

  • 虚拟机或云服务一台,基于centos7.5或以上的版本;
  • 配置,2C4G(或者更高);

3.2 安装JDK环境

后文中项目拉取到服务器目录之后,需要基于java环境打包,所以需提前配置jdk,这里选择jdk17版本。参考下面的步骤操作。

3.2.1 创建目录

cd /usr/local/soft
mkdir java
cd java/

3.2.2 下载安装包

使用下面的命令下载,也可以到官网进行下载

wget https://download.oracle.com/java/17/latest/jdk-17_linux-x64_bin.tar.gz

3.2.3 配置环境变量

解压安装包

tar -zxvf jdk-17_linux-x64_bin.tar.gz

配置环境变量,使用vi /etc/profile命令,将以下内容添加到文件末尾

JAVA_HOME=/usr/local/soft/java/jdk-17.0.11
PATH=$PATH:$JAVA_HOME/bin  
export JAVA_HOME PATH

使用source命令重新加载配置文件使其生效

source /etc/profile

2.2.4 查看java版本

使用下面的命令查看java是否安装成功

java -version

3.3 安装maven

后文中需要通过git拉取代码后在服务器打包构建,需要安装maven,这里使用的是3.6.3的版本,大于这个版本都可以

3.3.1 下载maven安装包并解压

使用下面的命令下载和解压

#下载
wget https://archive.apache.org/dist/maven/maven-3/3.6.3/binaries/apache-maven-3.6.3-bin.tar.gz
解压
tar -zxvf apache-maven-3.6.3-bin.tar.gz

3.3.2 配置setting文件

为了保证后续下载服务依赖的jar包时速度比较快,建议配置下中央仓库的地址

cd apache-maven-3.6.3/
cd conf
cp settings.xml settings.xml.bk
vi settings.xml

配置下面的镜像仓库地址,拷贝下面的配置信息到settings.xml中

#配置本地仓库地址
<localRepository>/usr/local/soft/mvn/repo</localRepository>#使用阿里镜像下载
<mirror><id>alimaven</id><name>aliyun maven</name><url>http://maven.aliyun.com/nexus/content/groups/public/</url><mirrorOf>central</mirrorOf>
</mirror>

3.3.3 配置maven的环境变量

将下面的配置信息追加到/etc/profile文件中

#改环境变量
vi /etc/profile#复制下面三行到最下面(路径要根据自己情况改)
#maven
export MAVEN_HOME=/usr/local/soft/mvn/apache-maven-3.6.3
export PATH=$PATH:$MAVEN_HOME/bin#保存退出,然后重载配置
source /etc/profile

3.3.4 验证maven是否配置成功

#查看
mvn -v#检查
mvn -version

3.4 安装git

使用下面的命令安装git

yum install git

3.5 准备docker环境和docker compose环境

提前在服务器安装docker和docker-compose环境,网上资料比较多,这里就略过了

docker安装和docker-compose的安装可以参考:docker-compose安装与配置

四、搭建springboot工程

快速搭建一个springboot工程,参考下面的流程,本例使3.2.4的版本

4.1 搭建步骤

4.1.1 添加依赖

    <parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>3.2.4</version><relativePath/></parent><properties><docker.image.prefix>dcloud</docker.image.prefix></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><scope>provided</scope></dependency></dependencies><build><finalName>boot-docker</finalName><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins></build>

4.1.2 添加测试接口

为了方便测试,这里添加一个测试接口

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;@RestController
public class DockerImageController {//localhost:8088/docker/v1@GetMapping("/docker/v1")public Object dockerTest1(){return "docker image test";}}

4.1.3 接口测试

调用一下上面的接口,确保可以正常使用

五、使用Dockerfile构建docker镜像

这种方式也是目前很多项目团队在使用的方式,主要是通过在工程中添加Dockerfile文件,最后再在安装了docker环境的服务器上通过docker-compose的命令构建镜像的方式,最后启动镜像服务,下面快速还原一下该过程。

5.1 添加Dockerfile文件

在工程根目录下添加一个Dockerfile文件,编写的配置内容参考如下

#FROM java:8
FROM  openjdk:17-jdk-alpine
ADD target/boot-docker.jar boot-docker.jar
ENTRYPOINT ["java","-jar","/boot-docker.jar"]

注意:

1、使用的java版本建议与服务器安装版本保持一致,这里使用的都是17;

2、Dockerfile文件中还可以添加更丰富的信息,比如指定java -jar启动的其他参数;

5.2 推送代码到仓库

然后将上面的代码推送至git

5.3 下载代码

登录目标服务器之后,使用git clone命令下载gitee的代码

git clone https://gitee.com/licong_congge/boot-docker.git

5.4 使用mvn命令打包

进入到项目主目录,依次执行下面的命令,初次执行可能比较慢,需要从远程的阿里仓库下载依赖包

cd boot-docker/
mvn clean package

5.5 docker-compose构建镜像

打包成功后,使用下面的命令构建服务jar包的镜像,注意,进入工程的主目录执行,即Dockerfile文件所在的目录

docker build -t boot-docker:1.0 .

构建完成后,使用下面的命令查看上面构建的镜像

5.6 启动镜像

确认镜像构建成功后,使用下面的docker命令启动服务

docker run -d -it -p 8088:8088 --name=boot-k8s boot-docker:1.0

参数说明:

  • -d,后天启动;

  • 8088:8088,宿主机8088端口,映射容器内的8088服务端口;

  • --name=boot-k8s,自定义启动的镜像名;

  • boot-docker:1.0,上述构建出来的镜像名;

通过docker ps检查容器是否启动成功

5.7 访问服务

开放服务器8088端口,访问一下接口做测试

5.8 优化改进

以上,我们通过手动的方式完成了一个微服务工程从编写Dockerfile文件开始,到最后构建出镜像,并通过制作的镜像启动容器服务的完整操作流程,但是从操作上来看,过程还是比较繁琐的,因此可以考虑从两个方面进行改进。

5.8.1 利用jenkins进行任务构建

在jenkins中,可以通过任务编排的方式,将上述的每一步进行前后的配置和编排,从而以任务的方式完成。

5.8.2 使用shell脚本

细心的同学不难发现,其实上面的每一步,都可以在shell脚本中进行编写,按照业务的顺序进行组装即可,shell脚本中大致的逻辑如下:

git clone 代码分支
cd 项目主目录
mvn clean package
查看目前运行中的docker镜像以及版本号
重新构建docker镜像(根据上一个版本号进行叠加)
停止之前运行中的docker镜像服务
启动本次最新构建的docker镜像

六、使用docker maven plugin插件构建镜像

从上面的打包到构建镜像的整个过程还是比较繁琐的,每一步出现错误都可能引起后面操作步骤的失败。那么是否还有更简单的方式来做这个事情呢?答案是肯定的,那就是使用docker-maven-plugin的方式来做。

6.1 docker maven plugin介绍

Docker Maven Plugin插件是一个 Maven 插件,它允许开发人员在构建过程中使用 Docker。使用该插件,开发人员可以在 Maven 构建过程中构建、运行和推送 Docker 镜像。这使得开发人员能够更轻松地集成 Docker 到他们的开发工作流程中,从而可以更方便地进行应用程序的构建和部署。

github地址:https://github.com/spotify/docker-maven-plugin

6.1.1 docker maven plugin 配置语法说明

要使用 Docker Maven 插件,你需要在 Maven 项目的 pom.xml 文件中添加相关配置。

1)你需要在 plugins 部分添加 Docker Maven 插件的依赖:

<build><plugins><plugin><groupId>io.fabric8</groupId><artifactId>docker-maven-plugin</artifactId><version>VERSION</version><configuration><!-- 插件的配置选项 --></configuration></plugin></plugins>
</build>

2)在 configuration 部分,你可以配置插件的选项,例如 Docker 镜像的名称、标签、构建上下文等。

6.1.2 Docker Maven 插件常用执行命令

可以使用以下命令来执行 Docker Maven 插件的常用功能

  • mvn docker:build:构建 Docker 镜像;

  • mvn docker:push:推送 Docker 镜像到 Docker Registry;

  • mvn docker:start:启动 Docker 容器;

  • mvn docker:stop:停止 Docker 容器;

6.2 使用Docker Maven Plugin插件构建镜像

参考下面的操作步骤

6.2.1 添加插件配置信息

在当前工程的pom文件中,在pom中增加新的Docker Maven插件相关配置信息,其他的与上述保持一致,参考如下配置,结合里面的注释进行理解

<build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin><plugin><groupId>io.fabric8</groupId><artifactId>docker-maven-plugin</artifactId><version>0.33.0</version><configuration><!-- Docker 推送镜像仓库地址(如果有,可以配置) --><!--<pushRegistry>http://localhost:5000</pushRegistry>--><images><image><!--由于推送到私有镜像仓库,镜像名需要添加仓库地址(相当于告诉去哪里拉取镜像)--><name>${project.name}:${project.version}</name><!--定义镜像构建行为--><build><!--定义基础镜像,这里使用的是JDK17--><from>openjdk:17</from><args><!-- jar的名称,一般配置为gav的av --><JAR_FILE>${project.build.finalName}.jar</JAR_FILE></args><!--定义哪些文件拷贝到容器中--><assembly><!--定义拷贝到容器的目录--><targetDir>/</targetDir><!--只拷贝生成的jar包--><descriptorRef>artifact</descriptorRef></assembly><!--定义容器启动命令--><entryPoint>["java", "-jar","/${project.build.finalName}.jar"]</entryPoint><!--定义维护者--><maintainer>firefish</maintainer><!--使用Dockerfile构建时打开--><!--<dockerFileDir>${project.basedir}</dockerFileDir>--></build><!--定义容器启动行为--><run><!--设置容器名,可采用通配符,像下面这样(一般配置为gav的a)--><!--<containerNamePattern>%a-%i</containerNamePattern>--><containerNamePattern>${project.artifactId}</containerNamePattern><!--设置宿主机与容器的端口映射--><ports><port>8088:8088</port></ports><!--设置超时等待时间--><!--<wait><http><url>http://localhost:8088</url></http><time>20000</time></wait>--></run></image></images></configuration></plugin></plugins></build>

这个配置做了些什么事情呢?总结来说,主要做了下面的步骤:

  • 引入docker-maven-plugin插件,通过maven坐标的方式进行引入;
  • 在image标签中指定镜像名;
  • 配置构建参数,即在build标签中配置;
    • 比如指定镜像名称,拷贝jar包到镜像容器中,指定启动镜像的命令等;
  • 定义容器的启动行为,比如指定容器对外的暴露端口;

6.2.2 添加一个测试接口

该接口给docker容器启动服务时候做服务检测时使用

	@GetMappingpublic Object hello(){return "hello";}

调用一下确保接口可用

6.2.3 推送代码到gitee

推送到代码仓库,方便后续做持续集成

6.2.4 再次拉取代码

可以重新创建一个目录,然后拉取代码到该目录

6.2.5 打包构建镜像

使用上面介绍的命令进行镜像的打包构建,进入代码的主目录,执行打包构建命令

mvn package docker:build

执行之后,等待镜像包的构建完成即可,从下面的执行日志中不难看出,这里相当于是通过在pom中声明配置的方式,完成了类似Dockerfile中执行命令的编写。简单来说,这个命令的执行,就会自动进行编译,打包,构建镜像。

执行完成后,在target目录下生成了jar包

6.2.6 镜像检查

有没有生成镜像呢?通过docker images检查下,发现已经按照预期的目标生成了镜像

5.2.7 启动镜像

在上面介绍了基于这种方式的常用命令,使用下面的命令启动容器

mvn docker:start

然后通过docker ps命令,可以看到镜像的容器已经启动

5.2.8 访问接口

最后访问一下接口,看是否能够正常访问服务

5.2.9 打包阶段触发构建镜像

即在使用mvn package的时候就可以直接构建镜像了,可以再在上述配置的后面添加如下配置

               <!--触发构建及推送操作的执行计划--><executions><!-- 定义特定的执行目标--><execution><!-- 执行的ID,用于唯一标识执行目标--><id>build</id><!-- 绑定到maven构建生命周期中的阶段,在此处为verify阶段--><phase>verify</phase><goals><!-- 指定构建docker镜像的目标--><goal>build</goal></goals></execution></executions>

七、使用dockerfile maven plugin插件构建镜像

从上面docker maven plugin的使用来看,需要在plugin中插件中配置较多的参数,虽然来说脱离了编写Dockerfile的麻烦,这也同时带来另一个问题,配置参数太多,对于初学者,或者对此研究不深的人来说,如果遇到某个参数配置错误,或者对某些配置参数不够理解,则在打包镜像的过程中遇到问题难以跨越,下面介绍另一种插件,即dockerfile maven plugin,使用这种插件,可以结合Dockerfile一起使用,同时配置参数更少,在实操过程中也将更为灵活。

7.1 dockerfile maven plugin概述

正常情况下,在开发一个应用程序后,会使用maven命令进行打包,生成对应的jar文件。而后,基于Dockerfile文件,使用docker-compose命令将jar文件build成一个镜像(docker image)。最后,就可以基于创建的镜像开启服务。这是在springboot工程中制作镜像的一般思路,在上文我们也做了演示。

如果使用dockerfile-maven-plugin 插件,可以在项目构建的时候自动生成镜像,也可以自动将生成的镜像push到指定的镜像库。

7.2 使用dockerfile maven plugin制作镜像

仍然以上面的代码为例,基于dockerfile maven plugin插件,来演示完整的镜像制作过程。

7.2.1 配置Dockerfile文件

在项目的根目录下添加一个Dockerfile文件,配置内容如下:

FROM  openjdk:17-jdk-alpine
ARG JAR_FILE
ADD target/${JAR_FILE} /boot-docker.jar
ENTRYPOINT ["java","-jar","/boot-docker.jar"]
EXPOSE 8088

参数说明:

  • FROM openjdk:17-jdk-alpine;
    • 基础镜像,即容器中启动服务时依赖的jdk镜像,并指定具体版本;
  • ARG JAR_FILE;
    • ARG 构建参数的格式是:ARG <参数名>[=<默认值>],Dockerfile 中的 ARG 指令是定义参数名称,以及定义其默认值。该默认值可以在构建命令 docker build 中用 --build-arg <参数名>=<值> 来覆盖。
  • ADD target/${JAR_FILE} /boot-docker.jar
    • 将构建的服务jar包添加到容器中;
  • ENTRYPOINT,容器中服务器启动命令,可以扩展更多参数,比如指定启动时的jvm参数信息等;
  • EXPOSE 8088,对外暴露的端口;

7.2.2 配置maven plugin信息

在build标签中添加dockerfile maven plugin的配置信息

<build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin><plugin><groupId>com.spotify</groupId><artifactId>dockerfile-maven-plugin</artifactId><version>1.4.13</version><executions><execution><id>default</id><phase>package</phase><goals><goal>build</goal></goals></execution></executions><configuration><repository>congge/${project.artifactId}</repository><tag>${project.version}</tag><buildArgs><JAR_FILE>${project.build.finalName}.jar</JAR_FILE></buildArgs></configuration></plugin></plugins></build>

核心配置参数说明如下:

  • execution,指定了该插件build目标使用默认绑定。

    • 默认情况下,build和tag目标绑定到package阶段,push目标绑定到deploy阶段,这里不涉及tag和push目标。

  • repository,表示生成的镜像的repository信息;

  • tag元素,指定生成镜像的tag,一般使用工程的版本号即可,这里使用Maven模块的版本号;

  • buildArgs,指定传递给Dockerfile的参数,比如上面Dockerfile中的JAR_FILE;

  • ${project.build.finalName}.jar,是jar包路径,这里使用了最终生成的jar包的文件名;

7.2.3 代码推送与拉代码

重新拉取上面最新的代码到服务器目录

7.2.4 镜像打包

使用下面的命令执行工程的打包与镜像构建

mvn package

从日志也能清晰看到完整的过程,执行完成后,在当前的目录下,可以看到jar包已经生成

7.2.5 镜像检查

使用docker images命令检查镜像是否生成

7.2.6 启动镜像服务

有了镜像之后,就可以启动并使用了,使用下面的命令启动镜像服务

docker run -d -p 8088:8088 congge/boot-docker:1.0-SNAPSHOT

7.2.7 接口测试

最后再次访问一下上述提供的接口

7.3 dockerfile maven plugin补充说明

相关命令补充说明

7.3.1 跳过插件目标

如果要临时跳过所有的Dockerfile相关的目标,执行如下Maven命令:

mvn clean install -Ddockerfile.skip

7.3.2 跳过某一个goal

如果只是想跳过某一个goal,执行如下Maven命令:

mvn clean package -Ddockerfile.build.skipmvn clean package -Ddockerfile.tag.skipmvn clean deploy -Ddockerfile.push.skip

八、写在文末

本文通过较大的篇幅详细介绍了如何基于springboot服务构建镜像的几种方式,随着k8s在越来越多的项目中落地使用,服务的容器化部署已经成为一种传统部署的很好补充,值得深入学习和研究,本篇到此结束,感谢观看。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/19310.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JVM学习-垃圾回收(二)

标记-清除(Mark-Sweep)算法 当堆中的有效内存空间被耗尽的时候&#xff0c;就会停止整个程序(stop the world)&#xff0c;然后进行两项工作&#xff0c;第一项则是标记&#xff0c;第二项是清除 标记&#xff1a;Collector从引用根节点开始遍历&#xff0c;标记所有被引用的…

深入理解哈希加密:md5在保护用户数据中的应用

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、md5加密方法简介 二、md5加密方法的实现 示例代码&#xff1a; 三、md5加密方法在实际…

老师如何对付挑事儿的家长?

身为老师&#xff0c;你有没有遇到过这样的家长&#xff1a;孩子在学校里闹点小矛盾&#xff0c;或者作业分数有点争议&#xff0c;他们就气势汹汹地来找你&#xff0c;说你偏心&#xff0c;甚至在其他家长面前说三道四&#xff1f;面对这种爱“挑事”的家长&#xff0c;老师们…

RestTemplet 自定义消息转换器总结

在RestTemplet 请求中&#xff0c;请求发送一个 HTTP 请求时&#xff0c;RestTemplet 会根据请求中的内容类型&#xff08;Content-Type&#xff09;选择合适的 HttpMessageConverter 来处理请求体的数据。同样地&#xff0c;当服务器返回一个 HTTP 响应时&#xff0c;RestTemp…

模型实战(21)之 C++ - tensorRT部署yolov8-det 目标检测

C++ - tensorRT部署yolov8-det 目标检测 python环境下如何直接调用推理模型转换并导出:pt -> onnx ->.engineC++ tensorrt 部署检测模型不写废话了,直接上具体实现过程+all代码 1.Python环境下推理 直接命令行推理,巨简单yolo detect predict model=yolov8n.pt source…

中英文语音合成芯片(TTS芯片)WT3000T8-在ETC上的应用案例

一&#xff1a;开发背景&#xff1a; 随着智能化浪潮的推进&#xff0c;ETC&#xff08;电子不停车收费&#xff09;系统已逐渐成为现代交通的重要组成部分。在这一背景下&#xff0c;唯创知音自主研发的WT3000T8中文转语音合成芯片&#xff08;TTS芯片&#xff09;以其卓越的性…

【开源项目】Excel数据表自动生成工具v1.0版

一、介绍 Excel数据表自动生成工具是Go语言编写的一款小型工具软件&#xff0c;用于将特定的Excel表格内容导出为多种编程语言的代码或可以直接读取的数据内容。 开源Github地址&#xff1a;https://github.com/SkyCreator/goproj 二、版本v1.0功能概览 1.编程语言支持 目前…

Spring Boot发送邮件时如何支持定时功能?

如何使用Spring Boot结合AokSend以实现高效邮件发送&#xff1f; 如何高效地进行sendmail发送邮件并支持定时功能是一个值得探讨的问题。本文将详细介绍如何在Spring Boot中实现定时sendmail发送邮件&#xff0c;并结合AokSend工具实现高效邮件发送。 Spring Boot发送邮件&am…

10W QPS高并发,如何防止重复下单?

小北说在前面 10wqps高并发&#xff0c;如何防止重复提交/支付订单&#xff1f; 10wqps高并发&#xff0c;如何防止重复下单&#xff1f; 10wqps高并发&#xff0c;如何防止重复支付&#xff1f; 10wqps高并发&#xff0c;如何解决重复操作问题&#xff1f; 最近有小伙伴在面试…

JSON.stringify()方法

JSON.stringify()方法 JSON.stringify()方法用于将一个值转为JSON字符串&#xff0c;该字符串符合JSON格式&#xff0c;它也可以被JSON.parse()方法还原。 一、使用JSON.stringify()方法返回的效果&#xff0c;以及需要注意的地方&#xff1a; 注意&#xff1a;原始类型的字符…

奥利奥罚单背后的启示:企业合规与反垄断的边界

在全球化的经济环境中&#xff0c;企业面临着激烈的市场竞争。为了在竞争中脱颖而出&#xff0c;一些企业可能会采取不正当的竞争手段&#xff0c;如垄断、价格歧视等。然而&#xff0c;这些行为往往会触犯反垄断法规&#xff0c;给企业带来严重的法律风险。最近&#xff0c;奥…

Java Class类简介

一、类图&#xff1a; 二、基本介绍&#xff1a; 1. Class也是类&#xff0c;因此也继承了Object类。 2. Class类的对象不是new出来的&#xff0c;是系统创建的。 类加载器ClassLoader有个方法LoadClass()&#xff0c;将某个类对应的Class对象生成在堆中。 通过调试可以发现&am…

Linux静态库、共享动态库介绍、制作及使用

参考学习&#xff1a;Linux下的各种文件 、动态库基本原理和使用方法&#xff0c;-fPIC选项的来龙去脉 、Linux静态库和动态库分析 文章写作参考&#xff1a;Linux共享库、静态库、动态库详解 - sunsky303 - 博客园 (cnblogs.com) 一.Linux共享库、静态库、动态库详解 使用G…

ResNet 原理剖析以及代码复现

原理 ResNet 解决了什么问题&#xff1f; 一言以蔽之&#xff1a;解决了深度的神经网络难以训练的问题。 具体的说&#xff0c;理论上神经网络的深度越深&#xff0c;其训练效果应该越好&#xff0c;但实际上并非如此&#xff0c;层数越深会导致越差的结果并且容易产生梯度爆炸…

市场巨变,移动开发行业即将迎来“第二春”?

随着鸿蒙生态的不断壮大&#xff0c;越来越多的企业开始加入其中&#xff0c;对鸿蒙OS开发工程师的需求也越来越迫切。 年初时还只有200个APP宣布加入鸿蒙生态&#xff0c;而最近华为也已经官宣&#xff0c;已经有4000多个应用加入鸿蒙&#xff0c;短短三个月就增加了20倍。 …

【Python】解决Python报错:TypeError: ‘xxx‘ object does not support item assignment

&#x1f9d1; 博主简介&#xff1a;阿里巴巴嵌入式技术专家&#xff0c;深耕嵌入式人工智能领域&#xff0c;具备多年的嵌入式硬件产品研发管理经验。 &#x1f4d2; 博客介绍&#xff1a;分享嵌入式开发领域的相关知识、经验、思考和感悟&#xff0c;欢迎关注。提供嵌入式方向…

Monaco-Editor在Vue中使用(实现代码编辑与diff代码比较)--类似vscode代码编辑器

Monaco-Editor 是一个由 Microsoft 开发的 Web 代码编辑器&#xff0c;它是 Visual Studio Code 的浏览器版本。在 Vue 项目中集成 Monaco-Editor 可以提供代码编辑、语法高亮、智能提示等功能 效果&#xff1a; 1、安装使用&#xff0c;最好安装指定版本&#xff0c;我是 vue…

六西格玛项目课题决策权归属:谁来做主更明智?

在实施六西格玛的过程中&#xff0c;一个关键的问题摆在了我们面前&#xff1a;应该由谁决定六西格玛的项目课题&#xff1f;这个问题不仅关系到项目能否顺利实施&#xff0c;更直接影响到企业整体绩效的提升。深圳天行健企业管理咨询公司解析如下&#xff1a; 首先&#xff0…

cocos creator 3.x 手搓背包拖拽装备

项目背景&#xff1a; 游戏背包 需要手动 拖拽游戏装备到 装备卡槽中&#xff0c;看了下网上资料很少。手搓了一个下午搞定&#xff0c;现在来记录下实现步骤&#xff1b; 功能拆分&#xff1a; 一个完整需求&#xff0c;我们一般会把它拆分成 几个小步骤分别造零件。等都造好了…

openresty(Nginx) 隐藏 软件包名称 版本号 升级新版本

1 访问错误或者异常的URL 2 修改配置&#xff0c;重新编译&#xff0c;升级 #修改版本等 vim ./bundle/nginx-1.13.6/src/core/nginx.h #define nginx_version 1013006 #define NGINX_VERSION "1.13.6" #define NGINX_VER "openresty/&q…