go 库 Cobra 现代化的命令行框架

news2024/12/30 3:16:56

go 库 Cobra 现代化的命令行框架

文章目录

  • go 库 Cobra 现代化的命令行框架
    • 1. 简介
    • 2. 主要功能
    • 3. 应用举例
    • 4. Cobra 安装
    • 5. 使用 Cobra 库创建命令
      • 5.1 创建 rootCmd
      • 5.2 创建 main.go
      • 5.3 添加命令
      • 5.4 编译并运行
    • 6. 特性
      • 6.1 使用标志
      • 6.2 非选项参数验证
      • 6.3 PreRun and PostRun Hooks
    • 7. cobra-cli 命令
      • 7.1 安装
      • 7.2 cobra-cli 初始化
      • 7.3 向项目添加命令

1. 简介

Cobra 既是一个可以创建强大的现代 CLI 应用程序的库,也是一个可以生成应用和命令文件的程序。有许多大型项目都是用 Cobra 来构建应用程序的,例如 KubernetesDockeretcdRktHugo 等。

Cobra 建立在 commands、arguments 和 flags 结构之上。

  • commands 代表命令,
  • arguments 代表非选项参数,
  • flags 代表选项参数(也叫标志)。

一个好的应用程序应该是易懂的,用户可以清晰地知道如何去使用这个应用程序。应用程序通常遵循如下模式:APPNAME VERB NOUN --ADJECTIVE或者APPNAME COMMAND ARG --FLAG,例如:

# server是 commands,port 是 flag
$ hugo server --port=1313

# clone 是 commands,URL 是 arguments,brae 是 flag
$ git clone URL --bare # clone 是一个命令,URL是一个非选项参数,bare是一个选项参数


$ docker info --help 
Usage:  docker info [OPTIONS]

Display system-wide information

Options:
  -f, --format string   Format the output using the given Go template

这里,VERB 代表动词,NOUN 代表名词,ADJECTIVE 代表形容词。

Cobra 提供了两种方式来创建命令:Cobra 命令和 Cobra 库。Cobra 命令可以生成一个 Cobra 命令模板,而命令模板也是通过引用 Cobra 库来构建命令的。

2. 主要功能

cobra 的主要功能如下,可以说每一项都很实用:

  • 简易的子命令行模式,如 app server, app fetch 等等
  • 完全兼容 posix 命令行模式
  • 嵌套子命令 subcommand
  • 支持全局,局部,串联 flags
  • 使用 cobra 很容易的生成应用程序和命令,使用 cobra create appname 和 cobra add cmdname
  • 如果命令输入错误,将提供智能建议,如 app srver,将提示 srver 没有,是不是 app server
  • 自动生成 commands 和 flags 的帮助信息
  • 自动生成详细的 help 信息,如 app help
  • 自动识别帮助 flag -h,–help
  • 自动生成应用程序在 bash 下命令自动完成功能
  • 自动生成应用程序的 man 手册
  • 命令行别名
  • 自定义 help 和 usage 信息
  • 可选的与 viper apps 的紧密集成

3. 应用举例

obra 被用于许多 Go 项目中,例如:Kubernetes、Hugo和Github CLI等,更多广泛使用的项目有:

  • Allero
  • Arduino CLI
  • Bleve
  • Cilium
  • CloudQuery
  • CockroachDB
  • Constellation
  • Cosmos SDK
  • Datree
  • Delve
  • Docker (distribution)
  • Etcd
  • Gardener
  • Giant Swarm’s gsctl
  • Git Bump
  • GitHub CLI
  • GitHub Labeler
  • Golangci-lint
  • GopherJS
  • GoReleaser
  • Helm
  • Hugo
  • Infracost
  • Istio
  • Kool
  • Kubernetes
  • Kubescape
  • KubeVirt
  • Linkerd
  • Mattermost-server
  • Mercure
  • Meroxa CLI
  • Metal Stack CLI
  • Moby (former Docker)
  • Moldy
  • Multi-gitter
  • Nanobox/Nanopack
  • nFPM
  • Okteto
  • OpenShift
  • Ory Hydra
  • Ory Kratos
  • Pixie
  • Polygon Edge
  • Pouch
  • ProjectAtomic (enterprise)
  • Prototool
  • Pulumi
  • QRcp
  • Random
  • Rclone
  • Scaleway CLI
  • Sia
  • Skaffold
  • Tendermint
  • Twitch CLI
  • UpCloud CLI (upctl)
  • VMware’s Tanzu Community Edition & Tanzu Framework
  • Werf
  • ZITADEL

