cron表达式 详解

corn表达式是:由若干数字、空格、符号按一定的规则,组成的一组字符串,从而表达时间的信息。

好像和正则表达式有点类似哈,都是一个字符串表示一些信息。

Cron 表达式生成器: https://www.smart-tools.cn/cron

简介

Cron 表达式是一个具有时间含义的字符串,字符串以 5 或 6 个空格隔开,分为 6 或 7 个域,每一个域代表一种含义。 Cron 有如下两种语法格式:

  • 秒 分 小时 日期 月份 星期

  • 秒 分 小时 日期 月份 星期 年

即:秒 分 小时 日期 月份 星期 年(可为空)

主流工具

目前的 Cron 表达式主要有两类,分别是:

  • Linux crontab 命令 (Crontab 是linux系统自带的定时任务,用于设置周期性执行的本地脚本。Crontab的cron表达式只能精确到分钟。例如,* * * * ?)

  • Java Quartz(Quartz 是一个完全由 Java 编写的开源作业调度框架,为 Java 应用进行任务调度提供了简单却强大的机制。Quartz的cron表达式可以精确到秒。例如,* * * * * ?)

其中,Linux crontab 仅支持分钟级别的任务调度;Java Quartz 则可以秒级别的任务调度;

Linux crontab 中的 cron 语法规范

* * * * *

- - - - -

| | | | |

| | | | +----- 星期中星期几 (0 - 6) (星期天为0)

| | | +---------- 月份 (1 - 12)

| | +--------------- 一个月中的第几天 (1 - 31)

| +-------------------- 小时 (0 - 23)

+------------------------- 分钟 (0 - 59)

Java Quartz 中的 cron 语法规范

* * * * * *

- - - - - -

| | | | | |

| | | | | +----- 星期中星期几 (0 - 6) (星期天为0)

| | | | +---------- 月份 (1 - 12)

| | | +--------------- 一个月中的第几天 (1 - 31)

| | +-------------------- 小时 (0 - 23)

| +------------------------- 分钟 (0 - 59)

+------------------------------ 秒 (0 - 60)

域取值

下表为 Cron 表达式中每个域能够取的值以及支持的特殊字符。

是否必需

取值范围

特殊字符

[0, 59]

* , - /

分钟

[0, 59]

* , - /

小时

[0, 23]

* , - /

日期

[1, 31]

* , - / ? L W C

月份

[1, 12]

* , - /

星期

[1, 7] 其中 1 表示星期一,7 表示星期日。

* , - / ? L C #

留空,1970~2099

, - * /

特殊字符

Cron 表达式中的每个域都支持一定数量的特殊字符,每个特殊字符有其特殊含义。

特殊字符

含义

示例

*

所有可能的值

月份域中,* 表示每个月;在星期域中,* 表示星期的每一天。

,

列出枚举值

分钟域中,5,20 表示分别在 5 分钟和 20 分钟触发一次。

-

范围

分钟域中,5-20 表示从 5 分钟到 20 分钟之间每隔一分钟触发一次。

/

表示起始时间开始触发,然后每隔固定时间触发一次

分钟域中,0/15 表示从第 0 分钟开始,每 15 分钟触发一次。在分钟域中3/20表示从第 3 分钟开始,每 20 分钟触发一次。

?

不指定值,仅日期和星期域支持该字符

当日期或星期域其中之一被指定了值以后,为了避免冲突,需要将另一个域的值设为?。

L

单词 Last 的首字母,表示最后一天,仅日期和星期域支持该字符

日期域中,L 表示某个月的最后一天。在星期域中,L 表示一个星期的最后一天,也就是星期日(SUN)。如果在 L 前有具体的内容,例如,在星期域中的 6L,表示这个月的最后一个星期六。

W

表示有效工作日(周一到周五),只能出现在日期域,系统将在离指定日期最近的有效工作日触发事件。W 字符寻找当前月份中最近有效工作日,连用字符 LW 时表示为指定月份的最后一个工作日。

日期域中使用 5W, 如果 5 号是星期六,则将在最近的工作日星期五,即 4 日触发。如果 5 日是星期天,则将在最近的工作日星期一,即 6 日触发;如果 5 日在星期一到星期五中的一天,则就在 5 日触发。

#

确定每个月第几个星期几,仅星期域支持该字符。

