如何使用 Github Action 管理 Issue

本文作者为 360 奇舞团前端开发工程师 Daryl

前言

8dff5425a37a841be58635a469034ee2.png很多小伙伴打开 github 上的仓库都只使用Code查看代码,或者只是把 github 当成一个代码仓库,但是 github 还提供了很多好用的功能。

其中,GitHub Action就是一个很好用的功能,本文将通过几个管理Issue的示例带大家了解GitHub Action:

什么是 Github Action

github 给所有用户都提供了临时可用的虚拟机, 我们通过创建 github action 工作流来使用这个虚拟机. 我们可以使用它来实现自动化部署、自动化测试、代码检查、管理 Issues...

使用步骤

在学习之前还需要准备一些资料:

  1. Github Action文档

  2. 官方仓库中有很多可以复用的 Action, 通过uses字段引用就可以直接使用了。

  3. 阮一峰的YAML 教程;

也推荐大家使用Vscode GitHub Action插件,这个插件在登录后可以用来做语法校验,还能查看运行过的记录。

除了这些资料之外还有些基础概念需要了解:

  1. 事件: 在工作流中可以监听 github 的一些事件, 在事件触发后执行我们定义的工作流;

  2. 上下文: github 上下文包含有关工作流运行和触发运行的事件的信息,可以读取环境变量中的大多数 github 上下文数据,并允许我们通过变量访问这些数据。

  3. 变量: 变量提供了一种存储和重用非敏感配置信息的方法。 可以将任何配置数据(如编译器标志、用户名或服务器名称)存储为变量。 变量在运行工作流的运行器计算机上插值。 在操作或工作流步骤中运行的命令可以创建、读取和修改变量。

  4. 表达式: 可以使用表达式来运算工作流程文件中的变量。

  5. 秘钥: 普通变量中存储的信息并不安全,很容易泄露,一些需要保密的信息就可以存储到秘钥中。

如果不想去从头学习yml语法, 可以先了解一些yml的基础用法:

  • 大小写敏感

  • 使用缩进表示层级关系

  • 缩进时不允许使用 Tab 键,只允许使用空格。

  • 缩进的空格数目不重要,只要相同层级的元素左侧对齐即可


下面开始介绍 GitHub Action 的用法

使用 github Action ,第一步需要在项目根目录下创建.github/workflows文件夹, 所有的工作流文件都要放到这个文件夹,当事件触发时会自动执行;fd320390b995a60d40181b200cef1eed.png

大家可以通过这个 workflow 文件示例来简单了解下各个字段的用处:

name: build # workflow的名称,缺省时会使用文件名
on: # workflow监听事件
  push # 具体的事件
    branches: # 在这些分支上才会触发
      - main
      - 'mona/octocat'
      - 'releases/**'
jobs: # 执行的工作的集合
  build: # ‘build’是一个自定义的工作的id<job_id>
    name: rele # 当前工作的名称
    runs-on: ubuntu-latest # 因为工作实际是运行在虚拟机上的,runs-on就是指定虚拟机的版本
    steps: # steps是步骤的集合
    - name: checkout # name是指定当前工作的名称 在workflow(工作流)文件的steps中,每个用‘-’代替缩进视为一个步骤的开始
      uses: actions/checkout@v2 # uses字段是选择一个可以直接复用的action,并且在github action store中的action可以直接使用,不需要下载
    - name: setting env
      id: setting # id 是步骤的唯一标识符,可以使用 id 在上下文中引用该步骤
      env: # 设置环境变量
        NODEV: 18
      run: echo "nodev=$NODEV" >> $GITHUB_OUTPUT # run字段会在命令行执行一条命令,这个命令是将"nodev=18"写入到$GITHUB_OUTPUT,这样可以为'output'添加test属性值为test1,详情参考
    - name: addnode
      uses: actions/setup-node@v3 # 使用node环境
      with: # 为‘uses’使用的action传递参数
        node-version: ${{steps.setting.output.nodev}} # 使用上面设置的变量