4. Cobra 安装

使用Cobra很简单。首先,使用go get安装最新版本的库。

$ go get -u github.com/spf13/cobra@latest
go: downloading github.com/inconshreveable/mousetrap v1.0.1
go: downloading github.com/inconshreveable/mousetrap v1.1.0
go: added github.com/inconshreveable/mousetrap v1.1.0
go: added github.com/spf13/cobra v1.6.1
go: added github.com/spf13/pflag v1.0.5

接下来 ,导入 Cobra:

import "github.com/spf13/cobra"

5. 使用 Cobra 库创建命令

如果要用 Cobra 库编码实现一个应用程序,需要首先创建一个空的 main.go 文件和一个 rootCmd 文件,之后可以根据需要添加其他命令。具体步骤如下:

5.1 创建 rootCmd

$ mkdir -p newApp2 && cd newApp2

通常情况下,我们会将 rootCmd 放在文件 cmd/root.go 中。

var rootCmd = &cobra.Command{
  Use:   "hugo",
  Short: "Hugo is a very fast static site generator",
  Long: `A Fast and Flexible Static Site Generator built with
                love by spf13 and friends in Go.
                Complete documentation is available at http://hugo.spf13.com`,
  Run: func(cmd *cobra.Command, args []string) {
    // Do Stuff Here
  },
}

func Execute() {
  if err := rootCmd.Execute(); err != nil {
    fmt.Println(err)
    os.Exit(1)
  }
}

还可以在 init() 函数中定义标志和处理配置,例如 cmd/root.go

import (
  "fmt"
  "os"

  homedir "github.com/mitchellh/go-homedir"
  "github.com/spf13/cobra"
  "github.com/spf13/viper"
)

var (
    cfgFile     string
    projectBase string
    userLicense string
)

func init() {
  cobra.OnInitialize(initConfig)
  rootCmd.PersistentFlags().StringVar(&cfgFile, "config", "", "config file (default is $HOME/.cobra.yaml)")
  rootCmd.PersistentFlags().StringVarP(&projectBase, "projectbase", "b", "", "base project directory eg. github.com/spf13/")
  rootCmd.PersistentFlags().StringP("author", "a", "YOUR NAME", "Author name for copyright attribution")
  rootCmd.PersistentFlags().StringVarP(&userLicense, "license", "l", "", "Name of license for the project (can provide `licensetext` in config)")
  rootCmd.PersistentFlags().Bool("viper", true, "Use Viper for configuration")
  viper.BindPFlag("author", rootCmd.PersistentFlags().Lookup("author"))
  viper.BindPFlag("projectbase", rootCmd.PersistentFlags().Lookup("projectbase"))
  viper.BindPFlag("useViper", rootCmd.PersistentFlags().Lookup("viper"))
  viper.SetDefault("author", "NAME HERE <EMAIL ADDRESS>")
  viper.SetDefault("license", "apache")
}

func initConfig() {
  // Don't forget to read config either from cfgFile or from home directory!
  if cfgFile != "" {
    // Use config file from the flag.
    viper.SetConfigFile(cfgFile)
  } else {
    // Find home directory.
    home, err := homedir.Dir()
    if err != nil {
      fmt.Println(err)
      os.Exit(1)
    }

    // Search config in home directory with name ".cobra" (without extension).
    viper.AddConfigPath(home)
    viper.SetConfigName(".cobra")
  }

  if err := viper.ReadInConfig(); err != nil {
    fmt.Println("Can't read config:", err)
    os.Exit(1)
  }
}

