OpenHarmony-3.HDF Display子系统(6)
- Display 子系统
1.Display驱动模型介绍
当前操作系统和 SOC 种类繁多,各厂商的显示屏器件也各有不同,随之针对器件的驱动代码也不尽相同,往往是某一款器件驱动,只适用于某单一内核系统或 SOC,如果要迁移到其他内核或者 SOC,可能会有不小的移植工作量。而且,不同驱动 IC 的驱动代码差异较大,产品更换驱动 IC,则又需要重新开发对应的器件驱动,造成重复工作。因此基于 HDF 驱动框架,编写一套较通用的 Display 器件驱动模型,尽可能降低驱动开发者的开发或移植工作量,简化器件驱动开发,提升开发效率。
Display驱动编程,通过对显示器上电、初始化显示器驱动IC(Integrated Circuit)内部寄存器等操作,使其可以正常工作。基于HDF(Hardware Driver Foundation)驱动框架构建的Display驱动模型作用如下:
-
为LCD器件驱动开发提供了基础驱动框架,提升驱动开发效率。
-
便于开发的器件驱动实现跨OS、跨芯片平台迁移。
基于HDF驱动框架的Display驱动模型如下所示:
模型各层设计说明:
-
Display 驱动模型基于 HDF 驱动框架、Platform 接口及 OSAL 接口开发,可以做到不区分OS(LiteOS、Linux)和芯片平台(Hi35xx、Hi38xx、V3S等),为 LCD 器件提供统一的驱动模型。
-
当前 HDF Display 驱动模型主要分为四层:标准架构适配层(DRM Panel Adapter Driver)、显示公共驱动层(Display Common Driver)、芯片平台适配层(SoC Adapter Driver)、器件驱动层(Display Panel Driver)。
1.1.标准架构适配层
- drivers_hdf_core\framework\model\display\driver\hdf_drm_panel.c
本层主要完成对接标准的显示驱动架构,如 DRM(Direct Rending Manager)或 FB(Framebuffer),以 DRM 为例,将 panel 侧驱动接口对接到标准框架中,保证在 DRM 框架中实现对 Panel 驱动的操作接口,当前注册的接口如下:
static struct drm_panel_funcs g_hdfDrmPanelFuncs = {.get_modes = HdfDrmPanelGetModes,.enable = HdfDrmPanelEnable,.disable = HdfDrmPanelDisable,.prepare = HdfDrmPanelPrepare,.unprepare = HdfDrmPanelUnprepare,
};
1.2.显示公共驱动层
- drivers_hdf_core\framework\model\display\driver\hdf_disp.c
此部分属于整个驱动模型的中枢,所有的屏端接口注册、Panel 信息管理、屏幕状态控制、用户态 HDI 接口命令处理、以及通用的基础显示特性,目前都是通过这部分实现。
在本层通过结构体 DispManager 管理所有的显示信息,其成员 PanelManager 用于记录与显示屏相关的接口及参数信息。同时接收并处理 HDI 层直接对 panel 操作相关的指令(主要用于 L0-L1 等轻量级系统),如 Panel 器件信息的获取、休眠唤醒、背光设置等指令。此外,本层还负责实现一些基础显示特性的业务框架,如 ESD 检查机制,力求将显示相关的共有逻辑集中到本层实现,以简化 Panel 器件驱动的实现,避免 panel 驱动中相同功能的重复实现,便于统一管理和维护。
1.3.芯片平台适配层
借助此 SoC 适配层,实现 Display 器件驱动和 SoC 侧硬件资源的解耦,主要完成芯片平台强相关的参数配置,如 mipi 速率计算及设置、管脚复用配置,以及其他和 SoC 强相关的差异化配置及初始化等。
1.4.器件驱动层
- drivers_hdf_core\framework\model\display\driver\panel
器件驱动层主要实现和器件自身强相关的驱动适配接口,例如发送初始化序列、休眠唤醒流程、背光设置、ESD 检测等,同时完成 panel 信息的解析,并将 panel 向上注册到公共驱动层进行管理。
1.5.驱动加载及运行
HDF 的驱动的加载方式,框架通过解析设备描述的 hcs 配置文件,获取到各设备的配置信息,根据 moduleName 来匹配对应设备的驱动文件入口,按照配置的加载优先级,依次加载驱动。
加载流程分为 9 步,分别说明如下:
- HDF Device Manager 解析设备描述;
- HDF 优先加载器件驱动层,构建 Panel 设备;
- 将 panel信息及操作接口注册到公共驱动层;
- HDF 其次加载芯片平台适配层,进行 SoC 相关硬件资源初始化;
- HDF再次加载公共驱动层,对共有特性进行初始化;
- HDF 最后加载标准架构适配层;
- 从公共驱动层中获取到 PanelManager,;
- 将对应panel 注册到 DRM 框架中;
- 在系统运行起来后,DRM 会调用 panel ops 进行显示屏控制。
备注:
对于LiteOS 这种轻量内核的系统,不会像 Linux 内核那样提供标准的显示框架,驱动模型也无法与其对接,因而上层图形系统可以通过 HDI 接口,来直接操控显示屏。
2.LCD接口
LCD接口通常可分为MIPI DSI接口、TTL接口和LVDS接口,常用的是MIPI DSI接口和TTL接口。
-
MIPI DSI接口
MIPI DSI接口是MIPI(Mobile Industry Processor Interface)联盟定义的显示接口,主要用于移动终端显示屏接口,接口数据传输遵循MIPI协议,MIPI DSI接口为数据接口,传输图像数据,通常情况下MIPI DSI接口的控制信息以MIPI包形式通过MIPI DSI接口发送到对端IC,不需要额外的外设接口。
-
TTL接口
TTL接口是并行方式传输数据的接口,有数据信号、时钟信号和控制信号(行同步、帧同步、数据有效信号等),在控制信号控制下完成数据传输。通常TTL接口的LCD,内部寄存器读写需要额外的外设接口,比如SPI接口、I2C接口等。
3.HDF LCD 驱动
3.1.LCD 内核配置
- Kconfig\Makefile
drivers_hdf_core\adapter\khdf\linux\model\display\Kconfig:
config DRIVERS_HDF_DISPbool "Enable HDF Display driver"default ndepends on DRIVERS_HDFhelpAnswer Y to choice HDF Display driver.
config DRIVERS_HDF_LCDKITbool "Enable HDF Lcdkit driver"default ndepends on DRIVERS_HDF_DISPhelpAnswer Y to enable HDF Lcdkit driver.
config DRIVERS_HDF_LCD_ICN9700bool "Enable HDF Icn9700 driver"default ndepends on DRIVERS_HDF_DISPhelpAnswer Y to enable HDF Icn9700 driver.drivers_hdf_core\adapter\khdf\linux\model\display\Makefile:
KHDF_DISPLAY_BASE_ROOT_DIR = ../../../../../../..
DISPLAY_ROOT_DIR = ../../../../../framework/model/display/driverifeq ($(CONFIG_DRIVERS_HDF_DISP), y)
obj-y += \$(DISPLAY_ROOT_DIR)/hdf_disp.o \$(DISPLAY_ROOT_DIR)/backlight/hdf_bl.o#if (LINUX_VERSION_CODE >= KERNEL_VERSION(5, 10, 0))
ifeq ($(CONFIG_ARCH_ROCKCHIP), y)
obj-y += $(DISPLAY_ROOT_DIR)/hdf_drm_panel.o
endif
#endifobj-$(CONFIG_DRIVERS_HDF_PLATFORM_PWM) += \$(DISPLAY_ROOT_DIR)/backlight/pwm_bl.oobj-$(CONFIG_ARCH_SPRD) += \$(DISPLAY_ROOT_DIR)/panel/ili9881c_boe.oobj-$(CONFIG_ARCH_HI3516DV300) += \$(DISPLAY_ROOT_DIR)/adapter_soc/hi35xx_disp.oobj-$(CONFIG_DRIVERS_HDF_LCDKIT) += \$(DISPLAY_ROOT_DIR)/lcdkit/lite_lcdkit.o \$(DISPLAY_ROOT_DIR)/lcdkit/lcdkit_parse_config.o
...ccflags-y += -lm -lc -lgcc \-I$(srctree)/drivers/hdf/framework/model/display/driver \-I$(srctree)/drivers/hdf/framework/model/display/driver/adapter_soc \...
ccflags-y +=-I$(srctree)/bounds_checking_function/includeendif
3.2.LCD 设备配置
- 配置设备描述信息(vendor/hihope/rk3568/hdf_config/khdf/device_info/device_info.hcs)
驱动注册到 HDF 框架所需要的设备驱动描述信息,如驱动是否加载以及加载次序等。device_info.hcs 中的信息主要提供给 HDF 框架使用,包含了 Input 模型各层驱动注册到 HDF 框架所必需的信息,开发者无特殊场景需求无需改动。各驱动层私有配置信息通过**“deviceMatchAttr”字段与 input_config.hcs 中的“match_attr”**相关内容进行匹配。
237 display :: host {
238 hostName = "display_host";
239 device_hdf_drm_panel :: device {
240 device0 :: deviceNode {
241 policy = 0;
242 priority = 197;
243 preload = 0;
244 moduleName = "HDF_DRMPANEL";
245 }
246 }/* Display平台驱动设备描述 */
247 device_hdf_disp :: device {
248 device0 :: deviceNode {
249 policy = 2;
250 priority = 196;
251 permission = 0660;
252 moduleName = "HDF_DISP";
253 serviceName = "hdf_disp";
254 }
255 }/* SOC适配层驱动设备描述 */
256 device_hi35xx_disp :: device {
257 device0 :: deviceNode {
258 policy = 0;
259 priority = 195;
260 moduleName = "HI351XX_DISP";
261 }
262 }/* LCD器件驱动设备描述 */
263 device_lcd :: device {
271 ...
283 device3 :: deviceNode {
284 policy = 0;
285 priority = 100;
286 preload = 0;
287 moduleName = "LCD_ILI9881_ST_5P5";
288 }
289 }
290 device_pwm_bl :: device {
291 device0 :: deviceNode {
292 policy = 0;
293 priority = 95;
294 preload = 0;
295 moduleName = "PWM_BL";
296 deviceMatchAttr = "pwm_bl_dev";
297 }
298 }
299 device_backlight :: device {
300 device0 :: deviceNode {
301 policy = 2;
302 priority = 90;
303 preload = 0;
304 permission = 0660;
305 moduleName = "HDF_BL";
306 serviceName = "hdf_bl";
307 }
308 }
309 }
- .板级配置及器件私有配置(vendor/hihope/rk3568/hdf_config/khdf/lcd/lcd_config.hcs )
1 root {2 backlightConfig {3 pwmBacklightConfig {4 match_attr = "pwm_bl_dev";5 pwmDevNum = 1;6 pwmMaxPeriod = 25000;7 backlightDevName = "hdf_pwm";8 minBrightness = 0;9 defBrightness = 127;10 maxBrightness = 255;11 }12 }13 }
3.3.驱动实现
- 对于LCD类型的设备,公共驱动框架已实现。
drivers_hdf_core\framework\model\display\driver\hdf_drm_panel.cdrivers_hdf_core\framework\model\display\driver\hdf_disp.cdrivers_hdf_core\framework\model\display\driver\backlight\hdf_bl.cdrivers_hdf_core\framework\model\display\driver\backlight\pwm_bl.c
-
适配 ili9881_st_5p5 需要完成器件层驱动初始化、释放资源、注册驱动至HDF框架。
drivers_hdf_core\framework\model\display\driver\panel\ili9881_st_5p5.c
3.3.1.ssp_st7789器件驱动
1).注册到HDF框架
struct HdfDriverEntry PanelDevEntry = {.moduleVersion = 1,.moduleName = "LCD_ILI9881_ST_5P5",.Init = PanelEntryInit,
};HDF_INIT(PanelDevEntry);
2).器件层驱动初始化
struct panel_ili9881_dev {bool power_invert;struct PanelData panel; //重要数据结构struct mipi_dsi_device *dsiDev;struct regulator *supply;struct gpio_desc *enable_gpio;struct gpio_desc *reset_gpio;struct gpio_desc *hpd_gpio;struct panel_hw_delay hw_delay;
};struct PanelData {struct HdfDeviceObject *object;int32_t (*init)(struct PanelData *panel);int32_t (*on)(struct PanelData *panel);int32_t (*off)(struct PanelData *panel);int32_t (*prepare)(struct PanelData *panel);int32_t (*unprepare)(struct PanelData *panel);struct PanelInfo *info;enum PowerStatus powerStatus;struct PanelEsd *esd;struct BacklightDev *blDev;void *priv;
};static int32_t PanelEntryInit(struct HdfDeviceObject *object)
{struct device_node *panelNode = NULL;struct panel_ili9881_dev *panel_dev = NULL;panel_dev = (struct panel_ili9881_dev *)OsalMemCalloc(sizeof(struct panel_ili9881_dev));g_panel_dev = panel_dev;panelNode = of_find_compatible_node(NULL, NULL, "simple-panel-dsi");panel_dev->dsiDev = of_find_mipi_dsi_device_by_node(panelNode);panel_dev->supply = devm_regulator_get(&panel_dev->dsiDev->dev, "power");if (PanelRequestGpio(panel_dev) != HDF_SUCCESS) {goto FAIL;}PanelResInit(panel_dev);panel_dev->panel.object = object;object->priv = panel_dev;if (RegisterPanel(&panel_dev->panel) != HDF_SUCCESS) {...}return HDF_SUCCESS;
}
- 申请struct panel_ili9881_dev结构体,根据"simple-panel-dsi"查找内核设备数节点panelNode;
- 获取plane 供电regulator和gpio 引脚;
- 调用 PanelResInit进行初始化panel结构体;
Panel 操作函数有:panel_dev->panel.init = PanelInit;panel_dev->panel.on = PanelOn;panel_dev->panel.off = PanelOff;panel_dev->panel.prepare = PanelPrepare;panel_dev->panel.unprepare = PanelUnprepare;
- 调用RegisterPanel 注册panel到g_panelManager。
3.3.2. HDF LCD 公共驱动层
3.3.2.1.HDF_DISP
- 结构体
drivers_hdf_core\framework\model\display\driver\hdf_disp.h
struct DispManager {struct PanelManager *panelManager;struct OsalMutex dispMutex;HdfWorkQueue dispWorkQueue;bool initialzed;struct DispEsd *esd;
};struct PanelManager {struct PanelData *panel[PANEL_MAX];uint32_t panelNum;
};
- 初始化及注册驱动至HDF框架
drivers_hdf_core\framework\model\display\driver\hdf_disp.c
struct HdfDriverEntry g_dispDevEntry = {.moduleVersion = 1,.moduleName = "HDF_DISP",.Init = HdfDispEntryInit,.Bind = HdfDispBind,
};HDF_INIT(g_dispDevEntry);
- .Init = HdfDispEntryInit,
DispManagerInit初始化struct DispManager ,将 panelManager 注册到g_dispManager->panelManager,然后创建一个工作队列 HdfWorkQueueInit(&g_dispManager->dispWorkQueue, “dispWQ”); - .Bind = HdfDispBind,
绑定服务HdfDispDispatch,调用DispCmdProcess通过g_dispCmdHandle进行显示命令处理。
static int HdfDispBind(struct HdfDeviceObject *dev)
{if (dev == NULL) {return HDF_FAILURE;}static struct IDeviceIoService dispService = {.object.objectId = 1,.Dispatch = HdfDispDispatch,};dev->service = &dispService;return HDF_SUCCESS;
}DispCmdHandle g_dispCmdHandle[] = {GetPowerStatus,GetInfo,SetPowerStatus,SetBacklight,GetBacklight,
};
3.3.2.2.HDF_DRMPANEL
- 结构体
drivers_hdf_core\framework\model\display\driver\hdf_drm_panel.h
struct HdfDrmPanel {struct drm_panel panel;struct DispManager *manager;struct drm_display_mode mode;struct mipi_dsi_device *dsiDev;uint32_t index;
};
- 初始化及注册驱动至HDF框架
drivers_hdf_core\framework\model\display\driver\hdf_drm_panel.c
struct HdfDriverEntry g_hdfDrmPanelEntry = {.moduleVersion = 1,.moduleName = "HDF_DRMPANEL",.Init = HdfDrmPanelEntryInit,
};
refer to
- https://gitee.com/openharmony/docs/blob/master/zh-cn/device-dev/driver/driver-peripherals-lcd-des.md
- https://blog.csdn.net/HarmonyOS_666/article/details/140824498?spm=1001.2101.3001.6650.7&utm_medium=distribute.pc_relevant.none-task-blog-2%7Edefault%7EBlogCommendFromBaidu%7ERate-7-140824498-blog-143099522.235%5Ev43%5Epc_blog_bottom_relevance_base2&depth_1-utm_source=distribute.pc_relevant.none-task-blog-2%7Edefault%7EBlogCommendFromBaidu%7ERate-7-140824498-blog-143099522.235%5Ev43%5Epc_blog_bottom_relevance_base2&utm_relevant_index=8
相关文章:

OpenHarmony-3.HDF Display子系统(6)
Display 子系统 1.Display驱动模型介绍 当前操作系统和 SOC 种类繁多,各厂商的显示屏器件也各有不同,随之针对器件的驱动代码也不尽相同,往往是某一款器件驱动,只适用于某单一内核系统或 SOC,如果要迁移到其他内核或者…...

Nginx中Server块配置的详细解析
Nginx中Server块配置的详细解析 一、Server块简介 在Nginx配置文件中,server块是非常关键的部分。它用于定义虚拟主机,一个server块就代表一个虚拟主机。这使得我们可以在一台Nginx服务器上通过不同的配置来处理多个域名或者基于不同端口的服务请求。 …...

php学习资料分享
php学习资料分享:夸克网盘分享...

EE308FZ_Sixth Assignment_Beta Sprint_Sprint Essay 3
Assignment 6Beta SprintCourseEE308FZ[A] — Software EngineeringClass Link2401_MU_SE_FZURequirementsTeamwork—Beta SprintTeam NameFZUGOObjectiveSprint Essay 3_Day5-Day6 (12.15-12.16)Other Reference1. WeChat Mini Program Design Guide 2. Javascript Style Guid…...

Eureka学习笔记-服务端
Eureka学习笔记 服务端 模块设计 Resources :这部分对外暴露了一系列的 Restful 接口。Eureka Client 的注册、心跳、获取服务列表等操作都需要调用这些接口。另外,其他的 Server 在同步 Registry 时也需要调用这些接口。Controller :这里提…...