在编写工作流文件之前有两件事要做:

一. 可以在 github 或者 github action 的仓库里查找公用 action.这样可以减少很多工作量:

常用的 action 有:

  • checkout: 帮我们自动把项目克隆到虚拟机上

  • Setup Node: 自动安装 node

  • issues-helper: 辅助处理 issues

二. 如果没有设置 action 的读写权限,第一次运行会报错:27be5e8c4bf2d81957c5d6cefbbd7ec4.png设置权限:52bbe2c1d8090cd1dbc378dd8d5caeba.png

再次运行:

2346f4f49bc77397a84bf140f4c61a7a.png
Alt text

下面通过一些具体示例给大家介绍一些Github Action的用法:

  • 检查 issues 格式是否规范, 并关闭不规范 issue

    在开源项目中,很多人都会提一些 issue。为了方便开发者查看,可以上传 issue 模版,并且根据模版校验 issue 格式,然后关闭不符合模板格式的 issue。

    在这个示例主要介绍脚本执行怎么向github上下文注入变量,我们可以参考github 设置环境变量的教程.

    这个脚本可以配合.github/ISSUE_TEMPLATE文件夹中的 issue 模板使用的,将.md文件放入这个文件夹中就可以作为 Issue 的默认模板;

    9c7a80f20f429e4aa7c3852d33033560.png
    Alt text

    选择 Bug 提交将使用这个模板:

    ---
    name: Bug 提交
    about: 使用此模板来提交一个 bug。
    ---
    
    # BUG 提交
    
    ## 描述该错误
    
    简明扼要地描述一下这个错误是什么。
    可以添加屏幕截图以帮助解释你的问题。
    
    ## 复现
    
    这个 BUG 的复现步骤:
    
    1. ...
    2. ...
    3. ...
       ...
       或者添加录屏链接
    
    ## 运行结果
    
    预期的结果:
    ... ...
    实际的结果:
    ... ...
    
    ## 运行环境信息
    
    - Device: [e.g. 设备名称]
    - OS: [e.g. 操作系统]
    - Browser [e.g. 浏览器]
    - Version [e.g. 浏览器版本]
    
    ## 其它
    
    在此添加关于问题的任何其它信息。

    选择功能请求将使用这个模板:

    ---
    name: 功能请求
    about: 使用此模板来提交一个功能请求。
    ---
    
    # 功能请求
    
    ## 你期望添加什么样的功能?
    
    你期望新增功能的描述,或者示例链接。

    检查 issue 格式的脚本:

    // action_script/lintIssue.js
    const issueText = process.env.ISSUE
    
    const textSplit = issueText
      .split(
        `
    `
      )
      .map((str) => str.replace('\r', ''))
    const bugHandle = () => {
      // 缺少错误描述
      const desIndex = textSplit.indexOf('## 描述该错误')
      if (desIndex === -1) {
        console.log('ISSUE_CHECK_RESULT=unqualified')
        console.log('ISSUE_CHECK_REPLY=缺少错误描述')
        return
      }
      const repeatIndex = textSplit.indexOf('## 复现')
      if (repeatIndex === -1) {
        console.log('ISSUE_CHECK_RESULT=unqualified')
        console.log('ISSUE_CHECK_REPLY=缺少复现步骤')
        return
      }
      const desContent = textSplit
        .slice(desIndex + 1, repeatIndex)
        .join('')
        .replaceAll(' ', '')
      if (!desContent) {
        console.log('ISSUE_CHECK_RESULT=unqualified')
        console.log('ISSUE_CHECK_REPLY=缺少错误描述')
        return
      }
      // 缺少复现步骤
      const runResultIndex = textSplit.indexOf('## 运行结果')
      if (runResultIndex === -1) {
        console.log('ISSUE_CHECK_RESULT=unqualified')
        console.log('ISSUE_CHECK_REPLY=缺少运行结果')
        return
      }
      const repeatContent = textSplit
        .slice(repeatIndex + 1, runResultIndex)
        .join('')
        .replaceAll(' ', '')
      if (!repeatContent) {
        console.log('ISSUE_CHECK_RESULT=unqualified')
        console.log('ISSUE_CHECK_REPLY=缺少复现步骤')
        return
      }
      // 运行结果
      const envIndex = textSplit.indexOf('## 运行环境信息')
      if (envIndex === -1) {
        console.log('ISSUE_CHECK_RESULT=unqualified')
        console.log('ISSUE_CHECK_REPLY=缺少运行环境信息')
        return
      }
      const resContent = textSplit
        .slice(runResultIndex + 1, envIndex)
        .join('')
        .replaceAll(' ', '')
      if (!resContent) {
        console.log('ISSUE_CHECK_RESULT=unqualified')
        console.log('ISSUE_CHECK_REPLY=缺少运行结果')
        return
      }
      // 运行环境信息
      const otherIndex = textSplit.indexOf('## 其他') || textSplit.length + 1
      const envContent = textSplit.slice(envIndex + 1, otherIndex)
      const envReg = /\[e\.g\..{5,}\]/
      let hasDevice = false
      let hasOS = false
      let hasBrowser = false
      let hasVersion = false
      let errMsg = ''
      envContent.forEach((str) => {
        if (str.match(envReg)) {
          if (str.includes('Device')) {
            hasDevice = true
          } else if (str.includes('OS')) {
            hasOS = true
          } else if (str.includes('Browser')) {
            hasBrowser = true
          } else if (str.includes('Version')) {
            hasVersion = true
          }
        }
      })
    
      if (!hasDevice) {
        errMsg += '缺少设备名称;'
      } else if (!hasOS) {
        errMsg += '缺少操作系统名称;'
      } else if (!hasBrowser) {
        errMsg += '缺少浏览器名称;'
      } else if (!hasVersion) {
        errMsg += '缺少浏览器版本;'
      }
      if (errMsg) {
        console.log('ISSUE_CHECK_RESULT=unqualified')
        console.log('ISSUE_CHECK_REPLY=' + errMsg)
      } else {
        console.log('ISSUE_CHECK_RESULT=pass')
      }
    }
    
    const featureHandle = () => {
      // 缺少错误描述
      const desIndex = textSplit.indexOf('## 你期望添加什么样的功能?')
      const desContent = textSplit
        .slice(desIndex + 1, textSplit.length)
        .join('')
        .replaceAll(' ', '')
        .replaceAll('\n', '')
      if (desIndex === -1 || desIndex === textSplit.length - 1 || !desContent) {
        console.log('ISSUE_CHECK_RESULT=unqualified')
        console.log('ISSUE_CHECK_REPLY=缺少功能描述')
        return
      } else {
        console.log('ISSUE_CHECK_RESULT=pass')
      }
    }
    if (textSplit[0] === '# BUG 提交') {
      console.log('ISSUE_CHECK_TYPE=bug')
      bugHandle()
    } else if (textSplit[0] === '# 功能请求') {
      console.log('ISSUE_CHECK_TYPE=feature')
      featureHandle()
    } else {
      console.log('ISSUE_CHECK_TYPE=invalid')
      console.log('ISSUE_CHECK_RESULT=unqualified')
      console.log('ISSUE_CHECK_REPLY=这不是一个BUG或者功能请求')
    }

    工作流示例文件:

    # .github/workflows/close-non_standard-issue.yml
      name: close non-standard issues
    
    on:
      issues:
        types: [opened, edited] # issue 打开或者编辑后
    
    jobs:
      close-issue:
        runs-on: ubuntu-latest
        env:
          ISSUE: ${{ github.event.issue.body }}
        steps:
          - name: 'checkout'
            uses: actions/checkout@v3
          - name: Setup node
            uses: actions/setup-node@v3
            with:
              node-version: 18
              registry-url: https://registry.npmjs.com/
          - name: lint sh
            run: node ./action_script/lintIssue.js  >> "$GITHUB_ENV" # 设置自定义github变量
          - name: add-label
            uses: actions-cool/issues-helper@v3
            with:
              actions: 'add-labels'
              token: ${{ secrets.GITHUB_TOKEN }}
              issue-number: ${{ github.event.issue.number }}
              labels: ${{env.ISSUE_CHECK_TYPE}}
    
          - name: 'close-issue'
            if: ${{env.ISSUE_CHECK_RESULT == 'unqualified'}}
            uses: actions-cool/issues-helper@v3
            with:
              actions: 'close-issue'
              token: ${{ secrets.GITHUB_TOKEN }}
              body: |
                Hello @${{ github.event.issue.user.login }}.你的Issue因为下面的原因被关闭了:
                ${{env.ISSUE_CHECK_REPLY}}
  • 如果是做一些简单的校验可以使用 actions-cool/issues-helper@v3 中的 check-issue:

    这个示例主要是演示运算符的使用:

    name: add label of non_standard Issue
    
    on:
      issues:
        types: [opened, edited]
    
    jobs:
      check-issue:
        runs-on: ubuntu-latest
        steps:
          - id: check-issue # 必须使用id,不然不能访问到运行结果
            uses: actions-cool/issues-helper@v3
            with:
              actions: "check-issue"
              token: ${{ secrets.GITHUB_TOKEN }}
              issue-number: ${{ github.event.issue.number }}
              title-includes: "【,BUG,】" # 标题包含['【','BUG','】']
              body-includes: "问题描述,问题复现步骤" # 内容包含 ['问题描述','问题复现步骤']
          - name: add-label
            uses: actions-cool/issues-helper@v3
            if: ${{steps.check-issue.outputs.check-result == 'true'}} # 如果判断条件是 'true', 继续运行
            with:
              actions: "add-labels"
              token: ${{ secrets.GITHUB_TOKEN }}
              issue-number: ${{ github.event.issue.number }} # 当前Issue的编号
              labels: "bug" # 添加标签: 'bug'
  • 关闭缺少复现步骤的 Issue

    有些 Issue 虽然通过了格式检查,但是还会缺少一些步骤,或者不能复现。我们可以手动添加 label,并且在三天不活跃的情况下自动关闭。

    这个示例主要是介绍怎么执行定时任务:

    # .github/workflows/close-inactive-issue.yml
    name: close inactive issue
    on:
      schedule:
        - cron: "00 12,00,18 * * *" # 在每天标准时间的12:00/00:00/18:00 执行
    
    jobs:
      check-inactive-info:
        runs-on: ubuntu-latest
        steps:
          - name: need reproduction
            uses: actions-cool/issues-helper@v3
            with: # 关闭有label是'need reproduction',并且三天不活跃的Issue
              actions: "close-issues"
              token: ${{ secrets.GITHUB_TOKEN }}
              labels: "need reproduction"
              inactive-day: 3
  • 添加标签时自动评论、关闭 BUG

    有些常用的评论内容,或者用的比较多的操作,每次使用都比较繁琐。我们可以在添加标签的时候自动评论,或者执行对应的操作。比如:自动评论欢迎词,关闭手动筛选出来不符合 issue 模版的 issue 并告诉用户原因。

    这个示例是介绍在.yml文件中长文本的写法

    name: Issue Labeled
    # .github/workflows/issue-labeled.yml
    # 在issue添加标签后回复对应的评论
    on:
      issues:
        types: [labeled]
    
    jobs:
      reply-labeled:
        runs-on: ubuntu-latest
        steps:
          # 需要帮助
          - name: contribution welcome
            if: github.event.label.name == 'help wanted'
            uses: actions-cool/issues-helper@v3
            with:
              actions: "create-comment"
              issue-number: ${{ github.event.issue.number }}
              body: |
                你好 @${{ github.event.issue.user.login }},我们完全同意你的提议/反馈,欢迎直接在此仓库 [创建一个 Pull Request](https://github.com/NI-Web-Infra-Team/vue3-template/pulls) 来解决这个问题。请将 Pull Request 发到 `dev` 分支,提供改动所需相应的 changelog、TypeScript 定义、测试用例、文档等,并确保 CI 通过,我们会尽快进行 Review,提前感谢和期待您的贡献。
            # 补充复现流程
          - name: need reproduction
            if: github.event.label.name == 'need reproduction'
            uses: actions-cool/issues-helper@v3
            with:
              actions: "create-comment"
              issue-number: ${{ github.event.issue.number }}
              body: |
                你好 @${{ github.event.issue.user.login }}, 我们需要你提供一个在线的重现实例以便于我们帮你排查问题。可以通过点击 [此处](https://codepen.io/pen/) 创建或者提供一个最小化的 GitHub 仓库。3 天内未跟进此 issue 将会被自动关闭。
    
          - name: invalid
            if: github.event.label.name == 'invalid'
            uses: actions-cool/issues-helper@v3
            with:
              actions: "create-comment, close-issue"
              issue-number: ${{ github.event.issue.number }}
              body: |
                你好 @${{ github.event.issue.user.login }},为了能够进行高效沟通,我们对 issue 有一定的格式要求,你的 issue 因为不符合要求而被自动关闭。你可以通过模板来创建 issue 以方便我们定位错误。谢谢配合!
  • 使用 ftp 自动部署

    如果有在公网的服务器,可以在编译完成后打包发送到自己的服务器实现自动部署。 在 github 中搜索 ftp action, 第一个 SamKirkland/FTP-Deploy-Action 就是将 github 项目部署到 ftp 服务器的:

    最后这个示例介绍的是怎么设置秘钥:

    name: ftp send file
    
    on: workflow_dispatch # 手动部署
    
    jobs:
      web-deploy:
        name: Deploy
        runs-on: ubuntu-latest
        steps:
          - name: Get latest code
            uses: actions/checkout@v3 # 拉取项目代码
          - name: Setup node
            uses: actions/setup-node@v3 # 安装node
            with:
              node-version: 18
              registry-url: https://registry.npmjs.com/
          - name: install package
            run: npm i # 安装项目依赖
          - name: build
            run: npm run build # 编译
          - name: pack
            run: zip -q -r dist.zip ./dist # 打包成zip
          - name: Sync files
            uses: SamKirkland/FTP-Deploy-Action@v4.3.4 # 上传到ftp服务器
            with:
              server: 10.52.0.x # 服务器地址
              username: testusername # 服务器用户名
              password: ${{ secrets.ftp_password }} # 服务器密码
    • 设置在 action 中可以访问的秘钥

      服务器密码secrets.ftp_password,就是在 github 中设置的秘钥:b3d99db7d108040500fff6d37fa69e6e.png867d509abfe9593c34cc56843d76a50d.pngb7c069e7c56c2492b06faab1c25cedc2.png

结语

通过使用 Github Action 来管理 Issue,可以有效的提高生产力和效率,在自动化、协作、代码质量管理等方面都有提升,并帮助我们更好地组织和管理问题。文章篇幅有限,我们暂且介绍到这里,感兴趣的小伙伴们可以再自行探索。

- END -

关于奇舞团

奇舞团是 360 集团最大的大前端团队,代表集团参与 W3C 和 ECMA 会员(TC39)工作。奇舞团非常重视人才培养,有工程师、讲师、翻译官、业务接口人、团队 Leader 等多种发展方向供员工选择,并辅以提供相应的技术力、专业力、通用力、领导力等培训课程。奇舞团以开放和求贤的心态欢迎各种优秀人才关注和加入奇舞团。

bcac0ea2df12f76a06b72b602213b300.png

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

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

相关文章

基于 Amazon EKS 搭建开源向量数据库 Milvus

一、前言 生成式 AI&#xff08;Generative AI&#xff09;的火爆引发了广泛的关注&#xff0c;也彻底点燃了向量数据库&#xff08;Vector Database&#xff09;市场&#xff0c;众多的向量数据库产品开始真正出圈&#xff0c;走进大众的视野。 根据 IDC 的预测&#xff0c;…

python采集小破站视频弹幕

嗨喽~大家好呀&#xff0c;这里是魔王呐 ❤ ~! python更多源码/资料/解答/教程等 点击此处跳转文末名片免费获取 环境使用]: Python 3.8 Pycharm模块使用]: import requests 数据请求 import jieba 分词 import wordcloud 词云 import re 正则通过爬虫程序采集数据 分析数…

微信小程序渲染的富文本里面除了img标签外什么都没有,该如何设置img的大小

微信小程序富文本渲染&#xff1a; <rich-text nodes"{{content}}"style"{{style}}" ></rich-text> content是接口得到的值 let cont object.contentlet a cont.replace(/<img/gi,<img style"max-width:94%;height:auto;margi…

Kafka快速入门

文章目录 Kafka快速入门1、相关概念介绍前言1.1 基本介绍1.2 常见消息队列的比较1.3 Kafka常见相关概念介绍 2、安装Kafka3、初体验前期准备编码测试配置介绍 bug记录 Kafka快速入门 1、相关概念介绍 前言 在当今信息爆炸的时代&#xff0c;实时数据处理已经成为许多应用程序和…

python科研绘图:P-P图与Q-Q图

目录 什么是P-P图与Q-Q图 分位数 百分位数 Q-Q图步骤与原理 Shapiro-Wilk检验 绘制Q-Q图 绘制P-P图 什么是P-P图与Q-Q图 P-P图和Q-Q图都是用于检验样本的概率分布是否服从某种理论分布。 P-P图的原理是检验实际累积概率分布与理论累积概率分布是否吻合。若吻合&#xf…

HPC 工作负载管理 —— IBM Spectrum LSF Suite

全面的工作负载管理解决方案&#xff0c;通过增强用户和管理员体验以及实现规模性能来简化 HPC。 IBM Spectrum LSF Suites 是面向分布式高性能计算 (HPC) 的工作负载管理平台和作业调度程序。基于 Terraform 的自动化现已可用&#xff0c;该功能可在 IBM Cloud 上为基于 IBM …

c语言:解决数组元素右旋问题,时间复杂度O(N)

题目&#xff1a; 给一个数组&#xff0c;如【1&#xff0c;2&#xff0c;3&#xff0c;4&#xff0c;5&#xff0c;6&#xff0c;7】,k3。 要求得到新数组【5&#xff0c;6&#xff0c;7&#xff0c;1&#xff0c;2&#xff0c;3&#xff0c;4】。 方法一&#xff0c;思路和…

GPT模型支持下的Python-GEE遥感云大数据分析、管理与可视化技术及多领域案例实践应用

随着航空、航天、近地空间等多个遥感平台的不断发展&#xff0c;近年来遥感技术突飞猛进。由此&#xff0c;遥感数据的空间、时间、光谱分辨率不断提高&#xff0c;数据量也大幅增长&#xff0c;使其越来越具有大数据特征。对于相关研究而言&#xff0c;遥感大数据的出现为其提…

EtherCAT从站EEPROM组成信息详解(2):字8-15产品标识区

0 工具准备 1.EtherCAT从站EEPROM数据&#xff08;本文使用DE3E-556步进电机驱动器&#xff09;1 字8-字15产品标识区 1.1 产品标识区组成规范 对于不同厂家和型号的从站&#xff0c;主站是如何区分它们的呢&#xff1f;这就要提起SII的字8-字15区域存储的产品标识&#xff…

计算机视觉:人脸识别与检测

目录 前言 识别检测方法 本文方法 项目解析 完整代码及效果展示 前言 人脸识别作为一种生物特征识别技术&#xff0c;具有非侵扰性、非接触性、友好性和便捷性等优点。人脸识别通用的流程主要包括人脸检测、人脸裁剪、人脸校正、特征提取和人脸识别。人脸检测是从获取的图…

电磁场与电磁波part1--矢量分析

目录 1、方向导数 2、散度定理&#xff08;高斯定理&#xff09; 3、散度与旋度的比较 4、旋度定理&#xff08;斯托克斯定理&#xff09; 5、关于点乘、叉乘、梯度、散度、旋度的计算 ~~~~~~~~~~~~~~~~~~~~~~~~ 确认过眼神&#xff0c;是我看不懂的 ~~~~~~~~~~~~~~~~…

5. HTML中常用标签

5. html常用标签 5.1 标签语义 学习标签是有技巧的&#xff0c;重点是记住每个标签的语义。简单理解就是指标签的含义。即这个标签是用来干嘛的。 根据标签的语义&#xff0c;在合适的地方给一个最为合理的标签。可以让页面结构给清晰。 5.2 标题标签 <h1>-<h6>…

【C++ std::max_element std::min_element std::minmax_element】

一 、std::max_element 寻找范围 [first, last) 中的最大元素。 (1) 用 operator< 比较元素。 (3) 用给定的二元比较函数 comp 比较元素。 (2),(4) 同 (1,3) &#xff0c;但按照 policy 执行。这些重载仅若 std::is_execution_policy_v<std::decay_t > (C20 前)std:…

【C++】泛型编程 ④ ( 函数模板 与 普通函数 调用规则 | 类型自动转换 | 类型自动转换 + 显式指定泛型类型 )

文章目录 一、普通函数 与 函数模板 的调用规则 - 类型自动转换1、函数模板和重载函数2、类型自动转换3、代码示例 - 类型自动转换 二、普通函数 与 函数模板 的调用规则 - 类型自动转换 显式指定泛型类型1、类型自动转换 显式指定泛型类型2、代码示例 - 类型自动转换 显式指…

string的简单操作

目录 string的接口说明 构造 constructor operator 迭代器操作 begin( )和end( ) rbegin( ) 和 rend( ) 范围for和迭代器的关系 范围for 迭代器 容量 size lengtn max_size resize capacity reserve clear empty string类的元素访问 operator[ ] at fro…

构造函数和初始化列表的关系和区别【详解】

构造函数和初始化列表关系和区别&#xff0c;以及为什么有初始化列表&#xff0c;和它的好处 一、构造函数和初始化列表的关系和区别二、为什么有初始化列表三、使用初始化列表的好处 一、构造函数和初始化列表的关系和区别 百度百科这样定义初始化列表&#xff1a;与其他函数…

基于STM32的LoRaWAN无线通信网络设计与实现

LoRaWAN (Long Range Wide Area Network) 是一种低功耗的无线通信技术&#xff0c;用于构建广域物联网。本篇文章将介绍基于STM32微控制器的LoRaWAN无线通信网络的设计与实现&#xff0c;并提供相应的代码示例。 概述 LoRaWAN的无线通信技术采用低功耗长距离传输&#xff0c;…

STM32 独立看门狗

目录 1.独立看门狗介绍 2.独立看门狗本质 3.独立看门狗框图​编辑 4.独立看门狗时钟 5.预分频寄存器&#xff08;IWDG_PR)​编辑 6.重装载寄存器&#xff08;IWDG_RLR) 7.键寄存器&#xff08;IWDG_KR) 8.独立看门狗实验和代码示例 9.独立看门狗和窗口看门狗的异同点 …

【原创】java+swing+mysql个人日记管理系统设计与实现

摘要&#xff1a; 个人日记管理系统是一个可以记录、管理、存储和检索个人日记的应用程序。这个系统允许用户创建和管理多个日记帐户&#xff0c;每个帐户都可以有多个日记条目。用户可以随时添加、编辑或删除日记条目&#xff0c;并可以将这些条目按照主题或其他标准进行分类…

python科研绘图:绘制X-bar图

目录 1.X-bar 图的基本概念 2.X-bar 图的绘制过程 3.X-bar 图的优势 4.X-bar 图的绘制 1.X-bar 图的基本概念 X-bar控制图是一种统计工具&#xff0c;用于监控和控制生产过程中的质量变量。它是过程能力分析和统计过程控制&#xff08;SPC&#xff0c;Statistical Process…