【Go】K8s 管理系统项目[Jenkins Pipeline K8s环境–应用部署]

K8s 管理系统项目[Jenkins Pipeline K8s环境–应用部署]

请添加图片描述

1. k8s-plantform-api-Pipeline

考虑到实际工作中前后端可能是不同的同学完成,一般Api部分完成后改动会比较小,web部分改动会比较频繁.于是将api和web分了2个pipeline实现

1.1 GIt仓库

  1. docker目录存放镜像构建相关文件
  2. k8s-plantform-api 存放api部分代码
  3. Jenkinsfile用作pipeline配置
  4. yaml用作生成k8s下k8s-plantform-api相关资源

请添加图片描述

1.1.1 docker目录

1.1.1.1 Dockerfile
# 设置基础镜像
FROM centos:7.9.2009
#FROM harbor.intra.com/k8s-dashboard/api:v1

# 设置作者信息
LABEL maintainer="qiuqin <13917099322@139.com>"

# 创建目录
RUN mkdir -p /data/k8s-plantform

# 复制应用程序
Add ../k8s-plantform/* /data/k8s-plantform

# 安装 Go 和创建目录
RUN cd /etc/yum.repos.d  && \
    rm -f *.repo 
RUN cd /data/k8s-plantform&& \
    curl -o /etc/yum.repos.d/CentOS-Base.repo https://mirrors.aliyun.com/repo/Centos-7.repo && \
    echo "export GO111MODULE=on" >> ~/.profile&& \
    echo "export GOPROXY=https://goproxy.cn" >> ~/.profile&& \
    source ~/.profile&& \
    rpm --import https://mirror.go-repo.io/centos/RPM-GPG-KEY-GO-REPO &&\
    curl -s https://mirror.go-repo.io/centos/go-repo.repo | tee /etc/yum.repos.d/go-repo.repo &&\
    yum install go -y &&\
    go mod tidy

WORKDIR /data/k8s-plantform
ADD ./start.sh /data/k8s-plantform
# 暴露端口
EXPOSE 9091

# 启动应用程序
CMD ["/data/k8s-plantform/start.sh"]
1.1.1.2 start.sh
#!/bin/bash
echo "export GO111MODULE=on" >> ~/.profile
echo "export GOPROXY=https://goproxy.cn" >> ~/.profile
echo ${Db_Ip}>/data/dip.txt
echo ${Db_Port}>/data/dprt.txt
echo ${Db_User}>/data/duser.txt
echo ${Db_Pass}>/data/dpas.txt
dip=`cat /data/dip.txt`
dprt=`cat /data/dprt.txt`
duser=`cat /data/duser.txt`
dpas=`cat /data/dpas.txt`
[ ${Db_Ip} ] && sed -Ei "s/Db_Ip/${dip}/g" /data/k8s-plantform/config/config.go
[ ${Db_Port} ] && sed -Ei "s/Db_Port/${dprt}/g" /data/k8s-plantform/config/config.go
[ ${Db_User} ] && sed -Ei "s/Db_User/${duser}/g" /data/k8s-plantform/config/config.go
[ ${Db_Pass} ] && sed -Ei "s/Db_Pass/${dpas}/g" /data/k8s-plantform/config/config.go
sleep 3
rm -f /data/*.txt
source ~/.profile
cd /data/k8s-plantform
go run main.go

1.1.2 k8s-plantform-api目录

1.1.2.1 config.go

数据库的配置会由configmap传递到容器,并由脚本替换

k8s-plantform-api/config/config.go

package config

import "time"

const (
	//监听地址
	ListenAddr = "0.0.0.0:9091"
	//kubeconfig路径
	Kubeconfig = "config/config"
	//pod日志tail显示行数
	PodLogTailLine = 2000
	//登录账号密码
	AdminUser = "admin"
	AdminPwd  = "123456"

	//数据库配置
	DbType = "mysql"
	DbHost = "Db_Ip"
	DbPort = Db_Port
	DbName = "k8s_dashboard"
	DbUser = "Db_User"
	DbPwd  = "Db_Pass"
	//打印mysql debug sql日志
	LogMode = false
	//连接池配置
	MaxIdleConns = 10               //最大空闲连接
	MaxOpenConns = 100              //最大连接数
	MaxLifeTime  = 30 * time.Second //最大生存时间
)

1.1.3 Jenkinsfile

pipeline {
    agent any
    environment {
        Harbor_Addr = '192.168.31.104'
        Username = 'admin'
        Passwd = 'root123'
        Port = '9091'
    }
    stages {
        stage('代码克隆') {
            steps {
                checkout([$class: 'GitSCM', branches: [[name: '*/main']], extensions: [], userRemoteConfigs: [[credentialsId: '3c67dc4c-db8a-4c78-8278-19cf9eca88ce', url: 'http://192.168.31.199/root/k8s-plantform-api.git']]])
            }
        }
        stage('SonarQube') {
            steps{
                sh '/var/jenkins_home/sonar-scanner/bin/sonar-scanner -Dsonar.projectname=${JOB_NAME} -Dsonar.projectKey=${JOB_NAME} -Dsonar.sources=./ -Dsonar.login=cdd09d5224d8623e4931bcf433274ae996c746f3'
            }
        }
        stage('镜像制作和上传harbor') {
            steps {
                sh '''\\cp -R k8s-plantform-api docker/k8s-plantform
                docker build -t k8sapi:${BUILD_TIMESTAMP} docker/
                docker login -u admin -p root123 192.168.31.104
                docker tag k8sapi:v1 192.168.31.104/k8s-dashboard/k8sapi:${BUILD_TIMESTAMP}
                docker push 192.168.31.104/k8s-dashboard/k8sapi:${BUILD_TIMESTAMP}'''
            }
        }
        stage('把yaml推送到k8s上') {
            steps {
                sshPublisher(publishers: [sshPublisherDesc(configName: 'k8sapi-192.168.31.41', transfers: [sshTransfer(cleanRemote: false, excludes: '', execCommand: '', execTimeout: 120000, flatten: false, makeEmptyDirs: false, noDefaultExcludes: false, patternSeparator: '[, ]+', remoteDirectory: '', remoteDirectorySDF: false, removePrefix: '', sourceFiles: 'yaml/api*.yaml')], usePromotionTimestamp: false, useWorkspaceInPromotion: false, verbose: false)])
            }
        }
        stage('k8s上部署') {
            steps {
                sh 'ssh 192.168.31.41 "sed -i \'s/Tag/\$BUILD_TIMESTAMP/g\' /opt/k8sapi/yaml/api-deployment.yaml && kubectl apply -f /opt/k8sapi/yaml/"'
            }
        }
    }
}

