SpringBoot源码解读与原理分析(三十二)SpringBoot整合JDBC(一)JDBC组件的自动装配

文章目录

  • 前言
  • 第10章 SpringBoot整合JDBC
    • 10.1 SpringBoot整合JDBC的项目搭建
      • 10.1.1 初始化数据库
      • 10.1.2 整合项目
        • 10.1.2.1 导入JDBC和MySQL驱动依赖
        • 10.1.2.2 配置数据源
      • 10.1.3 编写业务代码
        • 10.1.3.1 编写与t_user表对应的实体类User
        • 10.1.3.2 编写Dao层代码
        • 10.1.3.3 编写Service层代码
      • 10.1.4 编写主启动类
      • 10.1.5 测试结果
    • 10.2 整合JDBC后的自动装配
      • 10.2.1 配置数据源
        • 10.2.1.1 DataSourceInitializerInvoker
          • (1)createSchema
          • (2)initialize
        • 10.2.1.2 DataSourceInitializerPostProcessor
      • 10.2.2 创建JdbcTemplate
      • 10.2.3 配置事务管理器

前言

在实际SpringBoot项目中,通常都离不开与数据库的交互,更多的选择是使用持久层框架MyBatis或SpringData等,而不是原生的spring-jdbc。

但学习SpringBoot整合JDBC场景下的组件装配,以及注解声明式事务的生效原理、控制流程、事务传播行为等,依旧是必要的,对后续学习SpringBoot整合持久层框架具有很大帮助。

第10章 SpringBoot整合JDBC

10.1 SpringBoot整合JDBC的项目搭建

10.1.1 初始化数据库

选择MySQL作为本项目的数据源,创建一个新的数据库springboot_demo和一个新表t_user:

CREATE DATABASE springboot_demo CHARACTER SET 'utf8mb4';CREATE TABLE t_user(id INT(11) NOT NULL AUTO_INCREMENT,NAME VARCHAR(20) NOT NULL,tel VARCHAR(20) NULL,PRIMARY KEY (id)
);

10.1.2 整合项目

10.1.2.1 导入JDBC和MySQL驱动依赖
<dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-jdbc</artifactId></dependency><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>5.1.47</version></dependency>
</dependencies>
10.1.2.2 配置数据源

在resources目录下新建application.properties文件,并配置数据源:

spring.datasource.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.url=jdbc:mysql://localhost:3306/springboot_demo?characterEncoding=utf8
spring.datasource.username=root
spring.datasource.password=123456

10.1.3 编写业务代码

10.1.3.1 编写与t_user表对应的实体类User
public class User {private Integer id;private String name;private String tel;// getter setter toString ...
}
10.1.3.2 编写Dao层代码
@Repository
public class UserDao {@Autowiredprivate JdbcTemplate jdbcTemplate;public void save(User user) {jdbcTemplate.update("insert into t_user (name, tel) values (?, ?)",user.getName(), user.getTel());}public List<User> findAll() {return jdbcTemplate.query("select * from t_user",BeanPropertyRowMapper.newInstance(User.class));}}
10.1.3.3 编写Service层代码
@Service
public class UserService {@Autowiredprivate UserDao userDao;@Transactional(rollbackFor = Exception.class)public void test() {User user = new User();user.setName("齐天大圣");user.setTel("12306");userDao.save(user);List<User> userList = userDao.findAll();userList.forEach(System.out::println);}}

10.1.4 编写主启动类

主启动类注意两点:第一是要获取IOC容器,并提取出UserService类调用其test方法;第二是要标注@EnableTransactionManagement注解开启注解式声明式事务。

@SpringBootApplication
@EnableTransactionManagement
public class JDBCApp {public static void main(String[] args) {ConfigurableApplicationContext context = SpringApplication.run(JDBCApp.class, args);UserService userService = context.getBean(UserService.class);userService.test();}}

10.1.5 测试结果

运行主启动类,控制台可以正确打印出一条用户信息,说明SpringBoot整合JDBC场景顺利完成。

User{id=1, name='齐天大圣', tel='12306'}

10.2 整合JDBC后的自动装配

由 10.1 节可知,对于原生的JDBC整合后,主启动类中并没有声明与之相关的注解,因此有关JDBC的组件装配都是以自动配置类的方式实现的。

