Plugin - 插件开发04_Spring Boot中的SPI机制与Spring Factories实现
文章目录
- Pre
- 方案概览
- 使用插件的好处
- Spring Boot中的SPI机制与Spring Factories实现
- 1. Spring Boot中的SPI机制
- Spring Factories文件
- 2. Spring Factories实现原理
- 3. Code
- 3.1 定义一个服务接口
- 3.2 定义 实现类
- 3.3 配置 `spring.factories` 文件
- 3.4 创建一个Controller来加载插件
- 3.5 测试效果
- 小结

Pre
插件 - 通过SPI方式实现插件管理
插件 - 一份配置,离插件机制只有一步之遥
插件 - 插件机制触手可及
Plugin - 插件开发01_SPI的基本使用
Plugin - 插件开发02_使用反射机制和自定义配置实现插件化开发
Plugin - 插件开发03_Spring Boot动态插件化与热加载
Plugin - 插件开发04_Spring Boot中的SPI机制与Spring Factories实现
方案概览
常用的插件化实现思路:
- spi机制
- spring内置扩展点
- spring aop技术
- springboot中的Factories机制
- 第三方插件包:spring-plugin
- java agent 技术
使用插件的好处
- 模块解耦:插件机制能帮助系统模块间解耦,使得服务间的交互变得更加灵活。
- 提升扩展性和开放性:通过插件机制,系统能够轻松扩展,支持新的功能或服务,无需大规模修改核心代码。
- 方便第三方接入:第三方可以按照预定义的插件接口进行集成,不会对主系统造成过多侵入。
Spring Boot中的SPI机制与Spring Factories实现
Spring Boot作为一个快速开发框架,不仅简化了应用的开发过程,还提供了丰富的扩展点,其中最常用且强大的扩展机制之一就是Spring Factories。这一机制基于SPI(Service Provider Interface)设计理念,允许我们通过配置文件自动加载并实例化特定接口的实现类。这为插件化开发提供了理想的基础。
1. Spring Boot中的SPI机制
在Java中,SPI(Service Provider Interface)机制允许我们通过配置文件将接口与其实现类进行绑定。Spring Boot借鉴了这一机制,通过解析 META-INF/spring.factories
文件来加载指定接口的实现类。Spring Boot Starter正是基于这种机制实现的,它通过读取 spring.factories
文件来加载特定的配置和功能。
Spring Factories文件
Spring Boot通过META-INF/spring.factories
文件来管理和加载接口的实现类。在这个文件中,配置了接口及其对应的实现类,全类名被以key=value
的方式列出。Spring Boot应用程序在启动时,会扫描所有的jar包中的spring.factories
文件,并根据配置加载接口的实现类。
2. Spring Factories实现原理
Spring的核心包(spring-core
)中定义了SpringFactoriesLoader
类,这个类实现了从spring.factories
文件中加载接口实现类的功能。其核心方法有:
loadFactories(Class<?> factoryClass, ClassLoader classLoader)
:根据指定的接口类(如SmsPlugin
),返回实现类的实例列表。loadFactoryNames(Class<?> factoryClass, ClassLoader classLoader)
:根据接口类获取所有实现类的全类名。
具体的实现代码如下:
public static List<String> loadFactoryNames(Class<?> factoryClass, ClassLoader classLoader) {// 获取工厂类的全限定名String factoryClassName = factoryClass.getName();try {// 根据提供的类加载器获取资源文件的枚举集合Enumeration<URL> urls = (classLoader != null ? classLoader.getResources(FACTORIES_RESOURCE_LOCATION) :ClassLoader.getSystemResources(FACTORIES_RESOURCE_LOCATION));// 创建一个列表来存储工厂类名List<String> result = new ArrayList<String>();// 遍历所有找到的资源文件while (urls.hasMoreElements()) {URL url = urls.nextElement();// 加载属性文件Properties properties = PropertiesLoaderUtils.loadProperties(new UrlResource(url));// 获取指定工厂类名对应的工厂类名字符串String factoryClassNames = properties.getProperty(factoryClassName);// 将逗号分隔的字符串转换为列表并添加到结果列表中result.addAll(Arrays.asList(StringUtils.commaDelimitedListToStringArray(factoryClassNames)));}// 返回结果列表return result;} catch (IOException ex) {// 如果发生IO异常,抛出非法参数异常throw new IllegalArgumentException("无法从位置 [" + FACTORIES_RESOURCE_LOCATION + "] 加载 [" + factoryClass.getName() + "] 工厂", ex);}
}
在上述代码中,loadFactoryNames
方法会遍历整个ClassLoader中所有jar包下的spring.factories
文件,然后从中解析出接口对应的实现类列表。这意味着我们可以在自己的jar包中定义spring.factories
文件,而不会影响到其他模块的配置。
spring.factories
文件采用Properties
格式,内容一般为:
com.xxx.interface=com.xxx.classname
如果一个接口有多个实现类,可以使用,
来分隔。
3. Code
3.1 定义一个服务接口
首先,定义一个简单的服务接口,BootFileStorePlugin
,它包含一个saveFile的接口方法:
package com.artisan.plugins.bootspi.spi;import java.io.File;public interface BootFileStorePlugin {void saveFile(File file);
}
3.2 定义 实现类
接着,我们创建几个实现类, 它们分别实现了 BootFileStorePlugin
接口。
package com.artisan.plugins.bootspi.impl;import com.artisan.plugins.bootspi.spi.BootFileStorePlugin;
import lombok.extern.slf4j.Slf4j;import java.io.File;@Slf4j
public class BootFtpStorage implements BootFileStorePlugin {@Overridepublic void saveFile(File file) {log.info("Boot Ftp saveFile execute");}
}
package com.artisan.plugins.bootspi.impl;import com.artisan.plugins.bootspi.spi.BootFileStorePlugin;
import lombok.extern.slf4j.Slf4j;import java.io.File;@Slf4j
public class BootLocalStorage implements BootFileStorePlugin {@Overridepublic void saveFile(File file) {log.info("Boot Local saveFile execute");}
}
package com.artisan.plugins.bootspi.impl;import com.artisan.plugins.bootspi.spi.BootFileStorePlugin;
import lombok.extern.slf4j.Slf4j;import java.io.File;@Slf4j
public class BootS3Storage implements BootFileStorePlugin {@Overridepublic void saveFile(File file) {log.info("Boot S3 saveFile execute");}
}
3.3 配置 spring.factories
文件
在resources
目录下创建META-INF
目录,并在该目录下定义一个spring.factories
文件,内容如下:
com.artisan.plugins.bootspi.spi.BootFileStorePlugin=\com.artisan.plugins.bootspi.impl.BootFtpStorage,\com.artisan.plugins.bootspi.impl.BootLocalStorage,\com.artisan.plugins.bootspi.impl.BootS3Storage
3.4 创建一个Controller来加载插件
现在,我们需要创建一个Spring Boot的Controller来加载并使用这些插件:
package com.artisan.plugins.bootspi.test;import com.artisan.plugins.bootspi.spi.BootFileStorePlugin;
import com.artisan.plugins.customconfig.config.CustomPluginConfig;
import jakarta.annotation.Resource;
import org.springframework.core.io.support.SpringFactoriesLoader;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;import java.io.File;
import java.util.List;@RestController
public class BootFileController {@ResourceCustomPluginConfig pluginConfig;/*** 测试Spring Boot SPI(Spring Factories Loader)机制的接口* 该方法通过GET请求访问/testBootSpi,用于展示如何通过SPI加载和使用插件** @return 返回一个简单的"OK"字符串,表示操作成功* @throws Exception 如果文件保存过程中发生错误,可能会抛出异常*/@GetMapping("/testBootSpi")public String testBootSpi() throws Exception {// 加载所有的BootFileStorePlugin实现类List<BootFileStorePlugin> bootFileStorePlugins = SpringFactoriesLoader.loadFactories(BootFileStorePlugin.class, null);// 遍历每个插件并调用其saveFile方法bootFileStorePlugins.forEach(plugin -> plugin.saveFile(new File("/a")));// 返回操作成功信息return "OK";}
在这个Controller中,我们使用SpringFactoriesLoader
加载了SmsPlugin
接口的所有实现类,并遍历这些实现类来执行sendMessage
方法。
3.5 测试效果
启动应用后,访问http://localhost:8080/testBootSpi
接口,控制台会输出如下内容:
可以看到,Spring Boot正确加载并执行了配置文件中指定的插件实现。
小结
Spring Boot通过spring.factories
文件实现了类似于Java SPI机制的功能。通过这种机制,我们可以非常方便地扩展应用的功能,而无需修改原有代码或依赖项。这种方式不仅支持插件化开发,还能确保扩展与核心逻辑的解耦,为应用的灵活性和可扩展性提供了有力支持。
相关文章:
Plugin - 插件开发04_Spring Boot中的SPI机制与Spring Factories实现
文章目录 Pre方案概览使用插件的好处Spring Boot中的SPI机制与Spring Factories实现1. Spring Boot中的SPI机制Spring Factories文件 2. Spring Factories实现原理3. Code3.1 定义一个服务接口3.2 定义 实现类3.3 配置 spring.factories 文件3.4 创建一个Controller来加载插件3…...
GUNS搭建
一、准备工作 源码下载: 链接: https://pan.baidu.com/s/1bJZzAzGJRt-NxtIQ82KlBw 提取码: criq 官方文档 二、导入代码 1、导入后端IDE 导入完成需要,需要修改yml文件中的数据库配置,改成自己的。 2、导入前端IDE 我是用npm安装的yarn npm…...
路径规划之启发式算法之十四:蜘蛛蜂优化算法(Spider Wasp Optimizer, SWO)
蜘蛛蜂优化算法(Spider Wasp Optimizer, SWO)是一种受自然界中蜘蛛蜂行为启发的元启发式智能优化算法。由Mohamed Abdel-Basset等人于2023年提出,算法模拟了雌性蜘蛛蜂的狩猎、筑巢和交配行为,具有独特的更新策略,适用于具有不同探索和开发需求的广泛优化问题。 一、算法背…...
OpenGL ES详解——多个纹理实现混叠显示
目录 一、获取图片纹理数据 二、着色器编写 1. 顶点着色器 2. 片元着色器 三、绑定和绘制纹理 1. 绑定纹理 2. 绘制纹理 四、源码下载 一、获取图片纹理数据 获取图片纹理数据代码如下: //获取图片1纹理数据 mTextureId loadTexture(mContext, R.mipmap.…...
Java多线程与线程池技术详解(八)
《星游记》 “如果只有傻瓜才相信梦想,那么就叫我大傻瓜吧!” 《一人之下》 “想走的路不好走,想做人不好做,都说是身不由己,不是废话么。己不由心,身又岂能由己!” 目录 上一篇博客习题讲解 编…...
2024年12月11日Github流行趋势
项目名称:maigret 项目维护者:soxoj, kustermariocoding, dependabot, fen0s, cyb3rk0tik项目介绍:通过用户名从数千个站点收集个人档案信息的工具。项目star数:12,055项目fork数:870 项目名称:uv 项目维护…...
ThinkPHP 6.0 PHP新手教程
1、系统配置文件 下面系统自带的配置文件列表及其作用: 配置文件名描述app.php应用配置cache.php缓存配置console.php控制台配置cookie.phpCookie配置database.php数据库配置filesystem.php磁盘配置lang.php多语言配置log.php日志配置middleware.php中间件配置rou…...
【Excel学习记录】02-单元格格式设置
1.单元格格式工具美化表格 单元格格式位置 选中单元格,右键→设置单元格格式 合并居中 跨越合并 字体类型、大小、颜色、填充底纹、边框 斜线 软回车:alt enter 格式刷 2.单元格数字格式 格式不影响数值,只是展示形式 日期本质也是数…...
Paimon Tag和Branch创建文件存储过程
结论: 如果data-file被引用则不会被压缩,压缩仅针对未被引用的文件,创建tag时候根据当前快照进行创建 1、实际表和Manifest的内容 查看tag的内容 select * from table$tags;或者直接查看tag ossutil cat oss://test-dataware/warehouse/te…...
HCIA笔记8--DHCP、Telnet协议
1. DHCP介绍 对于主机的网络进行手动配置,在小规模的网络中还是可以运作的,但大规模网络是无力应对的。因此就有了DHCP协议来自动管理主机网络的配置。 DHCP(Dynamic Host Configuration Protocol): 动态主机配置协议,主要需要配置的参数有…...
Tableau数据可视化与仪表盘搭建
1.Tableau介绍 可视化功能 数据赋能 数据赋能就是将我们的数据看板发布到我们的线上去 这里的IP地址是业务部门可以通过账号密码登入的 我们也可以根据需要下载,选中并点击下载即可 下载下来之后,自己就能根据数据进行自定义的分析 也可以下载图片 还有…...
Django结合websocket实现分组的多人聊天
其他地方和上一篇大致相同,上一篇地址点击进入, 改动点1:在setting.py中最后再添加如下配置: # 多人聊天 CHANNEL_LAYERS {"default":{"BACKEND": "channels.layers.InMemoryChannelLayer"} }因此完整的se…...
自动驾驶控制与规划——Project 1: 车辆纵向控制
目录 零、任务介绍一、环境配置1.1 CARLA的配置1.2 Docker Ubuntu 20.04 ROS2 Foxy的配置 二、算法2.1 定速巡航2.2 自适应巡航2.3 离散PID控制 三、代码实现3.1 代码补全3.2仿真验证 零、任务介绍 课程主页 配置Carla仿真器配置carla-ros-bridge补全src\ros-bridge\carla_s…...
基于python实现自动化的验证码识别:探索与实践
基于python实现自动化的验证码识别:探索与实践 一、验证码的类型及特点(一)图像验证码(二)短信验证码(三)语音验证码 二、验证码识别的方法*(一)传统图像处理方法&#x…...
如何保证消息队列的高可用?(RabbitMQ)
RabbitMQ 是基于主从(非分布式)做高可用性的,RabbitMQ 有三种模式:单机模式、普通集群模式、镜像集群模式 1、单机模式:一般没人生产用单机模式 2、普通集群模式: 普通集群模式用于提高系统的吞吐量&…...
es的join是什么数据类型
在 Elasticsearch 中,parent 并不是一个独立的数据类型,而是与 join 数据类型一起使用的一个概念。join 数据类型用于在同一个索引中建立父子文档之间的关系,允许你在一个索引内表示层级结构或关联关系。通过 join 字段,你可以定义不同类型的文档(如父文档和子文档),并指…...
爬虫基础之代理的基本原理
在做爬虫的过程中经常会遇到一种情况,就是爬虫最初是正常运行、正常抓取数据的,一切看起来都是那么美好,然而一杯茶的工夫就出现了错误,例如 403 Forbidden,这时打开网页一看,可能会看到“您的IP访问频率太…...
golang实现简单的reids服务2
golang实现redis兼容的redis服务实现redis兼容的redis服务思路 golang实现redis兼容的redis服务 之前做的redis服务是通过tcp封装的自定义协议 原版项目地址:https://github.com/dengjiayue/my-redis.git 那么能不能实现一个redis兼容的redis服务,这样一般的redis包也可以调…...
CSS常用的尺寸单位
像素px 以屏幕上的一个点为单位,比较稳定和精确用的最多 em 以字体大小为参考,(是自身字体大小的倍数)当自身的字体大小改变时,em也会随着改变em * font-size px rem 以根元素 < html > 作为参考ÿ…...
计算机网络 备查
OSI 七层模型 七层模型协议各层实现的功能 简要 详细 TCP/IP协议 组成 1.传输层协议 TCP 2.网络层协议 IP 协议数据单元(PDU)和 封装 数据收发过程 数据发送过程 1. 2.终端用户生成数据 3.数据被分段,并加上TCP头 4.网络层添加IP地址信息…...
JavaEE 【知识改变命运】05 多线程(4)
文章目录 单例模式什么是单例模式饿汉模式懒汉模式多线程- 懒汉模式分析多线程问题第一种添加sychronized的方式第二种添加sychronized的方式改进第二种添加sychronized的方式(DCL检查锁) 阻塞队列什么是阻塞队列什么是消费生产者模型标准库中的阻塞队列…...
迭代器和生成器
一、迭代器(Iterator) 1. 什么是迭代器? 迭代器是一个可以在某一集合(如列表、元组等)中逐个访问元素的对象。它提供了一个方法,可以记住遍历的位置,每次取出一个元素,直到所有元素…...
小发现,如何高级的顺序输出,逆序输出整数的每一位(栈,队列)
当我还是初学者的时候,我经常思考有没有比慢慢求每一位数字然后考虑正序,逆序输出要快的办法...长期琢磨,必有所获! 我刚学数据结构的时候还没意识到栈,队列还能这样用,虽然说有点杀鸡用牛刀的感觉&#x…...
前端换行、空格的多种表现形式
换行 1、<br> 标签 这是最直接的方式,用于在文本中插入一个简单的换行。<br> 标签是一个空元素,意味着它不需要结束标签。 示例: <p>这是第一行。<br>这是第二行。</p>2、CSS white-space 属性 通过CSS的w…...
自己总结:selenium高阶知识
全篇大概10000字(含代码),建议阅读时间30min 一、等待机制 如果有一些内容是通过Ajax加载的内容,那就需要等待内容加载完毕才能进行下一步操作。 为了避免人为操作等待,会遇到的问题, selenium将等待转换…...
无线遥控红外通信
无线遥控红外通信 红外发射装置一般是指红外遥控器由 键盘电路 ,红外编码电路 电源电路 和红外发射 电路组成 一般的红外线波长为940nm左右,外形与普通发光二极管相同 红外遥控为了提高抗干扰性能和降低电源消耗,红外遥控器常用载波的方式传送…...
第一个C++程序--(蓝桥杯备考版)
第一个C程序 基础程序 #include <iostream>//头⽂件 using namespace std;//使⽤std的名字空间 int main()//main函数 {cout << "hello world!" << endl; //输出:在屏幕打印"hello world!" return 0;}main函数 main 函数是…...
Rust包管理和构建工具
Cargo 是 Rust 语言的包管理和构建工具。它提供了一套完整的工具链,用于管理 Rust 项目的依赖关系、编译代码、运行测试和生成文档。Cargo 极大地简化了 Rust 项目的开发和部署过程,使得开发者可以专注于编写代码,而不是处理构建系统的复杂性…...
STM32输入捕获详解
目录 一、引言 二、输入捕获原理 三、寄存器介绍 四、配置步骤 1.开启时钟 2.GPIO 初始化 3.初始化定时器 4.配置输入捕获模式 5.使能捕获和更新中断 6.设置中断分组并编写中断服务函数 7.使能定时器 五、程序示例 六、总结 一、引言 在嵌入式系统开发中࿰…...
利用高德地图API,如何在PHP与vue3中实现地图缩放功能
文章精选推荐 1 JetBrains Ai assistant 编程工具让你的工作效率翻倍 2 Extra Icons:JetBrains IDE的图标增强神器 3 IDEA插件推荐-SequenceDiagram,自动生成时序图 4 BashSupport Pro 这个ides插件主要是用来干嘛的 ? 5 IDEA必装的插件&…...
Selenium WebDriver:自动化网页交互的利器
Selenium WebDriver:自动化网页交互的利器 在当今快速发展的Web开发领域,自动化测试已经成为确保应用程序质量和用户体验的重要手段。Selenium WebDriver,作为Selenium工具包中的核心组件,正是这一领域的佼佼者。本文将详细介绍S…...
uniapp -- 实现页面滚动触底加载数据
效果 首选,是在pages.json配置开启下拉刷新 {"path": "pages/my/document/officialDocument","style": {"navigationStyle":</...
用ChatGPT-o1进行论文内容润色效果怎么样?
目录 1.引导问题发现 2.角色设定 3.整理常问修改 4.提供样例 5.小细节 小编在这篇文章中分享如何充分利用ChatGPT-o1-preview来提升论文润色的技巧。小编将持续跟进最新资源和最新的调研尝试结果,为宝子们补充更多实用的写作技巧。这些技巧将有助于您更有效地利…...
6 C/C++输⼊输出(下)(未完续)
1. OJ(online judge)题⽬输⼊情况汇总 在竞赛的 OJ 题⽬中,⼀般关于输⼊场景总结为下⾯四类: 接下来,我们就结合题⽬,给⼤家分别介绍。 1.1 单组测试⽤例 练习1 B2009 计算 (ab)/c 的值 - 洛谷 | 计算机科…...
题海拾贝:力扣 20、有效的括号
Hello大家好!很高兴我们又见面啦!给生活添点passion,开始今天的编程之路! 我的博客:<但凡.-CSDN博客 我的专栏:《编程之路》、《题海拾贝》、《数据结构与算法之美》 欢迎点赞、关注! 1、题目 2、题解 这…...
视频推拉流EasyDSS无人机直播技术巡查焚烧、烟火情况
焚烧作为一种常见的废弃物处理方式,往往会对环境造成严重污染。因此,减少焚烧、推广绿色能源和循环经济成为重要措施。通过加强森林防灭火队伍能力建设与长效机制建立,各地努力减少因焚烧引发的森林火灾,保护生态环境。 巡察烟火…...
基于Hermite多项式的三维反时间波的生成
原创:daode3056(daode1212) 反时间波,也称为时间反演波,是一种在特定条件下能够实现波的聚焦和传播的技术。反时间波的产生基于时间反演技术,其原理和方法通常有: 1. [时间反演信号处理原理]: 时间反演技术并不是指时间…...
数据结构与算法复习AVL树插入过程
环境 $ cat /proc/version Linux version 6.8.0-45-generic (builddlcy02-amd64-115) (x86_64-linux-gnu-gcc-13 (Ubuntu 13.2.0-23ubuntu4) 13.2.0, GNU ld (GNU Binutils for Ubuntu) 2.42) #45-Ubuntu SMP PREEMPT_DYNAMIC Fri Aug 30 12:02:04 UTC 2024 #include <std…...
MetaGPT源码 (Memory 类)
目录 MetaGPT源码:Memory 类例子 MetaGPT源码:Memory 类 这段代码定义了一个名为 Memory 的类,用于存储和管理消息(Message)对象。Memory 提供了多种操作消息的功能,包括添加单条或批量消息、按角色或内容筛选消息、删除最新消息…...
day1数据结构,关键字,内存空间存储与动态分区,释放
小练习 在堆区空间连续申请5个int类型大小空间,用来存放从终端输入的5个学生成绩,然后显示5个学生成绩,再将学生成绩升序排序,排序后,再次显示学生成绩。显示和排序分别用函数完成(两种排序方法࿰…...
C# 用封装dll 调用c++ dll 使用winapi
这里用c net 封装winapi函数 pch.h // pch.h: 这是预编译标头文件。 // 下方列出的文件仅编译一次,提高了将来生成的生成性能。 // 这还将影响 IntelliSense 性能,包括代码完成和许多代码浏览功能。 // 但是,如果此处列出的文件中的任何一个…...
vue2 如何设置i18n的默认语言为当前浏览器的语言
做到i18n这里设置默认语言的时候遇到了一些小问题,所以做个记录: 原始代码lang/index // index.js import Vue from vue import VueI18n from vue-i18n import Cookies from js-cookie // import elementEnLocale from element-ui/lib/locale/lang/en // element-…...
QT数据库SQLite:QsqlTableModel使用总结
数据库连接、数据模型与界面组件所涉及的类之间的关系如下所示: 数据库类 QSqlDatabase 类用于建立与数据库的连接,QSqlDatabase 对象就表示这种连接。QSqlDatabase 类的功能主要分为三大部分: 1、创建数据库连接,即创建 QSqlDat…...
服务器零配件
阵列卡 H3C电池 RAId 卡 内存条位置 HBA卡 MOC卡...
MySQL 学习 之 批量插入数据性能问题
文章目录 现象优化 现象 在使用 kettle 同步大数据的数据到我们的 MySQL 数据库中时发现,数据量大时插入效率很慢,大约在 2000/s 优化 在 MySQL 驱动连接中添加 rewriteBatchedStatementstrue 参数,减少 网络 IO DB IO 耗时 默认关闭指定…...
会话管理和身份验证和授权
Cookie、Session、Token Cookie 简介:[Cookie]是一种小型文本文件,由服务器发送到用户的浏览器并保存在用户的计算机上。其主要作用是识别用户身份、跟踪用户活动、保存用户设置等。Cookie通常由名称、值、域名、路径、过期时间等字段组成,并…...
RK3588 rknpu2/rkllm/rockit/mpp/rga 等源码验证
RK3588 简介 本项目基于rk3588硬件平台,将嵌入式、流媒体、AI等相关的技术验证源码地址 源码说明 buildroot 为buildroot使用方法dk_doc 为rk的文档mpp 在mpp例子上增加推流rga 为rk3588的硬件加速模块,可快速处理视频,提供的API接口与op…...
【CSS in Depth 2 精译_075】12.2 Web 字体简介 + 12.3 谷歌字体的用法
当前内容所在位置(可进入专栏查看其他译好的章节内容) 第四部分 视觉增强技术 ✔️【第 12 章 CSS 排版与间距】 ✔️ 12.1 间距设置 12.1.1 使用 em 还是 px12.1.2 对行高的深入思考12.1.3 行内元素的间距设置 12.2 Web 字体 ✔️12.3 谷歌字体 ✔️12.…...
【数字花园】个人知识库网站搭建:①netlify免费搭建数字花园
目录 [[数字花园]]的构建原理包括三个步骤:五个部署方案教程相关教程使用的平台 步骤信息管理 这里记录的自己搭建数字花园(在线个人知识库)的经历,首先尝试的是网上普遍使用的方法,也就是本篇文章介绍的。 后面会继续…...
访问者模式的理解和实践
在软件开发过程中,设计模式为我们提供了解决常见问题的最佳实践。访问者模式(Visitor Pattern)是行为设计模式之一,它将数据操作与数据结构分离,使得在不修改数据结构的前提下,能够定义作用于这些元素的新的…...