1.1.4 yaml

这里和之前的K8s 管理系统项目36[K8s环境–应用部署]配置一样,略

1.2 Jenkins Pipeline

1.2.1 Pipeline设置

请添加图片描述

1.2.2 Jenkins工作目录配置

[系统管理] --> [系统配置]

请添加图片描述

1.2.3 构建完成

请添加图片描述

1.3 Api测试

请添加图片描述

至此Api的部署已经完成

2. k8s-plantform-web-Pipeline

2.1 Git仓库

  1. docker目录存放镜像构建相关文件
  2. k8s-plantform-web 存放api部分代码
  3. Jenkinsfile用作pipeline配置
  4. yaml用作生成k8s下k8s-plantform-web相关资源

2.1.1 docker目录

2.1.1.1 Dockerfile

# 设置基础镜像
#FROM centos:7.9.2009
FROM 192.168.31.104/baseimages/base_centos:7.9.2009

# 设置作者信息
LABEL maintainer="qiuqin <13917099322@139.com>"

# 创建目录
RUN mkdir -p /data

# 复制应用程序
Add k8s-plantform-web /data/

# 配置yum仓库
#RUN cd /etc/yum.repos.d  && \
#    rm -f *.repo 
#ADD Centos.repo /etc/yum.repos.d/Centos.repo
#RUN yum install gcc gcc-c++ wget -y

# 安装npm		
RUN cd /data && \
	wget https://registry.npmmirror.com/-/binary/node/latest-v16.x/node-v16.15.0-linux-x64.tar.gz && \
	tar xf  node-v16.15.0-linux-x64.tar.gz &&\
	ln -sf /data/node-v16.15.0-linux-x64 /usr/local/node &&\
	ln -sf /data/node-v16.15.0-linux-x64/bin/npm /usr/bin/npm &&\
	export NODE_HOME=/usr/local/node &&\ 
	export PATH=$NODE_HOME/bin:$PATH &&\
	rm -f node-v16.15.0-linux-x64.tar.gz &&\
	cd /data/k8s-plantform-web/ &&\
	npm install

WORKDIR /data/k8s-plantform-web
ADD start.sh /data/k8s-plantform-web/
# 暴露端口
EXPOSE 9090

