python-38-python定时任务框架

news2025/1/24 14:29:38

Python定时任务
Python任务调度模块 – APScheduler
python调度框架APScheduler使用详解
APScheduler动态增、删、改任务
apscheduler mysql 持久化任务
APScheduler调度框架
在项目中,我们可能遇到有定时任务的需求。
其一:定时执行任务。例如每天早上 8 点定时推送早报。
其二:每隔一个时间段就执行任务。比如:每隔一个小时提醒自己起来走动走动,避免长时间坐着。

1 死循环使用线程睡眠函数sleep()

# -*- coding: utf-8 -*-
from datetime import datetime
import time


def timedTask():
    # 每隔5秒打印当前时间
    while True:
        print(datetime.now().strftime("%Y-%m-%d %H:%M:%S"))
        time.sleep(5)


if __name__ == '__main__':
    timedTask()

这种方法能够执行固定间隔时间的任务。如果timedTask()函数之后还有些操作,我们还使用死循环 + 阻塞线程。这会使得timedTask()一直占有 CPU 资源,导致后续操作无法执行。

2 threading中的Timer类

Python 标准库 threading 中有个 Timer 类。它会新启动一个线程来执行定时任务,所以它是非阻塞式。

如果你有使用多线程的话,需要关心线程安全问题。那么你可以选使用threading.Timer模块。

2.1 运行一次

# -*- coding: utf-8 -*-
from datetime import datetime
import threading
from threading import Timer
import time
import os


def timedTask():
    # 每隔5秒执行任务。
    # 第一个参数: 延迟多长时间执行任务(单位: 秒)
    # 第二个参数: 要执行的任务, 即函数
    # 第三个参数: 调用函数的参数(tuple)
    t = Timer(5, task, ())
    t.start()


# 待执行任务
def task():
    child_id = threading.get_ident()
    print("进程id=", os.getpid(), "子线程id=", child_id, datetime.now().strftime("%Y-%m-%d %H:%M:%S"))

if __name__ == '__main__':
    parent_id = threading.get_ident()
    timedTask()
    while True:
        print("进程id=", os.getpid(), "主线程id=", parent_id, time.time())
        time.sleep(10)

定时任务,启动了一个单独的线程,运行一次。
在这里插入图片描述

2.2 循环运行

# -*- coding: utf-8 -*-
from datetime import datetime
import threading
from threading import Timer
import time
import os


def timedTask():
    task()
    # 每隔5秒执行任务一次。
    # 第一个参数: 延迟多长时间执行任务(单位: 秒)
    # 第二个参数: 要执行的任务, 即函数
    # 第三个参数: 调用函数的参数(tuple)
    t = Timer(5, timedTask)
    t.start()


# 待执行任务
def task():
    child_id = threading.get_ident()
    print("进程id=", os.getpid(), "子线程id=", child_id, datetime.now().strftime("%Y-%m-%d %H:%M:%S"))

if __name__ == '__main__':
    parent_id = threading.get_ident()
    timedTask()
    while True:
        print("进程id=", os.getpid(), "主线程id=", parent_id, time.time())
        time.sleep(10)

循环运行,每次都启动一个新线程。
在这里插入图片描述

3 事件调度器sched

使用标准库中sched模块。sched 是事件调度器,它通过 scheduler 类来调度事件,从而达到定时执行任务的效果。

3.1 执行一次

# -*- coding: utf-8 -*-
from datetime import datetime
import sched
import time
import threading
import os

def timedTask():
    # 每隔5秒执行一次
    # 初始化sched模块的scheduler类
    scheduler = sched.scheduler(time.time, time.sleep)
    scheduler.enter(5, 1, task)  # 增加调度任务
    scheduler.run()  # 运行任务

# 待执行任务
def task():
    child_id = threading.get_ident()
    print("进程id=", os.getpid(), "子线程id=", child_id, datetime.now().strftime("%Y-%m-%d %H:%M:%S"))


if __name__ == '__main__':
    timedTask()

scheduler 中的每个调度任务只会工作一次,不会无限循环被调用。如果想重复执行同一任务, 需要重复添加调度任务即可。

3.2 循环运行

# -*- coding: utf-8 -*-
from datetime import datetime
import sched
import time
import threading
import os

def timedTask():
    task()
    # 每隔5秒执行一次
    # 初始化sched模块的scheduler类
    scheduler = sched.scheduler(time.time, time.sleep)
    scheduler.enter(5, 1, timedTask)  # 增加调度任务
    scheduler.run()  # 运行任务