5.2 创建 main.go

我们还需要一个 main 函数来调用 rootCmd,通常我们会创建一个 main.go 文件,在 main.go 中调用 rootCmd.Execute() 来执行命令:

package main

import (
  "newApp2/cmd/cmd"
)

func main() {
  cmd.Execute()
}

需要注意,main.go 中不建议放很多代码,通常只需要调用 cmd.Execute() 即可。

5.3 添加命令

除了 rootCmd,我们还可以调用 AddCommand 添加其他命令,通常情况下,我们会把其他命令的源码文件放在 cmd/ 目录下,例如,我们添加一个 version 命令,可以创建 cmd/version.go 文件,内容为:

package cmd

import (
  "fmt"

  "github.com/spf13/cobra"
)

func init() {
  rootCmd.AddCommand(versionCmd)
}

var versionCmd = &cobra.Command{
  Use:   "version",
  Short: "Print the version number of Hugo",
  Long:  `All software has versions. This is Hugo's`,
  Run: func(cmd *cobra.Command, args []string) {
    fmt.Println("Hugo Static Site Generator v0.9 -- HEAD")
  },
}

本示例中,我们通过调用rootCmd.AddCommand(versionCmd)rootCmd 命令添加了一个 versionCmd 命令。

5.4 编译并运行

创建.cobra.yaml文件

$ cat  /c/Users/XH/.cobra.yaml
author: Steve Francia <spf@spf13.com>
license: MIT
useViper: true

main.go{pathToYourApp}替换为对应的路径,例如本示例中 pathToYourAppgithub.com/marmotedu/gopractise-demo/cobra/newApp2

$ cd /d/goprojects/src/github.com/marmotedu/gopractise-demo/cobra/newApp2
$ go mod init newApp2
$ go build -v .
$ ./newApp2 -h
A Fast and Flexible Static Site Generator built with
love by spf13 and friends in Go.
Complete documentation is available at http://hugo.spf13.com
 
Usage:
hugo [flags]
hugo [command]
 
Available Commands:
help Help about any command
version Print the version number of Hugo
 
Flags:
-a, --author string Author name for copyright attribution (default "YOUR NAME")
--config string config file (default is $HOME/.cobra.yaml)
-h, --help help for hugo
-l, --license licensetext Name of license for the project (can provide licensetext in config)
-b, --projectbase string base project directory eg. github.com/spf13/
--viper Use Viper for configuration (default true)
 
Use "hugo [command] --help" for more information about a command.


$ $ ./newApp2.exe version
Hugo Static Site Generator v0.9 -- HEAD

6. 特性

6.1 使用标志

Cobra 可以跟 Pflag 结合使用,实现强大的标志功能。使用步骤如下:

  1. 使用持久化的标志
    标志可以是“持久的”,这意味着该标志可用于它所分配的命令以及该命令下的每个子命令。可以在 rootCmd 上定义持久标志:
rootCmd.PersistentFlags().BoolVarP(&Verbose, "verbose", "v", false, "verbose output")
  1. 使用本地标志
    也可以分配一个本地标志,本地标志只能在它所绑定的命令上使用:
rootCmd.Flags().StringVarP(&Source, "source", "s", "", "Source directory to read from")

--source标志只能在 rootCmd 上引用,而不能在 rootCmd 的子命令上引用。

  1. 将标志绑定到 Viper
    我们可以将标志绑定到 Viper,这样就可以使用 viper.Get() 获取标志的值。
var author string