# 启动应用程序
CMD ["/data/k8s-plantform-web/start.sh"]

2.1.1.2 start.sh

#!/bin/bash
export NODE_HOME=/usr/local/node
export PATH=$NODE_HOME/bin:$PATH
cd /data/k8s-plantform-web/
npm run serve

2.1.2 k8s-plantform-web

2.1.2.1 vue.config.js

k8s-plantform-web/vue.config.js

const { defineConfig } = require('@vue/cli-service')

module.exports = defineConfig({
  devServer:{
    host: '0.0.0.0',  // 监听地址
    port: 9090,           // 监听端口
    open: true,            // 启动后是否打开网页
    allowedHosts: [
      'web.k8s.intra.com',
      '.intra.com'
    ],
  },
  // 关闭语法检测
  lintOnSave: false,
})

2.1.2.2 Config.js

这里将url替换成api的ingress地址

k8s-plantform-web/src/views/common/Config.js

export default {
    //后端接口路径
    loginAuth: 'http://api.k8s.intra.com/api/login',
    k8sWorkflowCreate: 'http://api.k8s.intra.com/api/k8s/workflow/create',
    k8sWorkflowDetail: 'http://api.k8s.intra.com/api/k8s/workflow/detail',
    k8sWorkflowList: 'http://api.k8s.intra.com/api/k8s/workflows',
    k8sWorkflowDel: 'http://api.k8s.intra.com/api/k8s/workflow/del',
    k8sDeploymentList: 'http://api.k8s.intra.com/api/k8s/deployments',
    k8sDeploymentDetail: 'http://api.k8s.intra.com/api/k8s/deployment/detail',
    k8sDeploymentUpdate: 'http://api.k8s.intra.com/api/k8s/deployment/update',
    k8sDeploymentScale: 'http://api.k8s.intra.com/api/k8s/deployment/scale',
    k8sDeploymentRestart: 'http://api.k8s.intra.com/api/k8s/deployment/restart',
    k8sDeploymentDel: 'http://api.k8s.intra.com/api/k8s/deployment/del',
    k8sDeploymentCreate: 'http://api.k8s.intra.com/api/k8s/deployment/create',
    k8sDeploymentNumNp: 'http://api.k8s.intra.com/api/k8s/deployment/numnp',
    k8sPodList: 'http://api.k8s.intra.com/api/k8s/pods',
    k8sPodDetail: 'http://api.k8s.intra.com/api/k8s/pod/detail',
    k8sPodUpdate: 'http://api.k8s.intra.com/api/k8s/pod/update',
    k8sPodDel: 'http://api.k8s.intra.com/api/k8s/pod/del',
    k8sPodContainer: 'http://api.k8s.intra.com/api/k8s/pod/container',
    k8sPodLog: 'http://api.k8s.intra.com/api/k8s/pod/log',
    k8sPodNumNp: 'http://api.k8s.intra.com/api/k8s/pod/numnp',
    k8sDaemonSetList: 'http://api.k8s.intra.com/api/k8s/daemonsets',
    k8sDaemonSetDetail: 'http://api.k8s.intra.com/api/k8s/daemonset/detail',
    k8sDaemonSetUpdate: 'http://api.k8s.intra.com/api/k8s/daemonset/update',
    k8sDaemonSetDel: 'http://api.k8s.intra.com/api/k8s/daemonset/del',
    k8sStatefulSetList: 'http://api.k8s.intra.com/api/k8s/statefulsets',
    k8sStatefulSetDetail: 'http://api.k8s.intra.com/api/k8s/statefulset/detail',
    k8sStatefulSetUpdate: 'http://api.k8s.intra.com/api/k8s/statefulset/update',
    k8sStatefulSetDel: 'http://api.k8s.intra.com/api/k8s/statefulset/del',
    k8sServiceList: 'http://api.k8s.intra.com/api/k8s/services',
    k8sServiceDetail: 'http://api.k8s.intra.com/api/k8s/service/detail',
    k8sServiceUpdate: 'http://api.k8s.intra.com/api/k8s/service/update',
    k8sServiceDel: 'http://api.k8s.intra.com/api/k8s/service/del',
    k8sServiceCreate: 'http://api.k8s.intra.com/api/k8s/service/create',
    k8sIngressList: 'http://api.k8s.intra.com/api/k8s/ingresses',
    k8sIngressDetail: 'http://api.k8s.intra.com/api/k8s/ingress/detail',
    k8sIngressUpdate: 'http://api.k8s.intra.com/api/k8s/ingress/update',
    k8sIngressDel: 'http://api.k8s.intra.com/api/k8s/ingress/del',
    k8sIngressCreate: 'http://api.k8s.intra.com/api/k8s/ingress/create',
    k8sConfigMapList: 'http://api.k8s.intra.com/api/k8s/configmaps',
    k8sConfigMapDetail: 'http://api.k8s.intra.com/api/k8s/configmap/detail',
    k8sConfigMapUpdate: 'http://api.k8s.intra.com/api/k8s/configmap/update',
    k8sConfigMapDel: 'http://api.k8s.intra.com/api/k8s/configmap/del',
    k8sSecretList: 'http://api.k8s.intra.com/api/k8s/secrets',
    k8sSecretDetail: 'http://api.k8s.intra.com/api/k8s/secret/detail',
    k8sSecretUpdate: 'http://api.k8s.intra.com/api/k8s/secret/update',
    k8sSecretDel: 'http://api.k8s.intra.com/api/k8s/secret/del',
    k8sPvcList: 'http://api.k8s.intra.com/api/k8s/pvcs',
    k8sPvcDetail: 'http://api.k8s.intra.com/api/k8s/pvc/detail',
    k8sPvcUpdate: 'http://api.k8s.intra.com/api/k8s/pvc/update',
    k8sPvcDel: 'http://api.k8s.intra.com/api/k8s/pvc/del',
    k8sNodeList: 'http://api.k8s.intra.com/api/k8s/nodes',
    k8sNodeDetail: 'http://api.k8s.intra.com/api/k8s/node/detail',
    k8sNamespaceList: 'http://api.k8s.intra.com/api/k8s/namespaces',
    k8sNamespaceDetail: 'http://api.k8s.intra.com/api/k8s/namespace/detail',
    k8sNamespaceDel: 'http://api.k8s.intra.com/api/k8s/namespace/del',
    k8sPvList: 'http://api.k8s.intra.com/api/k8s/pvs',
    k8sPvDetail: 'http://api.k8s.intra.com/api/k8s/pv/detail',
    k8sTerminalWs: 'ws://localhost:8081/ws',
    //编辑器配置
    cmOptions: {
        // 语言及语法模式
        mode: 'text/yaml',
        // 主题
        theme: 'idea',
        // 显示行数
        lineNumbers: true,
        smartIndent: true, //智能缩进
        indentUnit: 4, // 智能缩进单元长度为 4 个空格
        styleActiveLine: true, // 显示选中行的样式
        matchBrackets: true, //每当光标位于匹配的方括号旁边时,都会使其高亮显示
        readOnly: false,
        lineWrapping: true //自动换行
    }
}