# 待执行任务
def task():
    child_id = threading.get_ident()
    print("进程id=", os.getpid(), "子线程id=", child_id, datetime.now().strftime("%Y-%m-%d %H:%M:%S"))


if __name__ == '__main__':
    timedTask()

不启动新的线程
在这里插入图片描述

4 定时任务调度框架APScheduler

4.1 基础组件

APScheduler的全称是Advanced Python Scheduler。它是一个轻量级的 Python 定时任务调度框架。APScheduler 支持三种调度任务:固定时间间隔,固定时间点(日期),Linux 下的 Crontab 命令。同时,它还支持异步执行、后台执行调度任务。

在这里插入图片描述
APScheduler由四个组件构成:
(1)调度器(scheduler)
它是任务调度器,属于控制器角色。它配置作业存储器和执行器可以在调度器中完成,例如添加、修改和移除作业。
(2)作业存储(job store)
任务持久化仓库,默认保存任务在内存中,也可将任务保存都各种数据库中,任务中的数据序列化后保存到持久化数据库,从数据库加载后又反序列化。
(3)触发器(trigger)
描述调度任务被触发的条件。触发器完全是无状态的。
(4)执行器(executor)
负责处理作业的运行,它们通常通过在作业中提交指定的可调用对象到一个线程或者进城池来进行。当作业完成时,执行器将会通知调度器。

4.1.1 schedulers调度器

调度器是将其余部分绑定在一起的工具。通常只有一个调度器(scheduler)在应用程序中运行。应用程序开发者通常不直接处理作业存储(job stores)、执行器(executors)或者触发器(triggers)。相反,调度器提供了适当的接口来处理它们。配置作业存储(job stores)和执行器(executors)是通过调度器(scheduler)来完成的,就像添加、修改和删除 job(作业)一样。

它提供 7 种调度器,能够满足我们各种场景的需要。
(1)BlockingScheduler : 调度器在当前进程的主线程中运行,也就是会阻塞当前线程。
(2)BackgroundScheduler : 调度器在后台线程中运行,不会阻塞当前线程。
(3)AsyncIOScheduler : 结合 asyncio 模块(一个异步框架)一起使用。
(4)GeventScheduler : 程序中使用 gevent(高性能的Python并发框架)作为IO模型,和 GeventExecutor 配合使用。
(5)TornadoScheduler : 程序中使用 Tornado(一个web框架)的IO模型,用 ioloop.add_timeout 完成定时唤醒。
(6)TwistedScheduler : 配合 TwistedExecutor,用 reactor.callLater 完成定时唤醒。
(7)QtScheduler : 你的应用是一个 Qt 应用,需使用QTimer完成定时唤醒。

4.1.2 triggers触发器

触发器包含调度逻辑。每个作业(job)都有自己的触发器,用于确定下一个作业何时运行。除了最初的配置,触发器是完全无状态的。
APScheduler有三种内建的trigger:
(1)date触发器
date是最基本的一种调度,作业任务只会执行一次。它表示特定的时间点触发。它的参数如下:

参数									说明
run_date (datetime或str)			作业的运行日期或时间
timezone (datetime.tzinfo或str)		指定时区

示例如下:

# -*- coding: utf-8 -*-
from datetime import datetime
from datetime import date
import time
from apscheduler.schedulers.background import BackgroundScheduler
import threading
import os


def job_task(text):
    # 待执行任务
    child_id = threading.get_ident()
    print("进程id=", os.getpid(), "子线程id=", child_id, datetime.now().strftime("%Y-%m-%d %H:%M:%S"), text)


if __name__ == '__main__':
    scheduler = BackgroundScheduler()
    # 添加调度任务
    # 在 2023-05-18 时刻运行一次 job_func 方法
    scheduler.add_job(job_task, 'date', run_date=date(2023, 5, 18), args=['text1'])
    # 在 2023-05-18 15:57:00 时刻运行一次 job_func 方法
    scheduler.add_job(job_task, 'date', run_date=datetime(2023, 5, 18, 15, 57, 0), args=['text2'])
    # 在 2023-05-18 15:57:30 时刻运行一次 job_func 方法
    scheduler.add_job(job_task, 'date', run_date='2023-5-18 15:57:30', args=['text3'])
    # 启动调度任务
    scheduler.start()

    parent_id = threading.get_ident()
    while True:
        print("进程id=", os.getpid(), "主线程id=", parent_id, time.time())
        time.sleep(20)