星期域中,4#2表示某月的第二个星期四。

C

这个字符依靠一个指定的“日历”。也就是说这个表达式的值依赖于相关的“日历”的计算结果,如果没有“日历”关联,则等价于所有包含的“日历”。仅日期和星期域支持该字符。

日期域是“5C”表示关联“日历”中第一天,或者这个月开始的第一天的后5天。星期域是“1C”表示关联“日历”中第一天,或者星期的第一天的后1天,也就是周日的后一天(周一)。

允许值范围: 0~59 ,不允许为空值,若值不合法,调度器将抛出SchedulerException异常

"*" 代表每隔1秒钟触发

"," 代表在指定的秒数触发,比如"0,15,45"代表0秒、15秒和45秒时触发任务

"-" 代表在指定的范围内触发,比如"25-45"代表从25秒开始触发到45秒结束触发,每隔1秒触发1次

"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"0"),后面的值代表偏移量,比如"0/20"或者"*/20"代表从0秒钟开始,每隔20秒钟触发1次,即0秒触发1次,20秒触发1次,40秒触发1次;"5/20"代表5秒触发1次,25秒触发1次,45秒触发1次;"10-45/20"代表在[10,45]内步进20秒命中的时间点触发,即10秒触发1次,30秒触发1次

分钟

允许值范围: 0~59 ,不允许为空值,若值不合法,调度器将抛出SchedulerException异常

"*" 代表每隔1分钟触发

"," 代表在指定的分钟触发,比如"10,20,40"代表10分钟、20分钟和40分钟时触发任务

"-" 代表在指定的范围内触发,比如"5-30"代表从5分钟开始触发到30分钟结束触 发,每隔1分钟触发

"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"0"),后面的值代表偏移量,比如"0/25"或者"*/25"代表从0分钟开始,每隔25分钟触发1次,即0分钟触发1次,第25分钟触发1次,第50分钟触发1次;"5/25"代表5分钟触发1次,30分钟触发1次,55分钟触发1次;"10-45/20"代表在[10,45]内步进20分钟命中的时间点触发,即10分钟触发1次,30分钟触发1次

小时

允许值范围: 0~23 ,不允许为空值,若值不合法,调度器将抛出SchedulerException异常

"*" 代表每隔1小时触发

"," 代表在指定的时间点触发,比如"10,20,23"代表10点钟、20点钟和23点触发任务

"-" 代表在指定的时间段内触发,比如"20-23"代表从20点开始触发到23点结束触发,每隔1小时触发

"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"0"),后面的值代表偏移量,比如"0/1"或者"*/1"代表从0点开始触发,每隔1小时触发1次;"1/2"代表从1点开始触发,以后每隔2小时触发一次

日期

允许值范围: 1~12 (JAN-DEC),不允许为空值,若值不合法,调度器将抛出SchedulerException异常

"*" 代表每个月都触发

"," 代表在指定的月份触发,比如"1,6,12"代表1月份、6月份和12月份触发任务

"-" 代表在指定的月份范围内触发,比如"1-6"代表从1月份开始触发到6月份结束触发,每隔1个月触发

"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"1"),后面的值代表偏移量,比如"1/2"或者"*/2"代表从1月份开始触发,每隔2个月触发1次;"6/6"代表从6月份开始触发,以后每隔6个月触发一次;"1-6/12"表达式意味着每年1月份触发

"C" 这个字符依靠一个指定的“日历”。也就是说这个表达式的值依赖于相关的“日历”的计算结果,如果没有“日历”关联,则等价于所有包含的“日历”。比如“5C”表示关联“日历”中第一天,或者这个月开始的第一天的后5天。

星期

允许值范围: 1~7 (SUN-SAT),1代表星期天(一星期的第一天),以此类推,7代表星期六(一星期的最后一天),不允许为空值,若值不合法,调度器将抛出SchedulerException异常

"*" 代表每星期都触发;

"?" 与{日期}互斥,即意味着若明确指定{日期}触发,则表示{星期}无意义,以免引起冲突和混乱

"," 代表在指定的星期约定触发,比如"1,3,5"代表星期天、星期二和星期四触发

"-" 代表在指定的星期范围内触发,比如"2-4"代表从星期一开始触发到星期三结束触发,每隔1天触发