借助IDEA通过spring-boot-autoconfigure依赖的spring.factories文件可以找到有关JDBC的自动配置类:

源码1:spring-boot-autoconfigure-2.3.11.RELEASE.jar!/META-INF/spring.factoriesorg.springframework.boot.autoconfigure.EnableAutoConfiguration=\
org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.JdbcTemplateAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.JndiDataSourceAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.XADataSourceAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.DataSourceTransactionManagerAutoConfiguration,\

可以发现,SpringBoot默认支持的自动配置包含数据源、JdbcTemplate、事务管理器、JNDI、XA协议等。

10.2.1 配置数据源

在SpringBoot的官方文档中,介绍了SpringBoot支持的数据源连接池:

1.1.4. Supported Connection Pools
Spring Boot uses the following algorithm for choosing a specific implementation:
We prefer HikariCP for its performance and concurrency. If HikariCP is available, we always choose it.
Otherwise, if the Tomcat pooling DataSource is available, we use it.
Otherwise, if Commons DBCP2 is available, we use it.
If none of HikariCP, Tomcat, and DBCP2 are available and if Oracle UCP is available, we use it.

可见,SpringBoot支持的数据源连接池包括HikariCP、Tomcat、DBCP2、Oracle UCP等,默认使用HikariCP

因此,自动配置类DataSourceAutoConfiguration的静态内部类PooledDataSourceConfiguration会生效。因为它通过@Import注解导入了DataSourceConfiguration.Hikari类。

源码2DataSourceAutoConfiguration.java@Configuration(proxyBeanMethods = false)
@ConditionalOnClass({ DataSource.class, EmbeddedDatabaseType.class })
@ConditionalOnMissingBean(type = "io.r2dbc.spi.ConnectionFactory")
@EnableConfigurationProperties(DataSourceProperties.class)
@Import({ DataSourcePoolMetadataProvidersConfiguration.class, DataSourceInitializationConfiguration.class })
public class DataSourceAutoConfiguration {@Configuration(proxyBeanMethods = false)@Conditional(PooledDataSourceCondition.class)@ConditionalOnMissingBean({ DataSource.class, XADataSource.class })@Import({ DataSourceConfiguration.Hikari.class, DataSourceConfiguration.Tomcat.class,DataSourceConfiguration.Dbcp2.class, DataSourceConfiguration.Generic.class,DataSourceJmxConfiguration.class })protected static class PooledDataSourceConfiguration {}
}
源码3DataSourceConfiguration.java@Configuration(proxyBeanMethods = false)
@ConditionalOnClass(HikariDataSource.class)
@ConditionalOnMissingBean(DataSource.class)
@ConditionalOnProperty(name = "spring.datasource.type", havingValue = "com.zaxxer.hikari.HikariDataSource",matchIfMissing = true)
static class Hikari {@Bean@ConfigurationProperties(prefix = "spring.datasource.hikari")HikariDataSource dataSource(DataSourceProperties properties) {HikariDataSource dataSource = createDataSource(properties, HikariDataSource.class);if (StringUtils.hasText(properties.getName())) {dataSource.setPoolName(properties.getName());}return dataSource;}
}

由 源码2、3 可知,SpringBoot默认会创建一个HikariDataSource。

此外,DataSourceAutoConfiguration还使用@Import注解导入了一个DataSourceInitializationConfiguration配置类。