func init() {
  rootCmd.PersistentFlags().StringVar(&author, "author", "YOUR NAME", "Author name for copyright attribution")
  viper.BindPFlag("author", rootCmd.PersistentFlags().Lookup("author"))
}
  1. 设置标志为必选
    默认情况下,标志是可选的,我们也可以设置标志为必选,当设置标志为必选,但是没有提供标志时,Cobra 会报错。
rootCmd.Flags().StringVarP(&Region, "region", "r", "", "AWS region (required)")
rootCmd.MarkFlagRequired("region")

6.2 非选项参数验证

在命令的过程中,经常会传入非选项参数,并且需要对这些非选项参数进行验证,Cobra 提供了机制来对非选项参数进行验证。可以使用 Command 的 Args 字段来验证非选项参数。Cobra 也内置了一些验证函数:

  • NoArgs:如果存在任何非选项参数,该命令将报错。
  • ArbitraryArgs:该命令将接受任何非选项参数。
  • OnlyValidArgs:如果有任何非选项参数不在 Command 的 ValidArgs 字段中,该命令将报错。
  • MinimumNArgs(int):如果没有至少 N 个非选项参数,该命令将报错。
  • MaximumNArgs(int):如果有多于 N 个非选项参数,该命令将报错。
  • ExactArgs(int):如果非选项参数个数不为 N,该命令将报错。
  • ExactValidArgs(int):如果非选项参数的个数不为 N,或者非选项参数不在 Command 的 ValidArgs 字段中,该命令将报错。
  • RangeArgs(min, max):如果非选项参数的个数不在 min 和 max 之间,该命令将报错。

使用预定义验证函数,示例如下:

var cmd = &cobra.Command{
  Short: "hello",
  Args: cobra.MinimumNArgs(1), // 使用内置的验证函数
  Run: func(cmd *cobra.Command, args []string) {
    fmt.Println("Hello, World!")
  },
}

当然你也可以自定义验证函数,示例如下:

var cmd = &cobra.Command{
  Short: "hello",
  // Args: cobra.MinimumNArgs(10), // 使用内置的验证函数
  Args: func(cmd *cobra.Command, args []string) error { // 自定义验证函数
    if len(args) < 1 {
      return errors.New("requires at least one arg")
    }
    if myapp.IsValidColor(args[0]) {
      return nil
    }
    return fmt.Errorf("invalid color specified: %s", args[0])
  },
  Run: func(cmd *cobra.Command, args []string) {
    fmt.Println("Hello, World!")
  },
}

6.3 PreRun and PostRun Hooks

在运行 Run 函数时,我们可以运行一些钩子函数,比如 PersistentPreRunPreRun 函数在 Run 函数之前执行,PersistentPostRunPostRun 在 Run 函数之后执行。如果子命令没有指定PersistentRun函数,则子命令将会继承父命令的PersistentRun函数。这些函数的运行顺序如下:

  • PersistentPreRun
  • PreRun
  • Run
  • PostRun
  • PersistentPostRun

注意,父级的 PreRun 只会在父级命令运行时调用,子命令是不会调用的。

Cobra 还支持很多其他有用的特性,比如:

  • 自定义 Help 命令;
  • 可以自动添加--version标志,输出程序版本信息;
  • 当用户提供无效标志或无效命令时,Cobra 可以打印出 usage 信息;
  • 当我们输入的命令有误时,Cobra 会根据注册的命令,推算出可能的命令,等等。

7. cobra-cli 命令

7.1 安装

cobra-cli是一个命令行程序,用于生成cobra应用程序和命令文件。它将引导您的应用程序脚手架,以快速开发基于cobra的应用程序。这是将Cobra合并到应用程序中最简单的方法。

可以通过以下命令安装:

$ go install github.com/spf13/cobra-cli@latest
go: downloading github.com/spf13/cobra-cli v1.3.0
go: downloading github.com/spf13/cobra v1.3.0
go: downloading github.com/spf13/viper v1.10.1
go: downloading github.com/fsnotify/fsnotify v1.5.1
go: downloading github.com/mitchellh/mapstructure v1.4.3
go: downloading github.com/magiconair/properties v1.8.5
go: downloading github.com/spf13/afero v1.6.0
go: downloading github.com/spf13/cast v1.4.1
go: downloading gopkg.in/ini.v1 v1.66.2
go: downloading github.com/pelletier/go-toml v1.9.4
go: downloading golang.org/x/text v0.3.7

cobra-cli命令自动将它安装在你的$GOPATH/bin目录中

$ ls $GOPATH/bin/cobra-cli.exe 
'D:\goprojects/bin/cobra-cli.exe'

7.2 cobra-cli 初始化

cobra-cli init [app]命令将为您创建初始应用程序代码。这是一个非常强大的应用程序,可以用正确的结构填充您的程序,这样您就可以立即享受 Cobra 的所有好处。它还可以将您指定的许可证应用于您的应用程序。

随着 Go 模块的引入,Cobra 生成器已被简化以利用模块。Cobra 生成器在 Go 模块中工作。

如果你想初始化一个新的 Go 模块:

cd $HOME/code 
mkdir myapp
cd myapp
go mod init github.com/spf13/myapp
cobra-cli init
go run main.go

cobra-cli init 也可以从子目录运行,例如cobra 生成器本身的组织方式。如果您想将应用程序代码与库代码分开,这将很有用。

可选标志:

  • 您可以为它提供您的作者姓名和--author标志。例如cobra-cli init --author "Steve Francia spf@spf13.com"

  • 您可以提供使用许可,--license 例如cobra-cli init --license apache

  • 使用--viper标志自动设置viper,Viper 是 Cobra 的辅助工具,旨在提供对环境变量和配置文件的轻松处理,并将它们无缝连接到应用程序标志。

7.3 向项目添加命令

初始化 Cobra 应用程序后,您可以继续使用 Cobra 生成器向您的应用程序添加其他命令。执行此操作的命令是cobra-cli add。

假设您创建了一个应用程序,并希望为它执行以下命令:

  • 应用服务
  • 应用配置
  • 应用程序配置创建

在您的项目目录(您的 main.go 文件所在的位置)中,您将运行以下命令:

cobra-cli add serve
cobra-cli add config
cobra-cli add create -p 'configCmd'

您会注意到这个最终命令有一个-p标志。这用于将父命令分配给新添加的命令。在这种情况下,我们要将“create”命令分配给“config”命令。如果未指定,所有命令都具有 rootCmd 的默认父级。

默认情况下cobra-cli将附加Cmd到提供的名称并将此名称用作内部变量名称。指定父级时,一定要与代码中使用的变量名相匹配。

注意:命令名称使用驼峰命名法(不是 snake_case/kebab-case)。否则,您将遇到错误。例如,cobra-cli add add-user不正确,但cobra-cli add addUser有效。

运行这三个命令后,您将拥有类似于以下的应用程序结构:

  ▾ app/
    ▾ cmd/
        config.go
        create.go
        serve.go
        root.go
      main.go

此时您可以运行go run main.go,它会运行您的应用程序。go run main.go serve, go run main.go config,go run main.go config create连同go run main.go help serve, 等都可以。

您现在已经启动并运行了一个基于 Cobra 的基本应用程序。

参考:

  • Cobra 快速入门 - 专为命令行程序而生
  • Golang : cobra 包简介

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

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

相关文章

手绘图说电子元器件-电声转换器件

电声转换器件包括能够将电信号转换为声音的扬声器、耳机、讯响器和蜂鸣器,能够将声音转换为电信号的传声器,能够进行电磁转换的磁头和具有压电效应的晶体等。 扬声器 扬声器俗称喇叭,是一种常用的电声转换器件,其基本作用是将电信号转换为声音,在收音机、录音机、电视机…

Linux | 套接字(socket)编程 | TCP协议讲解 | 通信模型搭建