"/" 代表触发步进(step),"/"前面的值代表初始值("*"等同"1"),后面的值代表偏移量,比如"1/3"或者"*/3"代表从星期天开始触发,每隔3天触发1次;"1-5/2"表达式意味着在[1,5]范围内,每隔2天触发,即星期天、星期二、星期四触发

"L" 如果{星期}占位符如果是"L",即意味着星期的的最后一天触发,即星期六触发,L= 7或者 L = SAT,因此,"5L"意味着一个月的最后一个星期四触发

"#" 用来指定具体的周数,"#"前面代表星期,"#"后面代表本月第几周,比如"2#2"表示本月第二周的星期一,"5#3"表示本月第三周的星期四,因此,"5L"这种形式只不过是"#"的特殊形式而已

"C" 这个字符依靠一个指定的“日历”。也就是说这个表达式的值依赖于相关的“日历”的计算结果,如果没有“日历”关联,则等价于所有包含的“日历”。比如:“1C”表示关联“日历”中第一天,或者星期的第一天的后1天,也就是周日的后一天(周一)。

年份

允许值范围: 1970~2099 ,允许为空,若值不合法,调度器将抛出SchedulerException异常

"*"代表每年都触发

","代表在指定的年份才触发,比如"2011,2012,2013"代表2011年、2012年和2013年触发任务

"-"代表在指定的年份范围内触发,比如"2011-2020"代表从2011年开始触发到2020年结束触发,每隔1年触发

"/"代表触发步进(step),"/"前面的值代表初始值("*"等同"1970"),后面的值代表偏移量,比如"2011/2"或者"*/2"代表从2011年开始触发,每隔2年触发1次

注意:除了{日期}和{星期}可以使用"?"来实现互斥,表达无意义的信息之外,其他占位符都要具有具体的时间含义,且依赖关系为:年->月->日期(星期)->小时->分钟->秒数

Cron 表达式示例

示例 1

说明

0 15 10 ? * *

每天上午 10:15 执行任务

0 15 10 * * ?

每天上午 10:15 执行任务

0 0 12 * * ?

每天中午 12:00 执行任务

0 0 10,14,16 * * ?

每天上午 10:00 点、下午 14:00 以及下午 16:00 执行任务

0 0/30 9-17 * * ?

每天上午 09:00 到下午 17:00 时间段内每隔半小时执行任务

0 * 14 * * ?

每天下午 14:00 到下午 14:59 时间段内每隔 1 分钟执行任务

0 0-5 14 * * ?

每天下午 14:00 到下午 14:05 时间段内每隔 1 分钟执行任务

0 0/5 14 * * ?

每天下午 14:00 到下午 14:55 时间段内每隔 5 分钟执行任务

0 0/5 14,18 * * ?

每天下午 14:00 到下午 14:55、下午 18:00 到下午 18:55 时间段内每隔 5 分钟执行任务

0 0 12 ? * WED

每个星期三中午 12:00 执行任务

0 15 10 15 * ?

每月 15 日上午 10:15 执行任务

0 15 10 L * ?

每月最后一日上午 10:15 执行任务

0 15 10 ? * 6L

每月最后一个星期六上午 10:15 执行任务

0 15 10 ? * 6#3

每月第三个星期六上午 10:15 执行任务

0 10,44 14 ? 3 WED

每年 3 月的每个星期三下午 14:10 和 14:44 执行任务

cron 表达式用途

cron 表达式最主要的就是在程序中做一些定时任务,比如某些系统的报表数据,某些游戏的排行榜,由于这些数据量实时统计非常消耗程序性能,所以就每隔一段时间,通过自动任务跑一次,这样可以极大的提升用户浏览体验,要是在游戏里,还可以增加一种神秘感。

另外,某些具体点的数据拉取,比如你如果从事平台对接工作,要从某些平台下载你的订单,那么肯定是每隔多久抓一次。

又比如你写个爬虫,要实时的了解你的某些数据,然后从这些数据中反应你的情况。

一个简单的java项目示例

导入依赖

        <dependency>
            <groupId>org.quartz-scheduler</groupId>
            <artifactId>quartz</artifactId>
            <version>2.2.3</version>
        </dependency>

定义Job