无限次使用 cursor pro
github地址 cursor-vip 使用方式 在 MacOS/Linux 中,请打开终端; 在 Windows 中,请打开 Git Bash。 然后执行以下命令来安装: 部分电脑可能会误报毒,需要关闭杀毒软件/电脑管家/安全防护再进行 方式1:通过…...

网站运维之整站同步
网站运维之整站同步 1、使用rsync安装rsync工具子服务器生成密钥子服务器发送公钥到服务端(需要root允许ssh登录)服务端添加密钥子服务器尝试免密登录子服务器添加任务计划 2、开启root用户远程ssh3、ubuntu开启root用户 1、使用rsync 很多时候由于访问…...

【机器人】Graspness 端到端 抓取点估计 | 论文解读
在复杂场景中实现抓取检测,Graspness是一种端到端的方法; 输入点云数据,输出抓取角度、抓取深度、夹具宽度等信息。 开源地址:GitHub - rhett-chen/graspness_implementation: My implementation of Graspnet Graspness. 论文地…...

力扣2300.咒语和药水的成功对数(二分法)
根据 灵茶山艾府 题解所写 题目描述: 给你两个正整数数组 spells 和 potions ,长度分别为 n 和 m ,其中 spells[i] 表示第 i 个咒语的能量强度,potions[j] 表示第 j 瓶药水的能量强度。 同时给你一个整数 success 。一个咒语和药…...