文章目录TCP模型的特性TCP接口介绍TCP服务器套接字设置TCP客户端套接字设置TCP模型的特性 TCP是属于传输层协议的一种&#xff0c;上篇博客介绍了另一种传输层协议——UDP&#xff0c;关于它们之间的区别&#xff0c;这里再提一下 TCPUDP传输层协议传输层协议有连接无连接可靠…

Word控件Spire.Doc 【评论】教程(3):在C#、VB.NET中从Word文档中提取注释并保存在TXT文件中

Spire.Doc for .NET是一款专门对 Word 文档进行操作的 .NET 类库。在于帮助开发人员无需安装 Microsoft Word情况下&#xff0c;轻松快捷高效地创建、编辑、转换和打印 Microsoft Word 文档。拥有近10年专业开发经验Spire系列办公文档开发工具&#xff0c;专注于创建、编辑、转…

[Leetcode] 合并两个有序数组、链表

1.合并两个有序数组 原地合并数组&#xff0c;即不使用额外的空间 --> 使用三个指针&#xff0c;从尾部往前处理 题目链接&#xff1a;https://leetcode.cn/problems/merge-sorted-array/ nums1 总长度 mn&#xff0c;自身长度m&#xff1b;nums2 自身长度n&#xff0c; 使…

SSRF渗透与攻防(一)

目录 前言 SSRF是什么 危害&#xff08;利用方式): SSRF漏洞原理&#xff1a; CURL协议&#xff1a; SSRF常见场景 社会化分享功能&#xff1a; 如何发现SSRF漏洞 工具利用&#xff1a; 如何防御SSRF漏洞 前言 SSRF(Server-Side Request Forgery:服务器端请求伪造) 是…

rocketmq 实战问题汇总

rocketmq 实战过程会遇到这样或者那样的问题&#xff0c;今天我们专门抽出一篇文章来分析一下汇总一下&#xff0c;避免以后踩同样的坑&#xff1a; 1、找不到JDK的问题&#xff1a; 综合分析&#xff0c;是因为JDK安装的目录有空格导致的&#xff1a;Program Files 两个单词之…

电子招标采购系统源码—企业战略布局下的采购寻源

​ 智慧寻源 多策略、多场景寻源&#xff0c;多种看板让寻源过程全程可监控&#xff0c;根据不同采购场景&#xff0c;采取不同寻源策略&#xff0c; 实现采购寻源线上化管控&#xff1b;同时支持公域和私域寻源。 询价比价 全程线上询比价&#xff0c;信息公开透明&#xff0…

CANoe—基于DoIP通过CAPL实现与ECU通信测试

如下连接是在CANoe中基于DoIP通过加载诊断数据库实现CANoe与待测ECU诊断通信: CANoe链接 本文继续此话题,通过一个简单的CAPL Demo,实现CANoe与ECU进行DoIP通信。 首先在CANoe新建Ethernet工程: 在CANoe “Simulation Setup”中新建CAPL Test Module: 在此例中采用CANo…

DHCP原理和实验

目录 DHCP基本认识和原理 场景一、同网段DHCP 场景二、不同段DHCP&#xff08;中继DHCP&#xff09; DHCP基本认识和原理 DHCP&#xff08;Dynamic Host Configuration Protocol动态主机协议&#xff09;。 作用&#xff1a;为局域网络中主机动态分发地址&#xff0c;以及…

C#里使用ExcelDataReader读取EXCEL文件的简单方法

C#里使用ExcelDataReader读取EXCEL文件的简单方法 读取EXCEL文件是比较常见的需求,所以在C#里也会经常遇到。 比如客户需要保存的条码数据,他们可以使用EXCEL来扫码进去,并且进行修改和核验, 然后软件就需要读取这些EXCEL文件,并且从这里得到所需要的条码。 要从EXCEL里…

Android 查看隐私权限方法调用者集合