2.1.3 Jenkinsfile

pipeline {
    agent any
    environment {
        Harbor_Addr = '192.168.31.104'
        Username = 'admin'
        Passwd = 'root123'
        Port = '9090'
    }
    stages {
        stage('代码克隆') {
            steps {
                checkout([$class: 'GitSCM', branches: [[name: '*/main']], extensions: [], userRemoteConfigs: [[credentialsId: '3c67dc4c-db8a-4c78-8278-19cf9eca88ce', url: 'http://192.168.31.199/root/k8s-plantform-web.git']]])
            }
        }
        stage('镜像制作和上传harbor') {
            steps {
                sh '''\\cp -R k8s-plantform-web docker/k8s-plantform-web
                docker build -t k8sweb:${BUILD_TIMESTAMP} docker/
                docker login -u admin -p root123 192.168.31.104
                docker tag k8sweb:${BUILD_TIMESTAMP} 192.168.31.104/k8s-dashboard/k8sweb:${BUILD_TIMESTAMP}
                docker push 192.168.31.104/k8s-dashboard/k8sweb:${BUILD_TIMESTAMP}'''
            }
        }
        stage('把yaml推送到k8s上') {
            steps {
                sshPublisher(publishers: [sshPublisherDesc(configName: 'k8sweb-192.168.31.41', transfers: [sshTransfer(cleanRemote: false, excludes: '', execCommand: '', execTimeout: 120000, flatten: false, makeEmptyDirs: false, noDefaultExcludes: false, patternSeparator: '[, ]+', remoteDirectory: '', remoteDirectorySDF: false, removePrefix: '', sourceFiles: 'yaml/web*.yaml')], usePromotionTimestamp: false, useWorkspaceInPromotion: false, verbose: false)])
            }
        }
        stage('k8s上部署') {
            steps {
                sh 'ssh 192.168.31.41 "sed -i \'s/Tag/\$BUILD_TIMESTAMP/g\' /opt/k8sweb/yaml/web-deployment.yaml && kubectl apply -f /opt/k8sweb/yaml/"'
            }
        }
    }
}