创建新的子线程
在这里插入图片描述
(2)interval触发器
固定时间间隔触发。interval 间隔调度,参数如下:

参数								说明
weeks (int)						间隔几周
days (int)						间隔几天
hours (int)						间隔几小时
minutes (int)					间隔几分钟
seconds (int)					间隔多少秒
start_date (datetime或str)		开始日期
end_date (datetime或str)		结束日期
timezone (datetime.tzinfo或str)	时区

示例如下

# -*- coding: utf-8 -*-
from datetime import datetime
import time
from apscheduler.schedulers.background import BackgroundScheduler
import threading
import os


def job_task(text):
    # 待执行任务
    child_id = threading.get_ident()
    print("进程id=", os.getpid(), "子线程id=", child_id, datetime.now().strftime("%Y-%m-%d %H:%M:%S"), text)


if __name__ == '__main__':
    scheduler = BackgroundScheduler()
    # 添加调度任务
    # 每隔两分钟执行一次 job_func 方法
    scheduler.add_job(job_task, 'interval', minutes=1, args=['text1'])
    # 在start_date ~ end_date之间, 每隔1分钟执行一次 job_func 方法
    scheduler.add_job(job_task, 'interval', minutes=1,
                      start_date='2023-05-18 16:08:01', end_date='2023-05-18 16:10:01', args=['text2'])
    # 启动调度任务
    scheduler.start()

    parent_id = threading.get_ident()
    while True:
        print("进程id=", os.getpid(), "主线程id=", parent_id, time.time())
        time.sleep(20)

在这里插入图片描述
(3)cron触发器
在特定时间周期性地触发,和Linux crontab格式兼容。它是功能最强大的触发器。 我们先了解 cron 参数:

参数								说明
year (int 或 str)				年,4位数字
month (int 或 str)(范围1-12)
day (int 或 str)(范围1-31
week (int 或 str)(范围1-53)
day_of_week (int或str)			周内第几天或者星期几 (范围0-6 或者 mon,tue,wed,thu,fri,sat,sun)

hour (int 或 str)(范围0-23)
minute (int 或 str)(范围0-59)
second (int 或 str)(范围0-59)
start_date (datetime 或 str)	最早开始日期(包含)
end_date (datetime 或 str)		最晚结束时间(包含)
timezone (datetime.tzinfo或str)	指定时区

示例如下

# 在每年 1-3、7-9 月份中的每个星期一、二中的 00:00, 01:00, 02:00 和 03:00 执行 job_func 任务
scheduler .add_job(job_func, 'cron', month='1-3,7-9',day='0, tue', hour='0-3')

4.1.3 job stores作业存储

job stores 是存放作业的地方,默认保存在内存中。作业数据序列化后保存至持久性数据库,从持久性数据库加载回来时会反序列化。作业存储(job stores)不将作业数据保存在内存中(默认存储除外),相反,内存只是充当后端存储在保存、加载、更新、查找作业时的中间人角色。作业存储不能在调度器(schedulers) 之间共享。

该组件是对调度任务的管理,添加 job 有两种添加方法:
(1)其中一种上述代码用到的 add_job()。
(2)另一种则是scheduled_job()修饰器来修饰函数。

import datetime
from apscheduler.schedulers.background import BackgroundScheduler

@scheduler.scheduled_job(job_func, 'interval', minutes=2)
def job_func(text):
    print(datetime.datetime.utcnow().strftime("%Y-%m-%d %H:%M:%S.%f")[:-3])

scheduler = BackgroundScheduler()
scheduler.start()

这个两种办法的区别是:第一种方法返回一个 apscheduler.job.Job 的实例,可以用来改变或者移除 job。第二种方法只适用于应用运行期间不会改变的 job。

一、移除 job 移除 job 也有两种方法:remove_job() 和 job.remove()。 remove_job() 是根据 job 的 id 来移除,所以要在 job 创建的时候指定一个 id。 job.remove() 则是对 job 执行 remove 方法即可

scheduler.add_job(job_func, 'interval', minutes=2, id='job_one')
scheduler.remove_job(job_one)

job = add_job(job_func, 'interval', minutes=2, id='job_one')
job.remvoe()

二、获取 job 列表 通过 scheduler.get_jobs() 方法能够获取当前调度器中的所有 job 的列表

修改 job 如果你因计划改变要对 job 进行修改,可以使用Job.modify() 或者 modify_job()方法来修改 job 的属性。但是值得注意的是,job 的 id 是无法被修改的。

scheduler.add_job(job_func, 'interval', minutes=2, id='job_one')
scheduler.start()
# 将触发时间间隔修改成 5分钟
scheduler.modify_job('job_one', minutes=5)

job = scheduler.add_job(job_func, 'interval', minutes=2)
# 将触发时间间隔修改成 5分钟
job.modify(minutes=5)

三、关闭 job 默认情况下调度器会等待所有正在运行的作业完成后,关闭所有的调度器和作业存储。如果你不想等待,可以将 wait 选项设置为 False。

scheduler.shutdown()
scheduler.shutdown(wait=false)

4.1.4 executors执行器

执行器处理作业的运行。它们通常通过将作业中的指定可调用部分提交给线程或进程池来实现这一点。 当作业完成后,执行器通知调度器,然后调度器发出一个适当的事件。

执行器顾名思义是执行调度任务的模块。最常用的 executor 有两种:ProcessPoolExecutor 和 ThreadPoolExecutor。

4.2 使用步骤

APScheduler 使用起来还算是比较简单。运行一个调度任务只需要以下三步。
(1)新建一个 schedulers (调度器) 。
(2)添加一个调度任务(job stores)。
(3)运行调度任务。

BlockingScheduler: 当调度器是你应用中唯一要运行的东西时使用。
BackgroundScheduler: 当你不运行任何其他框架时使用,并希望调度器在你应用的后台执行。

选择合适的作业存储,你需要决定是否需要作业持久化。如果你总是在应用开始时重建job,你可以直接使用默认的作业存储(MemoryJobStore).但是如果你需要将你的作业持久化,以避免应用崩溃和调度器重启时,你可以根据你的应用环境来选择具体的作业存储。例如:使用Mongo或者SQLAlchemyJobStore (用于支持大多数RDBMS)。

默认的ThreadPoolExecutor 通常用于大多数用途。如果你的工作负载中有较大的CPU密集型操作,你可以考虑用ProcessPoolExecutor来使用更多的CPU核。你也可以在同一时间使用两者,将进程池调度器作为第二执行器。

# -*- coding: utf-8 -*-
from datetime import datetime
import time
from apscheduler.schedulers.background import BackgroundScheduler
import threading
import os

def timedTask():
    task()

# 待执行任务
def task():
    child_id = threading.get_ident()
    print("进程id=", os.getpid(), "子线程id=", child_id, datetime.now().strftime("%Y-%m-%d %H:%M:%S"))

if __name__ == '__main__':
    # 创建后台执行的 schedulers
    scheduler = BackgroundScheduler()
    # 添加调度任务
    # 调度方法为 timedTask,触发器选择 interval(间隔性),间隔时长为 1 秒
    scheduler.add_job(timedTask, 'interval', seconds=1)
    # 启动调度任务
    scheduler.start()

    parent_id = threading.get_ident()
    while True:
        print("进程id=", os.getpid(), "主线程id=", parent_id, time.time())
        time.sleep(5)

启动不同的线程处理。
在这里插入图片描述

4.3 使用job stores【mysql】

from datetime import datetime

from apscheduler.executors.pool import ThreadPoolExecutor, ProcessPoolExecutor
from apscheduler.jobstores.sqlalchemy import SQLAlchemyJobStore
from apscheduler.schedulers.background import BackgroundScheduler
import time
import threading
import os

def job_task(text):
    # 待执行任务
    child_id = threading.get_ident()
    print("进程id=", os.getpid(), "子线程id=", child_id, datetime.now().strftime("%Y-%m-%d %H:%M:%S"), text)

if __name__ == "__main__":
    user = 'root'  # 用户名
    password = 'bigdata'  # 密码
    host = '127.0.0.1'  # 主机IP
    port = '3306'  # 主机端口
    dbname = 'test'  # 数据库名称
    # 使用pymysql连接数据库,字符集为UTF8
    url = f"mysql+pymysql://{user}:{password}@{host}:{port}/{dbname}?charset=utf8"

    job_stores = {
        'default': SQLAlchemyJobStore(url=url, tablename='my_tasks')  # 定时任务表名为my_tasks
    }
    executors = {
        'default': ThreadPoolExecutor(20),
        'processpool': ProcessPoolExecutor(5)
    }
    job_defaults = {
        'coalesce': True,  # 堆积后只执行最后一个
        'max_instances': 1,  # 最大的实例只能存在一个

    }
    scheduler = BackgroundScheduler(executors=executors,
                                  job_defaults=job_defaults,
                                  jobstores=job_stores,
                                  timezone='Asia/Shanghai')

    scheduler.start()

    while True:
        time.sleep(10)
        scheduler.add_job(job_task, 'interval', args=["test5"], id="task5", seconds=2, replace_existing=True)
        scheduler.add_job(job_task, 'interval', args=["test6"], id="task6", seconds=2, replace_existing=True)

4.4 增加修改移除任务

task_list = scheduler.get_jobs()  # 获取最新的任务列表
task_id_list = [task.id for task in task_list]  # 获取最新的任务id列表
if flag == "add":
   # 增加任务
   if task_id not in task_id_list:
       scheduler.add_job(job_task, 'interval',
                         seconds=int(data_dict["Interval"]),
                         id=task_id,
                         start_date=data_dict["effect_start_time"],
                         end_date=data_dict["effect_end_time"],
                         args=[task_id])
       task_id_list.append(task_id)  # 更新任务列表
   print("add", task_id)

if flag == "delete":
   # 移除任务
   if task_id in task_id_list:
       scheduler.remove_job(job_id=task_id)
       task_id_list.remove(task_id)  # 更新任务列表
   print("delete",task_id)

if flag == "update":
   # 修改任务
   if task_id in task_id_list:
       scheduler_trigger = scheduler._create_trigger(
           trigger="interval",  # 指定新的执行任务方式,这里还是用的时间间隔
           trigger_args={"seconds": int(data_dict["Interval"])}  # 多少秒执行一次
       )
       scheduler.modify_job(job_id=task_id, trigger=scheduler_trigger)
   print("update",task_id)

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

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

相关文章

为什么要学习源码之Java篇

为什么学习源码 大厂面试必问。二次开发。提升代码阅读能力,更能输出优质代码。提升技术功底。拥抱开源社区。快速定位线上问题。 学习源码的方式 首先最重要的是学会使用。具有全局观。先对大致有个细节的了解,一开始不要太关注于细节。学会看注释&a…

【计算机网络】TCP传输控制协议——三次握手

文章目录 握手的流程常考考点 握手的流程 一开始,客户端和服务端都处于CLOSE状态,先是服务端监听某个端口,处于LISTEN状态。然后客户端主动发起连接SYN,之后处于SYN-SEND状态。服务端收到发起的连接,返回SYN&#xff0…

Spring Data Commons远程命令执行漏洞复现(CVE-2018-1273)

一、漏洞说明 Spring Data是一个用于简化数据库访问,并支持云服务的开源框架,包含Commons、Gemfire、JPA、JDBC、MongoDB等模块。此漏洞产生于Spring Data Commons组件,该组件为提供共享的基础框架,适合各个子项目使用,支持跨数据…

快速学会git版本管理——创建分支和合并分支

首先创建分支 git创建分支只需要使用switch 命令: git switch -c 分支名 创建分支并切换到该分支 大家看后面的括号里已经变成了dev 说明我们切换成功了。 然后想要合并分支就在 创建的分支中 进行提交修改的内容,还是通过:add 命令和co…

EasyExcel入门(最简单的读)

官网:EasyExcel官方文档 - 基于Java的Excel处理工具 | Easy Excel (alibaba.com) 因为暂时项目没有用到,所以不急,知道了这个技术。就想着学着用一下! 最简单的读 先看官方文档给的用法和解释!!&#xff01…

开开心心带你学习MySQL数据库之第八篇

索引和事务 ~~ 数据库运行的原理知识 面试题 索引 索引(index) > 目录 索引存在的意义,就是为了加快查找速度!!(省略了遍历的过程) 查找速度是快了,但是付出了一定的代价!! 1.需要付出额外的空间代价来保存索引数据 2.索引可能会拖慢新增,删除,修改的速度 ~~ …

信息系统项目管理师(第四版)教材精读思维导图-第十四章项目沟通管理

请参阅我的另一篇文章,综合介绍软考高项: 信息系统项目管理师(软考高项)备考总结_计算机技术与软件专业技术_铭记北宸的博客-CSDN博客 本章思维导图XMind源文件 14.1 管理基础 14.2 管理过程 14.3 规划沟通管理 14.4 管理沟通 14.…

机器学习:使用PCA简化数据

文章目录 使用场景主成分分析(Principal component analysis)实验:对半导体数据(590个特征)进行降维处理 使用场景 我们通过电视看实况足球,电视显示屏有100万个像素点,球所占的点数为100个。人…

Collectors类作用:

一、Collectors类: 1.1、Collectors介绍 Collectors类,是JDK1.8开始提供的一个的工具类,它专门用于对Stream操作流中的元素各种处理操作,Collectors类中提供了一些常用的方法,例如:toList()、toSet()、to…

真的有线上兼职吗?推荐几个靠谱的线上兼职!

在这个互联网普及,信息爆炸的时代,线上赚钱已经成为一个热门的话题。每个人都想通过互联网赚钱,有些人得到钱,给普通人机会,给骗子一些机会,世界是两面,线上兼职赚钱的方式!有好有坏…

python中的继承

要理解继承首先要有父类和子类的概念,可以理解成子类从父类中继承父类的属性和方法 创建父类 class Pet:def __init__(self,name,age):self.name nameself.age agedef jump(self):print(self.name"在跳")创建子类 class Cat(Pet):pass mycatCat(&quo…

HTML的有序列表、无序列表、自定义列表

目录 背景: 过程: 无序列表: 简介: 代码展示: 效果展示:​ 无序列表: 简介: 效果展示:​ 自定义列表: 简介: 效果展示: 总结: 背景: 1.有序列表(Ordered List): 有序列表是最早的列表类型之一&#xff…

编译器02-词法分析

一:简述 词法分析含义:为了翻译语言,编译器把程序各种成分拆开,那如何拆,首先第一步就是将输入分解成一个个独立的单词(token),这一过程叫词法分析。 二:单词(token)分为哪些种类 保留字…

类和对象:构造函数,析构函数与拷贝构造函数

1.类的6个默认成员函数 如果一个类中什么成员都没有,简称为空类。 空类中真的什么都没有吗?并不是,任何类在什么都不写时,编译器会自动生成以下6个默认成员函数。 默认成员函数:用户没有显式实现,编译器…

MAC M1芯片安装mounty读写移动硬盘中的文件

因为移动硬盘中的文件是微软公司NTFS格式,MAC只支持自己的APFS或者HFS,与微软的NTFS不兼容,所以需要第三方的软件来支持读写硬盘中的文件,经过一上午的折腾,最终选择安装mounty这个免费的第三方软件 工具网址连接&am…

C++11 类型转换

#include<iostream> #include<string> using namespace std;//类型转换 void Insert(size_t pos, char ch) {size_t _size 5;//....int end _size - 1;while (end > pos) // end隐式类型转换{//_str[end 1] _str[end];--end;} } void Test1() {int i 1;// …

C++:内存管理

目录 学习目标 1. C/C内存分布 2. C/C动态内存管理 2.1 C动态内存管理 2.2 C动态内存管理&#xff1a; 3. operator new与operator delete函数 4. new和delete的实现原理 5. 定位new表达式 学习目标 1. C/C内存分布 2. C/C动态内存管理&#xff08;malloc,free 与new ,…

【深度学习实战—6】:基于Pytorch的血细胞图像分类(通用型图像分类程序)

✨博客主页&#xff1a;米开朗琪罗~&#x1f388; ✨博主爱好&#xff1a;羽毛球&#x1f3f8; ✨年轻人要&#xff1a;Living for the moment&#xff08;活在当下&#xff09;&#xff01;&#x1f4aa; &#x1f3c6;推荐专栏&#xff1a;【图像处理】【千锤百炼Python】【深…

SQL Server2022版本 + SSMS安装教程(手把手安装教程)

SqlServer安装步骤如下&#xff1a; 下载请点点我 1.选择Developer版 2.点击浏览器下载标志&#xff0c;找到SqlServer在文件夹的位置。 3.进入下面的界面&#xff1a;选择自定义版本 4.将下载的C盘的盘符改为D盘 5.点击安装–进入下一步&#xff0c;显示如下界面。 过…

构建知识库的核心要义,试试我的工具和方法吧!

对于企业来说&#xff0c;如果所有人员的知识和经验&#xff0c;都集中沉淀到统一的文档系统中&#xff0c;那么&#xff0c;久而久之&#xff0c;会形成一个丰富的知识和经验库。 在构建知识库之前&#xff0c;我们要先确定知识库范围包含&#xff1a;素材整理、问题提炼、知识…