背景 辛辛苦苦迭代完当前版本&#xff0c;准备推送 App 到应用市场上架&#xff0c;却收到拒审通知&#xff1a;App隐私合规上架护航版检测报告&#xff0c; 报告内容&#xff1a; 场景2:APP以隐私政策弹窗的形式向用户明示收集使用规则&#xff0c;未经用户同意&#xff0c;…

页面崩溃了!记录一次测试中出现的前端内存溢出现象

前情回顾 前几天在一次web应用测试过程中&#xff0c;前端发起了向后端接口的查询请求&#xff0c;由于后端响应较慢&#xff0c;前端一直处于等待响应返回状态。在几分钟后&#xff0c;突然页面出现让人惊悚的“噢噢&#xff0c;页面崩溃了”几个大字。 看到这几个字的一瞬间…

基于GitLab构建企业级CICD-Gitlab-Runner

背景 在过往企业开发中&#xff0c;大部分企业从开发到测试&#xff0c;到部署目前还是手工进行在一些某些中大型企业中&#xff0c;目前构建及部署还是直接使用二进制包部署&#xff0c;或直接单机运行在某些场合下&#xff0c;仓库中代码的编译需要硬件支持&#xff0c;致使…

SSM框架学习记录-Maven_day01

1.分模块开发 将原始模块按照功能拆分成若干个子模块&#xff0c;方便模块间的相互调用&#xff0c;接口共享&#xff1a;比如有订单和商品两个模块&#xff0c;它们都需要使用到商品的模型类&#xff0c;如果在这两个模块中都写模型类&#xff0c;就会出现重复代码&#xff0c…

Doris部分列更新在广告行业应用

背景&#xff1a;业务需要在不同的时间点对同一个session_id上的广告行为&#xff08;展示、点击、转换等&#xff09;数据的更新。 基于HBase归因 更新原理&#xff1a;以session_id为Key在HBase中写入数据&#xff0c;数据更新是先点查到历史数据&#xff0c;补齐当前数据后…

风已起,待云涌---多维度理解云安全

Fix the Unknown,Before You Know it. 新时代大门开启的时候&#xff0c;蜂拥而上的大都是勇士&#xff0c;风已起&#xff0c;待云涌&#xff01; 1.云安全&#xff1a; 未来安全的能力将成为计算、存储、网络之外的第四大基础设施&#xff0c;并全部融入到云基础设施中&…

Python爬虫入门 ~ selenium访问元素信息与交互基本使用

访问元素信息 前面我们成功定位到了页面的标签元素&#xff0c;那接下来就该轮到获取元素的信息了&#xff0c;常用的函数有以下几种: get_attributetexttag_name 前置准备 from selenium import webdriver from selenium.webdriver.chrome.service import Service from sel…

【Django项目开发】用户认证功能开发(四)

一、JWT Token配置 1、下载djangorestframework-jwt pip install djangorestframework-jwt2、settings.py指定使用的认证类JSONWebTokenAuthentication REST_FRAMEWORK {# 指定使用的认证类# a、在全局指定默认的认证类&#xff08;认证方式&#xff09;DEFAULT_AUTHENTICA…

企业金融App评测系列——微众银行以App构筑企业金融服务新生态,成为企业的随身数字银行

易观分析&#xff1a;近年来&#xff0c;疫情驱动小微企业线上化需求不断提升&#xff0c;经营面临的财力、人力、信息获取的紧迫性进一步提升。为更好发展普惠小微金融的商业银行对公服务&#xff0c;正聚焦更好满足小微企业的数字化需求&#xff0c;加快打造企业手机银行等移…

mybatis xml接收日期格式参数

实现方式一&#xff1a; mapper List<AsLLjgcfVO> selectjgcf(Param("rq")Date rq)mapper.xml <select id"selectjgcf" parameterType"string" resultType"com.bysen.mes.domain.VO.AsLLjgcfVO"> where ${rq} between …