2.1.4 yaml

这里和之前的K8s 管理系统项目36[K8s环境–应用部署]配置一样,略

2.2 Jenkins Pipeline

2.2.1 Pipeline 设置

请添加图片描述

2.2.2 Jenkins工作目录配置

[系统管理] --> [系统配置]

请添加图片描述

2.2.3 构建完成

请添加图片描述

2.3 Web测试

2.3.1 登录页面

请添加图片描述

2.3.2 主要功能测试

2.3.2.1 集群状态

请添加图片描述

2.3.2.2 Pods

请添加图片描述

2.3.2.3 Service

请添加图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

简介虚拟地址空间:保障进程间独立性的机制

我们知道&#xff0c;进程之间是相互独立的&#xff0c;在操作系统级别中&#xff0c;一个进程所执行的程序无法直接访问另一个进程所执行的内存区域&#xff08;即实现进程间通信比较困难&#xff09;&#xff1b;一个进程运行的失败也不会影响其它进程的运行。这使我们的操作…

vue编程方法

1&#xff0c;app.vue 其中的moundted只是被执行一次。 系统中所有的组件都放到app。vue文件中。放到根组件中的只是被执行一次的代码可以放到main.js中码&#xff1f; 不可以&#xff0c;因为main文件只是一个js文件不是一个组件。组件中的一些属性不能被使用。比如&#xff…

VS Code上搭建Vue开发环境超详细教程

这篇关于在Visual Studio Code上搭建vue开发环境的超详细教程手把手教会你! 首先在Visual Studio Code上搭建vue开发环境有几个步骤&#xff1a; 1、下载安装node.js 2、安装npm 3、安装cnpm 4、安装vue/cli脚手架 5、创建vue项目 6、运行vue项目 1.下载安装node.js 地址&…

鸟哥的Linux私房菜 正则表示法与文件格式化处理

第十一章、正则表示法与文件格式化处理 https://linux.vbird.org/linux_basic/centos7/0330regularex.php 简体版 http://cn.linux.vbird.org/linux_basic/0330regularex.php 11.2.2 grep的一些高级选项 例题一、搜索特定字符串 例题二、利用中括号 [] 来搜寻集合字符 例题四…

8个python自动化脚本提高打工人幸福感~比心~

人生苦短&#xff0c;我用Python 最近有许多打工人都找我说打工好难 每天都是执行许多重复的任务&#xff0c; 例如阅读新闻、发邮件、查看天气、打开书签、清理文件夹等等&#xff0c; 使用自动化脚本&#xff0c;就无需手动一次又一次地完成这些任务&#xff0c; 非常方便…

蓝桥杯嵌入式RTC实时时钟

文章目录 前言一、RTC是什么二、cubemx的配置三、函数的使用总结前言 本篇文章将给大家介绍RTC实时时钟。 一、RTC是什么 STM32的实时时钟RTC是一个独立的定时器,RTC时钟内部依靠BCD码计数。RTC实时时钟提高时钟、闹钟、日历功能。RTC功耗较低,可以使用在低功耗设备上。 …

Redis为什么选择单线程?Redis为什么这么快?

目录专栏导读一、Redis版本迭代二、Redis4.0之前为什么一直采用单线程&#xff1f;三、Redis6.0引入多线程四、Redis主线程和IO线程是如何完成请求的&#xff1f;1、服务端和客户端建立socket连接2、IO线程读取并解析请求3、主线程执行请求命令4、IO线程会写回socket和主线程清…

DM8:LINUX环境安装DM8数据库安装条件--GLIBC版本要求

DM8&#xff1a;LINUX环境安装DM8数据库安装条件--GLIBC版本要求环境介绍1 检查 GLIBC 版本号2 /tmp 临时目录空间要等于或大于2GB3 报错截图3.1 导入授权报错3.2 设置时区报错3.3 DmAPService启动失败3.4 初始化实例报错4 更多达梦数据库使用经验环境介绍 在LINUX环境安装达梦…

一线大厂软件测试常见面试题1500问,背完直接拿捏面试官,