WEB开发: 全栈工程师起步 - Python Flask +SQLite的管理系统实现
一、前言 罗马不是一天建成的。 每个全栈工程师都是从HELLO WORLD 起步的。 之前我们分别用NODE.JS 、ASP.NET Core 这两个框架实现过基于WebServer的全栈工程师入门教程。 今天我们用更简单的来实现: Python。 我们将用Python来实现一个学生管理应用࿰…...

云原生周刊:Kubernetes v1.32 正式发布
开源项目推荐 Helmper Helmper 简化了将 Helm Charts导入OCI(开放容器倡议)注册表的过程,并支持可选的漏洞修复功能。它确保您的 Helm Charts不仅安全存储,还能及时应用最新的安全修复。该工具完全兼容 OCI 标准,能够…...

京准电钟:电厂自控NTP时间同步服务器技术方案
京准电钟:电厂自控NTP时间同步服务器技术方案 京准电钟:电厂自控NTP时间同步服务器技术方案 随着计算机和网络通信技术的飞速发展,火电厂热工自动化系统数字化、网络化的时代已经到来。一方面它为控制和信息系统之间的数据交换、分析和应用…...

深入探索Flink的复杂事件处理CEP
深入探索Flink的复杂事件处理CEP 引言 在当今大数据时代,实时数据处理变得愈发关键。Apache Flink作为一款强大的流处理框架,其复杂事件处理(CEP)组件为我们从海量实时数据中提取有价值信息提供了有力支持。本文将详细介绍Flink…...