定义一个HelloJob类实现Job接口

    import org.quartz.Job;
    import org.quartz.JobExecutionContext;
    import org.quartz.JobExecutionException;
    
    import java.util.Date;
    
    public class HelloJob implements Job {
    
        @Override
        public void execute(JobExecutionContext arg0) throws JobExecutionException {
            System.out.println("Say hello to Quartz {" + new Date() +"}");
        }
}

测试

写个主函数,在主函数里面完成整个Quartz的操作过程

    import org.quartz.*;
    import org.quartz.impl.StdSchedulerFactory;
    
    import java.util.Date;
    
    public class HelloQuartz {
    
        public static void main(String[] args) throws SchedulerException {
            //1.创建Scheduler的工厂
            SchedulerFactory sf = new StdSchedulerFactory();
            //2.从工厂中获取调度器实例
            Scheduler scheduler = sf.getScheduler();
    
            //3.创建JobDetail
            JobDetail jb = JobBuilder.newJob(HelloJob.class)
                    .withDescription("this is a job") //job的描述
                    .withIdentity("Job", "Group") //job 的name和group
                    .build();
    
            //任务运行的时间,SimpleSchedle类型触发器有效
            long time=  System.currentTimeMillis() + 3*1000L; //3秒后启动任务
            Date statTime = new Date(time);
    
            //4.创建Trigger
            //使用SimpleScheduleBuilder或者CronScheduleBuilder
            Trigger t = TriggerBuilder.newTrigger()
                    .withDescription("")
                    .withIdentity("Trigger", "TriggerGroup")
                    //.withSchedule(SimpleScheduleBuilder.simpleSchedule())
                    .startAt(statTime)  //默认当前时间启动
                    .withSchedule(CronScheduleBuilder.cronSchedule("0/10 * * * * ?")) // 10秒执行一次
                    .build();
    
            //5.注册任务和定时器
            scheduler.scheduleJob(jb, t);
    
            //6.启动 调度器
            scheduler.start();
        }
}

运行截图:

一个简单的sprin boot项目示例

导入依赖

    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-quartz</artifactId>
    </dependency>

定义Job

定义一个HelloJob类实现Job接口

    package com.example.demo;
    
    import org.quartz.Job;
    import org.quartz.JobExecutionContext;
    import org.quartz.JobExecutionException;
    
    import java.util.Date;
    
    public class HelloJob implements Job {
    
        @Override
        public void execute(JobExecutionContext arg0) throws JobExecutionException {
            System.out.println("Say hello to Quartz {" + new Date() +"}");
        }
    }

测试

写个主函数,在主函数里面完成整个Quartz的操作过程

    package com.example.demo;
    
    
    import org.quartz.*;
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    
    // 要配置Quartz的调度器Scheduler
    // 调度器由SpringBoot管理,所以就变成了配置Spring
    @Configuration
    public class Demo {
    
        // 配置的核心是向Spring容器保存一个job和保存一个Trigger
    
        // 创建一个封装Job对象的类型JobDetail
        // 使用@Bean注解标记的方法将这个对象保存到Spring容器
        @Bean
        public JobDetail addStock(){
            //newJob方法就是在绑定要运行的Job接口实现类,需要实现类的反射做参数
            return JobBuilder.newJob(HelloJob.class)
                    // 给当前JobDetail对象在调度环境中起名
                    .withIdentity("addStock")
                    // 即使没有触发器绑定当前JobDetail对象,也不会被删除
                    .storeDurably()
                    .build();
        }
    
        // 下面是触发器的声明,也会保存到Spring容器中
        // 它能够设置job的运行时机
        @Bean
        public Trigger addStockTrigger(){
            System.out.println("Trigger保存到Spring容器中");
            // 定义Cron表达式
            CronScheduleBuilder cron=
                    CronScheduleBuilder.cronSchedule("0/10 * * * * ?"); // 10秒执行一次
            return TriggerBuilder.newTrigger()
                    // 绑定要运行的JobDetail对象
                    .forJob(addStock())
                    // 为触发器起名
                    .withIdentity("addStockTrigger")
                    // 绑定cron表达式
                    .withSchedule(cron)
                    .build();
        }
    }

运行截图:

Linux的crontab命令

在Linux中的cron表达式与上文所述有所不同,更准确地说是更加精简了。

crontab的命令构成为 cron表示式 + command,这里的cron表达

Minutes Hours DayOfMonth Month DayOfWeek

而操作符则有:

* / - ,

一起来看看几个例子:

1. 每晚的21:30重启smb

30 21 * * * /etc/init.d/smb restart

2. 每星期六的晚上11:00 pm重启smb

0 23 * * 6 /etc/init.d/smb restart

3. 晚上11点到早上7点之间,每隔一小时重启smb

0 23-7/1 * * * /etc/init.d/smb restart

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

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

相关文章

部署私有npm 库

使用verdacciohttps://verdaccio.org/安装verdaccio使用npm全局安装npm install -g verdaccio安装完成以后&#xff0c;输入verdaccio -h出现如下相关提示&#xff0c;说明verdaccio安装成功。运行verdaccio直接执行verdaccio出现如下相关提示&#xff0c;说明verdaccio启动成功…

【OJ比赛日历】快周末了,不来一场比赛吗? #03.25-03.31 #12场

CompHub 实时聚合多平台的数据类(Kaggle、天池…)和OJ类(Leetcode、牛客…&#xff09;比赛。本账号同时会推送最新的比赛消息&#xff0c;欢迎关注&#xff01;更多比赛信息见 CompHub主页 或 点击文末阅读原文以下信息仅供参考&#xff0c;以比赛官网为准目录2023-03-25&…

React 入门(超详细)

目录前言&#xff1a;一、React 简介1. 什么是 React2. React 的特点3. React 高效的原因4. React 官网5. React的主要原理6. Facebook为什么要建造React?二、React 的基本使用1. 基础代码2. 效果3. 相关 js 库4. 创建虚拟DOM的两种方式5. 虚拟DOM与真实DOM6. 虚拟DOM与真实DO…

Linux命令运行原理shell和bash

目录前言什么是shell,什么是bash?ls -l 执行过程前言 学习操作系统的过程中我们经常在自己的shell中执行一些Linux命令&#xff0c;那么当我们输入一个类似于 ls -a 这样的命令式&#xff0c;发生了什么&#xff1f; 换句话说&#xff0c;从我们在shell中输入ls -a 按下回车…

基于深度学习的瓶盖检测系统(Python+YOLOv5深度学习模型+清新界面)

摘要&#xff1a;基于深度学习的瓶盖检测系统用于传送带或日常场景中瓶盖检测识别&#xff0c;提供实时瓶盖检测定位和计数&#xff0c;辅助瓶盖生产加工过程的自动化识别。本文详细介绍基于深度学习的瓶盖检测系统&#xff0c;在介绍算法原理的同时&#xff0c;给出Python的实…

E - 积木画(状态压缩DP)

E - 积木画&#xff08;状态压缩DP&#xff09; 1、问题 E - 积木画 2、分析 这道题很明显是一道DP题&#xff0c;而且是一个简化版的状态压缩DP。 &#xff08;1&#xff09;状态表示 f[i][j]f[i][j]f[i][j]表示前面i−1i-1i−1已经摆好&#xff0c;并且第iii列的状态是j…

第七讲 贪心

文章目录股票买卖 II货仓选址&#xff08;贪心:排序中位数&#xff09;糖果传递&#xff08;❗贪心&#xff1a;中位数&#xff09;雷达设备&#xff08;贪心排序&#xff09;付账问题&#xff08;平均值排序❓&#xff09;乘积最大&#xff08;排序/双指针&#xff09;后缀表达…

SpringBoot——基于SpringBoot整合Mybatis的入门案例+sql提示配置

开发流程图 先选择springboot项目创建&#xff0c;此处3.0以上的springboot项目最低都要java17版本 让数据库表中的属性名和实体类中的属性名保持一致就可以完成查询结果的自动封装。 2.引入myabtis相关依赖&#xff0c;配置mybatis 这里可以手动选择mybatis框架的依赖和mysq…

常见的卷积神经网络结构——分类、检测和分割

本文持续更新~~ 本文整理了近些年来常见的卷积神经网络结构&#xff0c;涵盖了计算机视觉领域的几大基本任务&#xff1a;分类任务、检测任务和分割任务。对于较复杂的网络&#xff0c;本文只会记录其中的核心模块以及重要的网络设计思想&#xff0c;并不会记录完整的网络结构。…

POM依赖报错解决方案汇总

POM依赖报错解决方案汇总 POM依赖报错解决方案汇总 状况 1 创建完一个maven项目,在pom文件在引入依赖,等下方自动导入加载完毕,去查看IDEA工具的Maven Projects工具选项中Dependencies 依然后依赖红色报错 2 在pom文件中,引用依赖后,该依赖的版本号处直接出现红色 3 IDEA工具…

蓝桥杯Web前端练习题-----水果拼盘

一、水果拼盘 介绍 目前 CSS3 中新增的 Flex 弹性布局已经成为前端页面布局的首选方案&#xff0c;本题可以使用 Flex 属性快速完成布局。 准备 开始答题前&#xff0c;需要先打开本题的项目代码文件夹&#xff0c;目录结构如下&#xff1a; ├── css │ └── style.…

值得记忆的STL常用算法,分分钟摆脱容器调用的困境,以vector为例,其余容器写法类似

STL常用算法 概述&#xff1a; 算法主要是由头文件<algorithm> <functional> <numeric>组成 <algorithm>是所有STL头文件中最大的一个&#xff0c;范围涉及到比较、交换、查找、遍历操作、复制、修改等等 <nuneric>体积很小&#xff0c;只包括…

【AWS入门】通过AWS存储网关(Storage Gate Way)实现文件共享

目录1. 创建网关2. 创建文件共享3. Windows文件共享4. LINUX文件共享1. 创建网关 配置缓存存储需要加载一会儿&#xff0c;此处需要等候 根据上述配置&#xff0c;会自动生成一个EC2实例 2. 创建文件共享 网关&#xff1a;选择上述步骤中创建的网关&#xff0c;选择即可 文…

电路设计的一些概念

锁存器的产生 论述1 (转)时序电路&#xff0c;生成触发器&#xff0c;触发器是有使能端的&#xff0c;使能端无效时数据不变&#xff0c;这是触发器的特性。 组合逻辑&#xff0c;由于数据要保持不变&#xff0c;只能通过锁存器来保存。 第一个代码&#xff0c;由于是时序逻…

基于XML的自动装配~

基于XML的自动装配之场景模拟&#xff1a; 自动装配&#xff1a;根据指定的策略&#xff0c;在IOC容器中匹配某一个bean&#xff0c;自动为指定的bean中所依赖的类类型或者接口类型赋值 之前我们学过的依赖注入&#xff0c;我们在为不同属性赋值时&#xff0c;例如类类型的属性…

可别再用BeanUtils了(性能拉胯),试试这款转换神器

老铁们是不是经常为写一些实体转换的原始代码感到头疼&#xff0c;尤其是实体字段特别多的时候。有的人会说&#xff0c;我直接使用get/set方法。没错&#xff0c;get/set方法的确可以解决&#xff0c;而且也是性能较高的处理方法&#xff0c;但是大家有没有想过&#xff0c;要…

数据结构与算法——堆的基本存储

目录 一、概念及其介绍 二、适用说明 三、结构图示 四、Java 实例代码 五.堆和栈的区别 一、概念及其介绍 堆(Heap)是计算机科学中一类特殊的数据结构的统称。 堆通常是一个可以被看做一棵完全二叉树的数组对象。 堆满足下列性质&#xff1a; 堆中某个节点的值总是不大…

MySQL主从复制

主从复制概述 如何提升数据库并发能力 在实际工作中&#xff0c;我们常常将 Redis 作为缓存与 MySQL 配合来使用&#xff0c;当有请求的时候&#xff0c;首先会从缓存中进行查找&#xff0c;如果存在就直接取出。如果不存在再访问数据库&#xff0c;这样就 提升了读取的效率&…

I2C和SPI总线以及通信

通讯属性 概括 Serial/parallel 串行/并行Synchronous/asynchronous 同步/异步Point-to-point / bus 点对点 总线Half-duplex/full-duplex 半双工/全双工Master-slave/ equal partners 主从/对等single-ending / differential 单端/差分 点对点和总线 点对点通讯 只有两个通…

【简陋Web应用2】人脸检测——基于Flask和PaddleHub

文章目录&#x1f6a9; 前言&#x1f33a; 效果演示&#x1f966; 分析与设计&#x1f349; 实现&#x1f36c; 1. 部署人脸检测模型&#x1f36d; 2. 使用Flask构建app2.1 目录结构2.2 forms.py2.3 utils.py2.4 app.py2.5 index.html&#x1f95d; Bug(s)&#x1f6a9; 前言 本…