三、测试理论 3.1 你们原来项目的测试流程是怎么样的? 我们的测试流程主要有三个阶段&#xff1a;需求了解分析、测试准备、测试执行。 1、需求了解分析阶段 我们的SE会把需求文档给我们自己先去了解一到两天这样&#xff0c;之后我们会有一个需求澄清会议&#xff0c; 我…

基于Springboot实现口腔牙诊所网站平台【源码+论文】

基于Springboot实现口腔牙诊所网站平台【源码论文】开发语言&#xff1a;Java 框架&#xff1a;springboot JDK版本&#xff1a;JDK1.8 服务器&#xff1a;tomcat7 数据库&#xff1a;mysql 5.7 数据库工具&#xff1a;Navicat11 开发软件&#xff1a;eclipse/myeclipse/idea M…

整合SpringCache

整合SpringCache 1、引入依赖cache还有redis <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-cache</artifactId> </dependency>2、写配置 spring:cache:type: redis3、测试使用缓存 Cache…

大数据现在找工作难么

大数据行业工作好找还是难找不是光靠嘴说出来的结合实际&#xff0c;看看市场上的招聘需求和岗位要求就大致知道了 要想符合企业用人规范&#xff0c;学历&#xff0c;工作经验&#xff0c;掌握技能都是非常重要的~ 先来看几个招聘网站的报告数据&#xff1a; Boss直聘发布的…

【蓝桥杯】 C++ 数字三角形 动态规划 ⭐⭐

文章目录题目描述输入描述输出描述实现代码解题思路注意点知识点题目描述 上图给出了一个数字三角形。从三角形的顶部到底部有很多条不同的路径。对于每条路径&#xff0c;把路径上面的数加起来可以得到一个和&#xff0c;你的任务就是找到最大的和&#xff08;路径上的每一步…

Python嵌套函数(Nested function)和闭包(closure)

Python嵌套函数&#xff08;Nested function&#xff09;和闭包&#xff08;closure&#xff09; 闭包&#xff08;closure&#xff09;是建立在嵌套函数基础上的&#xff0c;是一种特殊的嵌套函数结构。 先看嵌套函数&#xff08;Nested function&#xff09;。 Python允许…

gan实战(DCGAN、)

一、DCGAN 1.1 参数 &#xff08;1&#xff09;输入&#xff1a;会被放缩到6464 &#xff08;2&#xff09;输出&#xff1a;6464 &#xff08;3&#xff09;数据集&#xff1a; 1.2 实现 import glob import torch from PIL import Image from torch import nn from torch.u…

web前端框架——Vue的特性

目录 前言&#xff1a; 一.vue 二.特性 1.轻量级 2.数据绑定 3.指令 4.插件 三.比较Angular 、React 、Vue 框架之间的比较 1. Angular Angular的优点&#xff1a; 2. React React 的优点&#xff1a; 3.vue 3.Vue的优点&#xff1a; 前言&#xff1a; 本篇文章…

有效的括号长按键入验证外星语词典字符的最短距离用栈实现队列

有效的括号来源&#xff1a;杭哥20. 有效的括号 - 力扣&#xff08;LeetCode&#xff09;bool isValid(char * s) {int szstrlen(s);char stack[sz];int k0;for (int i0;i<sz;i){if (s[i]( || s[i][ || s[i]{){stack[k]s[i];}else{if (k0){return false;}else if (s[i]} &am…

C++ Qt自建网页浏览器

C Qt自建网页浏览器如需安装运行环境或远程调试&#xff0c;见文章底部个人QQ名片&#xff0c;由专业技术人员远程协助&#xff01;前言这篇博客针对<<C Qt自建网页浏览器>>编写代码&#xff0c;代码整洁&#xff0c;规则&#xff0c;易读。 学习与应用推荐首选。文…

VSCode嵌入式开发环境搭建

Vscode开发环境搭建 看这个链接就可以了&#xff0c;后面下载调试有点问题看下3.3。 在VSCode上部署STM32F1的开发环境 1. MXCube配置工程生成Makefile文件 借助正确的编译工具链进行编译&#xff0c; 2. 编译工具链搭建 编译工具链使用GCC的ARM版本 arm-none-eabi-gcc &am…

servlet

✏️作者&#xff1a;银河罐头 &#x1f4cb;系列专栏&#xff1a;JavaEE &#x1f332;“种一棵树最好的时间是十年前&#xff0c;其次是现在” 目录Servlet 是什么第一个 Servlet 程序1.创建项目2.引入依赖3.创建目录结构4.编写代码5.打包程序6.部署程序7.验证更方便的部署方…