clickhouse-数据库引擎
1、数据库引擎和表引擎 数据库引擎默认是Ordinary,在这种数据库下面的表可以是任意类型引擎。 生产环境中常用的表引擎是MergeTree系列,也是官方主推的引擎。 MergeTree是基础引擎,有主键索引、数据分区、数据副本、数据采样、删除和修改等功…...

力扣hot100——哈希
1. 两数之和 class Solution { public:vector<int> twoSum(vector<int>& nums, int target) {vector<int> ans;map<int, int> mp;for (int i 0; i < nums.size(); i) {if (mp.count(target - nums[i])) {ans.push_back(mp[target - nums[i]])…...

少样本学习之CAML算法
上下文感知元学习(Context-Aware Meta-Learning, CAML) 概述 在机器学习和深度学习领域,元学习(Meta-Learning)旨在通过学习如何学习,使模型能够在面对新任务时快速适应。传统的元学习方法通常需要在特定…...

C# 中的闭包
文章目录 前言一、闭包的基本概念二、匿名函数中的闭包1、定义和使用匿名函数2、匿名函数捕获外部变量3、闭包的生命周期 三、Lambda 表达式中的闭包1、定义和使用 Lambda 表达式2、Lambda 表达式捕获外部变量3、闭包的作用域 四、闭包的应用场景1、事件处理2、异步编程3、迭代…...

网络编程 03:端口的定义、分类,端口映射,通过 Java 实现了 IP 和端口的信息获取
一、概述 记录时间 [2024-12-19] 前置文章: 网络编程 01:计算机网络概述,网络的作用,网络通信的要素,以及网络通信协议与分层模型 网络编程 02:IP 地址,IP 地址的作用、分类,通过 …...

制作项目之前的分析
对网页的分析可以从多个角度入手,具体包括内容分析、技术分析、用户体验分析。 以下是对网页分析的详细步骤,帮助你从不同维度评估一个网页的效果与质量: 1. 内容分析 内容是网页最核心的部分,确保其符合用户需求是网页设计的首…...
LeetCode 1925 统计平方和三元组的数目
探索平方和三元组:从问题到 Java 代码实现 在数学与编程的交叉领域,常常会遇到一些有趣且富有挑战性的问题。今天,就让我们深入探讨一下 “平方和三元组” 这个有趣的话题,并使用 Java 语言来实现计算满足特定条件的平方和三元组…...

java开发入门学习三-二进制与其他进制
常见的进制 常用的进制有二进制,八进制,十进制,十六进制。而我们最熟悉的是十进制,他们分别是怎么表达的呢? 定义不同的进制,写法不同 二进制(Binary): 使用前缀 0b 或…...

C/S软件授权注册系统(Winform+WebApi+.NET8+EFCore版)
适用软件:C/S系统、Winform桌面应用软件。 运行平台:Windows .NETCore,.NET8 开发工具:Visual Studio 2022,C#语言 数据库:Microsoft SQLServer 2012,Oracle 21c,MySQL8…...

Linux —— 管理进程
一、查看进程 运行态(Running) 定义:处于运行态的进程正在 CPU 上执行指令。在单 CPU 系统中,同一时刻只有一个进程处于运行态;在多 CPU 或多核系统中,可能有多个进程同时处于运行态。示例: 当…...

Diffusino Policy学习note
Diffusion Policy—基于扩散模型的机器人动作生成策略 - 知乎 建议看看,感觉普通实验室复现不了这种工作。复现了也没有太大扩展的意义。 Diffusion Policy 是监督学习吗 Diffusion Policy 通常被视为一种基于监督学习的方法,但它的实际训练过程可能结…...

【Python】*args和**kwargs
【Python】*args和**kwargs 一、*args: 接收不定数量的位置参数示例1:简单的加法计算器示例2:转发参数给另一个函数 二、**kwargs: 接收不定数量的关键字参数示例3:创建用户配置文件示例4:合并多个字典 三、组合使用*args和**kwar…...

使用正则表达式提取PDF文件页数的实现方案
文章目录 背景介绍实现原理代码实现1. 基础函数结构2. 页数提取逻辑3. 使用示例 正则表达式解析优点与局限性优点局限性 错误处理建议性能优化建议最佳实践建议总结参考资源 背景介绍 在Web应用开发中,我们经常需要获取上传PDF文件的页数信息。虽然可以使用pdf.js等第三方库,但…...

Android实现RecyclerView边缘渐变效果
Android实现RecyclerView边缘渐变效果 1.前言: 是指在RecyclerView中实现淡入淡出效果的边缘效果。通过这种效果,可以使RecyclerView的边缘在滚动时逐渐淡出或淡入,以提升用户体验。 2.Recyclerview属性: 2.1、requiresFading…...

springboot443旅游管理系统(论文+源码)_kaic
摘 要 如今社会上各行各业,都喜欢用自己行业的专属软件工作,互联网发展到这个时候,人们已经发现离不开了互联网。新技术的产生,往往能解决一些老技术的弊端问题。因为传统旅游管理系统信息管理难度大,容错率低&#…...

利用git上传项目到GitHub
GitHub是基于git实现的代码托管。git是目前最好用的版本控制系统了,非常受欢迎,比之svn更好。 GitHub可以免费使用,并且快速稳定。 利用GitHub,你可以将项目存档,与其他人分享交流,并让其他开发者帮助你一…...

Rust之抽空学习系列(四)—— 编程通用概念(下)
Rust之抽空学习系列(四)—— 编程通用概念(下) 1、函数 函数用来对功能逻辑进行封装,能够增强复用、提高代码的可读 以下是函数的主要组成部分: 名称参数返回类型函数体 1.1、函数名称 在Rust中&…...