源码4DataSourceInitializationConfiguration.java@Configuration(proxyBeanMethods = false)
@Import({DataSourceInitializerInvoker.class, DataSourceInitializationConfiguration.Registrar.class})
class DataSourceInitializationConfiguration {static class Registrar implements ImportBeanDefinitionRegistrar {private static final String BEAN_NAME = "dataSourceInitializerPostProcessor";@Overridepublic void registerBeanDefinitions(AnnotationMetadata importingClassMetadata,BeanDefinitionRegistry registry) {if (!registry.containsBeanDefinition(BEAN_NAME)) {// 注册DataSourceInitializerPostProcessorGenericBeanDefinition beanDefinition = new GenericBeanDefinition();beanDefinition.setBeanClass(DataSourceInitializerPostProcessor.class);beanDefinition.setRole(BeanDefinition.ROLE_INFRASTRUCTURE);beanDefinition.setSynthetic(true);registry.registerBeanDefinition(BEAN_NAME, beanDefinition);}}}}

由 源码4 可知,这个配置类又使用@Import注解导入了一个DataSourceInitializerInvoker和一个Registrar注册器。注册器又向BeanDefinition中注册了一个DataSourceInitializerPostProcessor。

10.2.1.1 DataSourceInitializerInvoker

由类名理解,这是一个数据源初始化的执行器

源码5DataSourceInitializerInvoker.javaclass DataSourceInitializerInvoker implements ApplicationListener<DataSourceSchemaCreatedEvent>, InitializingBean {@Overridepublic void afterPropertiesSet() {DataSourceInitializer initializer = getDataSourceInitializer();if (initializer != null) {boolean schemaCreated = this.dataSourceInitializer.createSchema();if (schemaCreated) {initialize(initializer);}}}}

由 源码5 可知,DataSourceInitializerInvoker实现了ApplicationListener接口,因此是一个监听器,监听的事件是DataSourceSchemaCreatedEvent;此外,它还实现了InitializingBean接口,会在对象创建后回调afterPropertiesSet方法执行初始化逻辑。

afterPropertiesSet方法中,首先会调用getDataSourceInitializer方法获取DataSourceInitializer实例,随后执行DataSourceInitializer的createSchema方法,如果执行成功则继续执行initialize方法。

(1)createSchema

该方法名直译为“创建约束”,即执行DDL语句。也就在项目开发中,可以先不创建数据库的表结构,而是在应用程序启动时,自动读取自定义的SQL文件执行DDL语句进行创建。

源码6DataSourceInitializer.javaboolean createSchema() {// 加载Schema资源List<Resource> scripts = getScripts("spring.datasource.schema", this.properties.getSchema(), "schema");// 解析Schema资源if (!scripts.isEmpty()) {if (!isEnabled()) {logger.debug("Initialization disabled (not running DDL scripts)");return false;}String username = this.properties.getSchemaUsername();String password = this.properties.getSchemaPassword();runScripts(scripts, username, password);}return !scripts.isEmpty();
}private List<Resource> getScripts(String propertyName, List<String> resources, String fallback) {if (resources != null) {// 如果全局配置文件中配置了spring.datasource.schema属性// 则直接根据该spring.datasource.schema属性加载资源文件return getResources(propertyName, resources, true);}// 默认返回字符串"all"String platform = this.properties.getPlatform();List<String> fallbackResources = new ArrayList<>();// 拼接文件名:schema-all.sql和schema.sqlfallbackResources.add("classpath*:" + fallback + "-" + platform + ".sql");fallbackResources.add("classpath*:" + fallback + ".sql");// 加载文件schema-all.sql和schema.sqlreturn getResources(propertyName, fallbackResources, false);
}

由 源码6 可知,createSchema方法先读取全局配置文件中的spring.datasource.schema属性,根据该属性配置的路径加载资源文件,再对资源文件进行解析。在调用getScripts方法读取资源文件时,分为两种情况:

  • 如果全局配置文件中配置了spring.datasource.schema属性,则直接根据该属性的值加载资源文件;
  • 如果全局配置文件没有配置该属性,则加载名为schema-all.sql和schema.sql的资源文件。
(2)initialize
源码7DataSourceInitializerInvoker.javaprivate void initialize(DataSourceInitializer initializer) {try {// 广播DataSourceSchemaCreatedEvent事件this.applicationContext.publishEvent(new DataSourceSchemaCreatedEvent(initializer.getDataSource()));if (!this.initialized) {// 解析资源文件this.dataSourceInitializer.initSchema();this.initialized = true;}} // catch ......
}@Override
public void onApplicationEvent(DataSourceSchemaCreatedEvent event) {DataSourceInitializer initializer = getDataSourceInitializer();if (!this.initialized && initializer != null) {initializer.initSchema();this.initialized = true;}
}
源码8DataSourceInitializer.javavoid initSchema() {// 加载资源文件List<Resource> scripts = getScripts("spring.datasource.data", this.properties.getData(), "data");// 解析资源文件if (!scripts.isEmpty()) {if (!isEnabled()) {logger.debug("Initialization disabled (not running data scripts)");return;}String username = this.properties.getDataUsername();String password = this.properties.getDataPassword();runScripts(scripts, username, password);}
}

由 源码7 可知,initialize方法会广播一个DataSourceSchemaCreatedEvent事件,回调onApplicationEvent方法,最终执行DataSourceInitializer类的initSchema方法。

由 源码8 可知,initSchema方法和createSchema方法大同小异,不同的是initSchema方法寻找资源文件的依据是全局配置文件中的spring.datasource.data属性,如果没有配置该属性,则加载名为data-all.sql和data.sql的资源文件。

总结一下,有了DataSourceInitializerInvoker的设计,使得项目开发中,可以自定义DDL语句和DML语句并保存在SQL文件中,放置在resources目录下,在项目启动时自动初始化数据库表结构和数据。

10.2.1.2 DataSourceInitializerPostProcessor

由类名理解,这是一个专门为DataSourceInitializer定制的后置处理器。

源码9DataSourceInitializerPostProcessor.java@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {if (bean instanceof DataSource) {this.beanFactory.getBean(DataSourceInitializerInvoker.class);}return bean;
}

由 源码9 可知,当DataSourceInitializerPostProcessor检测到当前正在创建的bean对象的类型是DataSource,主动调用BeanFactory的getBean方法创建一个DataSourceInitializerInvoker对象。这样做的目的是使预定定义好的SQL脚本立即执行,以确保DataSource与数据库表结构、数据的同步初始化。

10.2.2 创建JdbcTemplate

源码10JdbcTemplateAutoConfiguration.java@Configuration(proxyBeanMethods = false)
@ConditionalOnClass({ DataSource.class, JdbcTemplate.class })
@ConditionalOnSingleCandidate(DataSource.class)
@AutoConfigureAfter(DataSourceAutoConfiguration.class)
@EnableConfigurationProperties(JdbcProperties.class)
@Import({ JdbcTemplateConfiguration.class, NamedParameterJdbcTemplateConfiguration.class })
public class JdbcTemplateAutoConfiguration {}

由 源码10 可知,自动配置类JdbcTemplateAutoConfiguration会使用@Import注解注册一个JdbcTemplateConfiguration配置类和一个NamedParameterJdbcTemplateConfiguration配置类。

源码11JdbcTemplateConfiguration.java@Configuration(proxyBeanMethods = false)
@ConditionalOnMissingBean(JdbcOperations.class)
class JdbcTemplateConfiguration {@Bean@PrimaryJdbcTemplate jdbcTemplate(DataSource dataSource, JdbcProperties properties) {JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);JdbcProperties.Template template = properties.getTemplate();jdbcTemplate.setFetchSize(template.getFetchSize());jdbcTemplate.setMaxRows(template.getMaxRows());if (template.getQueryTimeout() != null) {jdbcTemplate.setQueryTimeout((int) template.getQueryTimeout().getSeconds());}return jdbcTemplate;}}

由 源码11 可知,JdbcTemplateConfiguration配置类会注册一个JdbcTemplate对象,用于与数据库的简单交互。

源码12NamedParameterJdbcTemplateConfiguration.java@Configuration(proxyBeanMethods = false)
@ConditionalOnSingleCandidate(JdbcTemplate.class)
@ConditionalOnMissingBean(NamedParameterJdbcOperations.class)
class NamedParameterJdbcTemplateConfiguration {@Bean@PrimaryNamedParameterJdbcTemplate namedParameterJdbcTemplate(JdbcTemplate jdbcTemplate) {return new NamedParameterJdbcTemplate(jdbcTemplate);}}

由 源码12 可知,NamedParameterJdbcTemplateConfiguration配置类会注册一个NamedParameterJdbcTemplate对象,用于支持参数命名化的JdbcTemplate增强。

10.2.3 配置事务管理器

源码13DataSourceTransactionManagerAutoConfiguration.java@Configuration(proxyBeanMethods = false)
@ConditionalOnClass({JdbcTemplate.class, PlatformTransactionManager.class})
@AutoConfigureOrder(Ordered.LOWEST_PRECEDENCE)
@EnableConfigurationProperties(DataSourceProperties.class)
public class DataSourceTransactionManagerAutoConfiguration {@Configuration(proxyBeanMethods = false)@ConditionalOnSingleCandidate(DataSource.class)static class DataSourceTransactionManagerConfiguration {@Bean@ConditionalOnMissingBean(PlatformTransactionManager.class)DataSourceTransactionManager transactionManager(DataSource dataSource,ObjectProvider<TransactionManagerCustomizers> transactionManagerCustomizers) {DataSourceTransactionManager transactionManager = new DataSourceTransactionManager(dataSource);transactionManagerCustomizers.ifAvailable((customizers) -> customizers.customize(transactionManager));return transactionManager;}}}

由 源码13 可知,自动配置类DataSourceTransactionManagerAutoConfiguration会注册一个DataSourceTransactionManager,用于支持基于数据源的事务控制。

······

本节完,更多内容请查阅分类专栏:SpringBoot源码解读与原理分析

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

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

相关文章

golang实现一个BasicAuth的HTTP server

之前写的《golang实现一个简单的HTTP server》没有包含认证部分 本例给出了支持BasicAuth的实现&#xff0c;以及如何在一个项目中导入自己定义的package 编写代码 创建项目所在文件夹 adminhpc-1:~/go$ mkdir auth_http adminhpc-1:~/go$ cd auth_http adminhpc-1:~/go/auth…

Oracle内存计算应用模式

前言 内存计算是利用内存来加速数据访问和应用的性能&#xff0c;并降低应用开发复杂度的技术。近十年来&#xff0c;随着软硬件技术的发展和用户需求的成熟&#xff0c;内存计算技术已经得到了广泛地应用。 Oracle在内存计算领域具有非常重要的地位&#xff0c;这主要得益于…

spring(六):事务(概念介绍、基于注解使用、事务属性介绍)

1. 事务基本概念 1.1 什么是事务 数据库事务( transaction)是访问并可能操作各种数据项的一个数据库操作序列&#xff0c;这些操作要么全部执行,要么全部不执行&#xff0c;是一个不可分割的工作单位。事务由事务开始与事务结束之间执行的全部数据库操作组成。 1.2 事务的特…

如果我有一台服务器的话

如果我拥有一台服务器&#xff0c;我可以设想几种不同且具有创新价值的用途&#xff0c;每种用途都充分利用了服务器的计算能力和网络连接优势。 1. 部署个人创意项目 机器学习/数据分析平台&#xff1a; 我可能会选择搭建一个基于服务器的机器学习服务平台&#xff0c;用于训…

数据库应用:Windows 部署 MySQL 8.0.36

目录 一、实验 1.环境 2.Windows 部署 MySQL 8.0.36 3.Windows配置环境变量 4.Navicat链接MySQL 二、问题 1.安装MySQL 报错 一、实验 1.环境 &#xff08;1&#xff09;主机 表1 主机 主机软件版本IP备注WindowsMySQL8.0.36localhost 2.Windows 部署 MySQL 8.0.…

最新Sora人工智能视频资源网址分享

1&#xff0c;了解什么是Sora * 什么是 OpenAI Sora&#xff1f; Sora 是由 OpenAI 开发的文本到视频模型。它可以按照用户的提示生成长达一分钟的高质量和一致的视频。 * 如何使用 OpenAI Sora 模型&#xff1f; 目前&#xff0c;OpenAI Sora 模型处于内测阶段&#xff0c;并将…

Linux软件高级编程-进程基本概念--day6

1.进程&#xff1a; 程序&#xff1a; 存放在外存的一段数据组成的文件 进程&#xff1a; 是一个程序动态执行的过程&#xff0c;包括进程的创建、进程的调度、进程的消亡 2.进程相关命令&#xff1a; 1&#xff09;top&#xff1a; 动态查看当前系统中所有进程信息&#xff08…

Linux的系统错误

strerror()函数是用于获取指定错误码对应的错误信息字符串的函数。它的声明为&#xff1a; #include <string.h> char *strerror(int errnum);该函数接收一个整型参数errnum&#xff0c;表示错误码&#xff0c;然后返回一个指向错误信息字符串的指针。 使用strerror()函…

STM32程序(移植)中头文件的路径

例:#include "./BSP/LCD/lcd.h"为什么有的头文件加了路径? 先看AI的回答: 在C和C编程中&#xff0c;当我们在源文件中包含&#xff08;或称为“引入”或“导入”&#xff09;一个头文件时&#xff0c;编译器需要知道这个头文件的位置。通常&#xff0c;编译器会在…

FariyGUI × Cocos Creator 3.x 弹窗制作

在fgui里制作一个弹窗 新建一个按钮&#xff0c;作为返回按钮 新建一个标签 做成这个样子 其中包含两个节点&#xff0c;名称分别为title和closeButton 可以阅读fgui的源码window.js得到&#xff0c;closeButton按钮只需要输入名称即可在contentPane设置时自动绑定。 且会…

【C++ QT项目5】——基于HTTP与JSON数据流的天气预报界面设计

【C QT项目5】——基于HTTP与JSON数据流的天气预报界面设计 一、项目概述二、UI设计与stylesheet样式表三、天气预报数据接口四、JSON数据4.1 概述4.2 QT生成JSON数据4.3 QT解析JSON数据4.4 将JSON数据解析到QMap中 五、软件开发网络通信架构5.1 BS架构/CS架构5.2 HTTP基本概念…

Object转List

1.背景 工作中经常会遇到一个map存key为string类型 value存object&#xff0c;方便我们下文代码获取数据 2.例如 Map<String, Object> result new HashMap<>(); List<Map<String, Object>> sheet1Result new ArrayList<>(); List<String&…

二、系统知识笔记-系统架构概述

一、系统架构定义 系统架构是指对一个系统的整体结构和组成部分进行描述和规划的过程。系统架构定义决定了系统的设计、开发和实施过程中的关键方向和决策。是系统的骨架和根基&#xff0c;支撑和链接各个部分&#xff0c;包括组件、连接件、约束规范以及指导这些内容设计与演…

go语言的切片理解

var numbers4 [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}myslice : numbers4[4:6]//这打印出来长度为2fmt.Printf("myslice为 %d, 其长度为: %d\n", myslice, len(myslice))myslice myslice[:cap(myslice)]//为什么 myslice 的长度为2&#xff0c;却能访问到第四个元…

MyBatis使⽤PageHelper(MySQL)

MyBatis使⽤PageHelper&#xff08;MySQL&#xff09; 一、 limit分⻚二、PageHelper插件第⼀步&#xff1a;引⼊依赖第⼆步&#xff1a;在mybatis-config.xml⽂件中配置插件第三步&#xff1a;编写Java代码第四步&#xff1a;格式化结果查看 三、SpringBoot3 集成 PageHelper …

自动驾驶---Motion Planning之Path Boundary

1 背景 在上文《自动驾驶---Motion Planning之LaneChange》中,笔者提到过两种LaneChange的思路,这里再简单回顾一下:(1)利用Routing和周围环境的信息,决定是否进行换道的决策;(2)采用的博弈思想(蒙特卡洛树搜索---MCTS)决定是否进行换道的决策。不管是变道,避让还是…

2024-02-25 Unity 编辑器开发之编辑器拓展6 —— Event

文章目录 1 Event 介绍2 重要 API3 代码示例 1 Event 介绍 ​ Event 提供许多属性和方法&#xff0c;允许检查和处理用户输入&#xff0c;主要用于 Unity 编辑器拓展开发。 ​ Input 相关内容需要在运行时才能监听输入&#xff0c;而 Event 专门提供给编辑模式下使用&#xf…

爬虫入门四(抽屉半自动点赞、xpath使用、动作链、打码平台、scrapy框架介绍与安装及创建项目)

文章目录 一、抽屉半自动点赞二、xpath的使用三、动作链四、打码平台介绍超级鹰打码基本测试 五、自动登录超级鹰六、scrapy框架介绍安装创建爬虫项目 一、抽屉半自动点赞 登录抽屉账号保存cookiesimport timeimport jsonfrom selenium import webdriverfrom selenium.webdrive…

(正规api接口代发布权限)短视频账号矩阵系统源码独立搭建技术开发

短视频账号矩阵系统源码独立搭建技术开发&#xff1a; 搭建短视频账号矩阵系统源码独立技术开发需要以下步骤&#xff1a; 1. 确定需求&#xff1a;明确系统需要实现的功能和目标&#xff0c;包括账号管理、视频发布、数据分析等。 2. 学习相关知识和技术&#xff1a;了解短视…

Adobe将类ChatGPT集成到PDF中

2月21日&#xff0c;全球多媒体巨头Adobe在官网宣布&#xff0c;推出生成式AI助手AI Assistant&#xff0c;并将其集成在Reader 和Acrobat 两款PDF阅读器中。 据悉&#xff0c;AI Assistant的功能与ChatGPT相似&#xff0c;可以基于PDF文档提供摘要、核心见解、基于文档内容&a…