嵌入式系统 第七讲 ARM-Linux内核

• 7.1 ARM-Linux内核简介

内核:是一个操作系统的核心。是基于硬件的第一层软件扩充, 提供操作系统的最基本的功能,是操作系统工作的基础,它负责管理系统的进程、内存、设备驱动程序、文件和网络系统, 决定着系统的性能和稳定性。现代操作系统设计中,为减少系 统本身的开销,往往将一些与硬件紧密相关的(如中断处理程 序、设备驱动程序等)、基本的、公共的、运行频率较高的模 块(如时钟管理进程调度等)以及关键性数据结构独立开来, 使之常驻内存,并对他们进行保护。通常把这一部分称之为操 作系统的内核。

Linux内核:Linux内核的主要模块(或组件)分以下几个部分: 存储管理、CPU和进程管理、文件系统、设备管理和驱动、网 络通信,以及系统的初始化(引导)、系统调用等。

ARM-Linux内核:基于ARM处理器的Linux内核。

• 7.1.1 ARM-Linux内核和普通Linux内核的区别(背)

– 相对于ARM Linux,我们说的普通Linux指的是x86 Linux,它们 都是Linux系统,但是由于ARM和x86是不同的CPU架构,它们的 指令集不同,所以软件编译环境不同软件代码一般不能互用, 一般需要进行兼容性移植

x86是经典的CISC指令集指令集复杂功能多串行执行,但 是也意味着执行效率低下,但性价比突出,所以称为民用终端 的主流处理器内置指令集。Intel和AMD的家用处理器都是x86指 令集。以x86为代表的CISC,理论并发线程1-2条。

ARM是Advanced RISC Machine 的缩写。它的指令集比RISC还要 精简。通常使用ARM架构处理器的机型,多为嵌入式或者便携 机。主频通常不高,现在高通公司的ARM架构处理器有1.0GHz 的,已经算相当高了。另外,ARM 7沿用冯·诺依曼结构;而从 ARM 9以后,就都采用了哈佛结构。ARM的并发线程,理论上 有4条左右处理效率较X86高不少

• 7.1.2 ARM-Linux的版本控制

– Linux的版本号

        • 主版本号:序号的第1位

        • 次版本号:序号的第2位

        • 修订号:序号的第3位

        • 稳定版:序号的第2位(次版本号)为偶数

        • 测试版:序号的第2位(次版本号)为奇数

– 查看Linux系统的版本号(Ubuntu)

• linux@linux-pc:~$ cat /etc/issue
– Ubuntu 18.04.4 LTS \n • linux@linux-pc:~$ cat /etc/os-release
– NAME="Ubuntu"
– VERSION="18.04.4 LTS (Bionic Beaver)"
– ID=ubuntu
– ID_LIKE=debian
– PRETTY_NAME="Ubuntu 18.04.4 LTS"
– VERSION_ID="18.04"
– HOME_URL="https://www.ubuntu.com/"
– SUPPORT_URL="https://help.ubuntu.com/"
– BUG_REPORT_URL="https://bugs.launchpad.net/ubuntu/"
– PRIVACY_POLICY_URL="https://www.ubuntu.com/legal/terms-and-policies/privacy-policy"
– VERSION_CODENAME=bionic
– UBUNTU_CODENA

 – 查看Linux内核的版本号(Ubuntu)

• linux@linux-pc:~$ cat /proc/version
– Linux version 5.4.0-139-generic (buildd@lcy02-amd64-036) (gcc version 7.5.0 (Ubuntu 7.5.0-3ubuntu1~18.04)) #156~18.04.1-Ubuntu SMP Wed Jan 25 15:56:22 UTC 2023• linux@linux-pc:~$ uname-a
– Linux linux-pc 5.4.0-139-generic #156~18.04.1-Ubuntu SMP Wed Jan 25 15:56:22 UTC 2023 x86_64 x86_64 x86_64 GNU/Linux• linux@linux-pc:~$ uname-r
– 5.4.0-139-gen

 • 7.1.3 ARM-Linux的代码结构

– 位于Ubuntu的/home/linux/workdir/fs3399/system/kernel/目录下

 1.android:Android系统的配置

2.arch :包含和硬件体系结构相关的代码,每种平台(处理器)占一个相应的目录,如i386、 arm、arm64、powerpc、mips等

3.block:块设备驱动程序

4.certs:存储认证和签名相关代码

5.crypto:常用加密散列算法(如AES、SHA 等),还有一些压缩CRC 校验算法

6.Documentation:内核各部分的通用解释和注释(文档)

7.drivers :设备驱动程序,每个不同的驱动占用一个子目录,如char、block、net、mtd、i2c 等

8. firmware:固件,包含了让计算机读取和理解从设备发来的信号的代码

9. fs:所支持的各种文件系统,如EXT、FAT、NTFS、JFFS2 等

10. include:头文件与系统相关头文件放置在include/linux子目录下

11. init:内核初始化代码,著名的start_kernel() 就位于init/main.c文件中

12. ipc:进程间通信的代码

13. kernel :内核最核心的部分,包括进程调度、定时器等,而和平台(处理器)相关的一部分 代码放在arch/*/kernel 目录下

14. lib:库文件代码

15. mm:内存管理代码,和平台(处理器)相关的一部分代码放在arch/*/mm 目录下

16. net:网络相关代码,实现各种常见的网络协议

17. samples:一些内核编程的范例 文 件 夹

18. scripts:用于配置内核的脚本文件

19. security:Linux安全模型的代码

20. sound:ALSA、OSS 音频设备的驱动核心代码和常用设备驱动

21. tools:这个文件夹中包含了和内核交互的工具

22. usr:实现用于打包和压缩的cpio等

23. virt:此文件夹包含了虚拟化代码,它允许用户一次运行多个操作系统

23. Kbuild:这是一个设置一些内核设定脚本

24. Kconfig:内核配置选项文件

25. Makefile:这个脚本是编译内核的主要文件,这个文件将编译参数编译所需的文件和必要的 信息传给编译器

26. System.map:该文件可以帮助我们理解内核编译,它记录了所有代码的运行地址

27. vmlinux:是可引导的压缩的Linux内核

28. vmlinux.o:是vmlinux的目标文件

 

 

• 7.2 ARM-Linux内存管理

• 7.2.1 影响内存管理的两个方面

– Linux内核对内存的管理(Linux操作系统的内存管理)

        • 内存管理是操作系统必不可少也是非常重要的一部分,包括:

                ① 地址映射

                ② 内存空间的分配

                ③ 地址访问的限制(即保护机制

                ④ I/O地址的映射(I/O编址与内存编址相同)

– ARM体系结构对内存的管理(MMU)

        • MMU(存储器管理单元)的主要作用有两个方面:

                ① 地址映射

                ② 对地址访问进行保护和限制

        • MMU可以做在CPU芯片中,也可以作为一个协处理器(用协处理器实现)

• 7.2.2 ARM-Linux的存储机制

– 基于x86体系结构的Linux内核的存储空间

• 32位地址形成4GB虚拟地址空间,被分为两部分:

        ① 内核空间(系统空间):位于高端的1GB,属于Linux操作系统

        ② 用户空间:位于低端的3GB,属于应用程序

 – ARM-Linux内核的存储空间 

• 32位地址形成4GB的虚拟地址空间,也被分为两部分,但是内核空间(系统空 间)和用户空间的具体划分,可以因CPU芯片和开发板(实验箱)而有所不同

• 另外,ARM将I/O也放在内存地址空间中

• 7.2.3 虚拟内存

虚拟内存(虚拟存储器):是计算机系统内存管理的一种技术, 它使得应用程序认为它拥有连续的可用的内存(一个连续完整的 地址空间),而实际上,它通常是被分隔成多个物理内存碎片, 还有部分暂时存储在外部磁盘存储器上,在需要时进行数据交换

– Linux虚拟内存的实现需要6种机制的支持

        ① 地址映射机制

        ② 请求页机制

        ③ 内存分配回收机制         

        ④ 缓存和刷新机制

        ⑤ 交换机制

        ⑥ 内存共享机制

• 7.3 ARM-Linux进程管理和调度

• 进程:也称为任务,是一个动态的执行过程,是处于执行期的程序,进程是系统资源 分配的最小单位

• 狭义定义:进程是正在运行的程序的实例。

• 广义定义:进程是一个具有一定独立功能的程序关于某个数据集合的一次运行活动。 它是操作系统动态执行的基本单元,在传统的操作系统中,进程既是基本的分配单元, 也是基本的执行单元

• 进程的概念主要有两点:

        ① 进程是一个实体。每一个进程都有它自己的地址空间,一般情况下,包括文本区域(text region)、数据 区域(data region)和堆栈(stack region)。文本区域存储处理器执行的代码;数据区域存储变量和进程 执行期间使用的动态分配的内存;堆栈区域存储着活动过程调用的指令和本地变量

        ② 进程是一个“执行中的程序”。程序是一个没有生命的实体,只有处理器赋予程序生命时(操作系统 执行之),它才能成为一个活动的实体,我们称其为进程。

• 7.3.1 进程的表示和生命周期

– 进程描述符:用task_struct{}数据结构表示

– 进程的状态(重要):

        ① TASK_RUNNING:可执行状态,进程要么正在执行,要么准备 执行

        ② TASK_INTERRUPTIBLE:可中断睡眠状态

        ③ TASK_UNINTERRUPTIBLE:不可中断的睡眠状态

        ④ TASK_STOPPED:暂停状态,进程停止执行

        ⑤ TASK_TRACED:跟踪状态,进程被追踪

        ⑥ EXIT_ZOMBIE:僵尸状态的进程,表示进程被终止

        ⑦ EXIT_DEAD:进程的最终状态,进程死亡

        ⑧ TASK_DEAD:死亡

        ⑨ TASK_WAKEKILL:唤醒并杀死的进程

        ⑩ TASK_WAKING:唤醒进程

– 进程标识符:PID,Process ID(进程ID)

• 7.3.3 Linux进程的调度

– Linux进程的创建:

        • 通过fork函数创建进程

        • 创建用户空间进程:vfork()、fork()

        • 创建内核空间进程:copy_process()、kernel_thread()、do_fork()、 sys_vfork()、sys_fork()、sys_clone()

– Linux进程的执行

        • 通过exec函数执行进程

        • exec函数族:execl() 、execlp()、execle()、execv()、execvp()、execve()

– Linux进程的销毁

        • 通过do_exit函数结束进程

• 7.3.2 Linux进程的创建、执行和销毁(重要)

– Linux是一个多进程系统。多进程就是指计算机同时执行多个进程,即同 时运行多个程序。对于多进程系统,就存在多个进程如何进行调度的问 题,包括进程调度时机进程调度依据

– 进程调度时机

① 主动调度:随时可以进行。

② 被动调度:发生在系统调用返回的夕、中断异常处理返回、用户态处理软中 断返回

③ 抢占式内核:处于内核态的进程也可能被调度出去

– 进程调度依据

        • 在所有处于可运行状态的进程中,如何选择最值得运行的进程投入运行,以下4项 是选择的依据:

        ① policy:进程的调度策略。

        ② priority:进程的静态优先级。

        ③ counter:进程剩余的时间片(进程的动态优先级)。

        ④ rt_priority:用于实时进程间的选择

– 进程调度函数:schedule()函数

        ① 主动调度:主动调用schedule()函数。

        ② 被动调度:被动调用schedule()函数。

• 7.4 ARM-Linux模块机制

• Linux是单内核的,单内核最大的优点是效率高,因为所有的内容都集中在一起, 单内核也有可扩展性差、可维护性差的缺点。

– 单内核,是个很大的进程。它的内部又能够被分为若干模块(或是层次或其他)。但是在运行的时候,它是个单独的二 进制大映象。其模块间的通讯是通过直接调用其他模块中的函数实现的,而不是消息传递。单内核结构的例子:传统的 UNIX内核----例如伯克利大学发行的版本,Linux内核。

微内核结构由一个非常简单的硬件抽象层和一组比较关键的原语或系统调用组成,这些原语仅仅包括了建立一个系统必 需的几个部分,如线程管理地址空间进程间通信等。微内核的例子:AIX,BeOS,L4微内核系列,.Mach中用于GNU Hurd和Mac OS X,Minix,MorphOS,QNX,RadiOS,VSTa,鸿蒙OS。

混合内核它很像微内核结构,只不过它的的组件更多的在核心态中运行,以获得更快的执行速度。混合内核实质上是微 内核,只不过它让一些微核结构运行在用户空间的代码运行在内核空间,这样让内核的运行效率更高些。混合内核的例 子: BeOS 内核 ,DragonFly BSD,ReactOS内核,Windows NT、Windows 2000、Windows XP、Windows Server 2003以 及Windows Vista等基于NT技术的操作系统。

外内核系统,也被称为纵向结构操作系统,是一种比较极端的设计方法。 外内核这种内核不提供任何硬件抽象操作,但 是允许为内核增加额外的运行库,通过这些运行库应用程序可以直接地或者接近直接地对硬件进行操作。外核设计还停 留在研究阶段,没有任何一个商业系统采用了这种设计。几种概念上的操作系统正在被开发,如剑桥大学的Nemesis, 格拉斯哥大学的Citrix系统和瑞士计算机科学院的一套系统。麻省理工学院也在进行着这类研究。

模块机制的引入就是为了弥补这一缺点(可扩展性差、可维护性差)。

• 模块(内核模块,动态可加载内核模块,Loadable Kernel Module,LKM)是Linux 内核向外部提供的一个插口

• 7.4.1 Linux模块概述

– Linux内核支持动态可加载模块(Loadable Kernel Module,LKM), 模块是内核的一部分,模块通常是设备驱动程序,但是并没有编 译到内核里面去。

– 与模块相关的命令:

insmod:加载模块

rmmod:卸载模块

lsmod:列出已经安装的模块

depmod:产生模块依赖的映射文件

⑤ modprob:根据depmod命令所产生的相依关系,决定要载入哪些 模块

• 7.4.2 模块代码结构

– module_test实验驱动程序(模块)的代码结构:

• 头文件

#include <linux/kernel.h>
#include <linux/module.h>

 • 模块宏声明

MODULE_LICENSE("GPL");
MODULE_ALIAS("hqyj:module");
MODULE_AUTHOR("HQYJ <yanfa@hqyj.com>");
MODULE_DESCRIPTION("A sample Hello World module");

 • 模块初始化函数

static int __init hello_init(void){printk("hello init\n");return 0;}

 • 模块退出函数

static void __exit hello_exit(void){printk("hello exit\n");}

 • 入口、出口函数设置

module_init(hello_init);
module_exit(hello_exit);

• 7.4.3 模块的加载

– 模块加载的两种方式:

手工加载模块:通过insmod命令将模块加载到内核:

– insmodmodule_test.ko:加载module_test.ko模块

② 根据需要加载模块到内核:当内核发现需要某个模块时,内核守 护进程(kerneld)加载该模块到内核。

• 7.4.4 模块的卸载

– 使用rmmod命令卸载模块:

• rmmod module_test:卸载module_test.ko模块

– 但是当内核在使用模块时,该模块是不能被卸载的。

• 7.4.5 版本依赖

模块代码一定要在连接到不同内核版本之前重新编译, 因为模块是结合到某个特殊内核版本的数据结构和数 据原型上,不同的内核版本的接口可能差别很大。

模块依赖于内核的版

• 7.5 ARM-Linux系统启动和初始化

• 7.5.1 使用Boot Loader将内核映像载入

– Boot Loader将Linux的内核加载到内存(SDRAM)后,将跳到函数 start_kernel()进入初始化过程。

– start_kernel()函数:位于 /home/linux/workdir/fs3399/system/kernel/init/main.c中。

• 7.5.2 内核引导第一部分:内核数据结构初 始化

– start_kernel()函数中调用了一系列初始化函数,以完成内核 (Kernel)本省的设置。

– start_kernel()函数最后启动init过程,创建第一个内核线程,调用 init()函数

• 7.5.3 内核引导第二部分:外设初始化

init()函数作为内核线程,首先锁定内核,然后调用 do_basic_setup()函数,完成外设及其驱动程序的加载和初始化

• 7.5.4 init进程和inittab脚本

init进程是系统所有进程的起点,内核在完成核内引导以后,即在 本线程(进程)空间内加载init程序,它的进程号是1

– init程序需要读取Ubuntu的inittab脚本文件作为其行为指针

• 7.5.5 rc启动脚本

– Linux系统运行后将启动rc脚本S05rc.local)。

– rc启动脚本(S05rc.local)位于实验箱的/etc/rc2.d/目录下。

• 7.5.6 Shell的启动

– Login用户(Ubuntu和实验箱都是linux用户)将启动一个用户指定的Shell,这个指 定的Shell就是/bin/bash

• Shell:俗称壳(用来区别于核),是指“为使用者提供操作界面”的软件(命令解析器)

bash:是一个为GNU计划编写的Unix shell。它的名字是一系列缩写:Bourne-Again SHell — 这是关于Bourne shell(sh)的一个双关语(Bourne again / born again)。 Bourne shell是一个早期的重要shell,由史蒂夫·伯恩在1978年前后编写,并同 Version 7 Unix一起发布。bash则在1987年由布莱恩·福克斯创造。在1990年,Chet Ramey成为了主要的维护者。

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

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

相关文章

win11蓝屏死机 TPM-WMI

1. 打开win11的事件查看器&#xff0c;定位错误 最近两次都是 KB5016061&#xff1a;安全启动数据库和 DBX 变量更新事件 - Microsoft 支持 事件源 TPM-WMI 事件 ID 1796 2. 解决方案 打开注册表&#xff1a;计算机\HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Contro…

Linux命令——3.网络与用户

文章目录 一、网络1.网络测试与诊断2.网络接口配置3.无线网络配置4.防火墙与网络管理6.防火墙管理1&#xff09;firewalld命令2&#xff09;iptables命令 二、用户和群组1.管理员模式2.用户账户管理1&#xff09;useradd创建2&#xff09;usermod修改3&#xff09;userdel 删除…

机器学习算法基础知识1:决策树

机器学习算法基础知识1&#xff1a;决策树 一、本文内容与前置知识点1. 本文内容2. 前置知识点 二、场景描述三、决策树的训练1. 决策树训练方式&#xff08;1&#xff09;分类原则-Gini&#xff08;2&#xff09;分类原则-entropy&#xff08;3&#xff09;加权系数-样本量&am…

_使用CLion的Vcpkg安装SDL2,添加至CMakelists时报错,编译报错

语言&#xff1a;C20 编译器&#xff1a;gcc 14.2 摘要&#xff1a;初次使用Vcpkg添加SDL2&#xff0c;出现CMakelists找不到错误、编译缺失main错误、运行失败错误。 CMakelists缺失错误&#xff1a; 使用CLion的Vcpkg安装SDL2时&#xff0c;按照指示把对应代码添加至CMakel…

Lumos学习王佩丰Excel第二十二讲:制作甘特图与动态甘特图

一、制作双向条形图 1. 分离坐标轴 2. 自定义坐标轴数字格式&#xff1a;加分号加正常数字 3. 修改图表背景 修改图片艺术效果&#xff1a;虚化图片 二、制作甘特图 1、甘特图定义 甘特图&#xff08;Gantt chart&#xff09;又称为横道图、条状图&#xff08;Bar chart&…

el-pagination 为什么只能展示 10 条数据(element-ui@2.15.13)

好的&#xff0c;我来帮你分析前端为什么只能展示 10 条数据&#xff0c;以及如何解决这个问题。 问题分析&#xff1a; pageSize 的值&#xff1a; 你的 el-pagination 组件中&#xff0c;pageSize 的值被设置为 10&#xff1a;<el-pagination:current-page"current…

【网络安全实验室】SQL注入实战详情

如果额头终将刻上皱纹&#xff0c;你只能做到&#xff0c;不让皱纹刻在你的心上 1.最简单的SQL注入 查看源代码&#xff0c;登录名为admin 最简单的SQL注入&#xff0c;登录名写入一个常规的注入语句&#xff1a; 密码随便填&#xff0c;验证码填正确的&#xff0c;点击登录…

ruoyi 多租户 开启后针对某一条sql不适用多租户; 若依多租户sql规则修改

文章参考&#xff1a;多租户功能 | Ruoyi-TDesign 忽略租户​ 1.如果需要指定单独 SQL 不开启过滤&#xff0c;可在对应的 Mapper 接口添加如下忽略注解&#xff1a; InterceptorIgnore(tenantLine "true", dataPermission "false") 此处注意事项 使…

一文理解条件竞争漏洞

视频教程在我主页简介或专栏里 目录&#xff1a; 理解竞争条件的基本概念 限制超越型竞争条件 使用 Burp Repeater 检测和利用限制超限竞态条件 方法论 1 — 预测潜在的冲突 2 — 线索 3 — 概念验证 如何防止竞态条件漏洞 理解竞争条件的基本概念 竞争条件(也就是条件竞…

一种基于动态部分重构的FPGA自修复控制器

1.FPGA动态部分重构技术 动态部分重构技术指在FPGA运行时&#xff0c;通过加载部分位流文件来修改FPGA可重构区域中的逻辑设计&#xff0c;修改过程中其余逻辑功能不受影响整个系统也能够持续运行。 下图为FPGA动态部分重构的基本原理图。通过下载A1.bit、A2.bit、A3.bit 或A4.…

计算机网络体系结构基础知识

一、计算机网络的两个目标&#xff1a; ①两台计算机之间通信 ②两台计算机之间的资源共享 二、计算机网络概述 1.定义&#xff1a;利用通信线路将地理上分散的、具有独立功能的计算机系统和通信设备按不同 的形式连接起来&#xff0c;以功能完善的网络软件及协…

云计算学习架构篇之HTTP协议、Nginx常用模块与Nginx服务实战

一.HTTP协议讲解 1.1rsync服务重构 bash 部署服务端: 1.安装服务 [rootbackup ~]# yum -y install rsync 2.配置服务 [rootbackup ~]# vim /etc/rsyncd.conf uid rsync gid rsync port 873 fake super yes use chroot no max connections 200 timeout 600 ignore erro…

合合信息亮相CSIG AI可信论坛,全面拆解AI视觉内容安全的“终极防线”

合合信息亮相CSIG AI可信论坛&#xff0c;全面拆解视觉内容安全的“终极防线”&#xff01; &#x1f42f; AI伪造泛滥&#xff0c;我们还能相信“眼见为实”吗&#xff1f; 近期&#xff0c;由中国图象图形学学会主办的CSIG青年科学家会议 AI可信论坛在杭州成功举办。本次论…

AI 智能助手对话系统

一个基于 React 和 Tailwind CSS 构建的现代化 AI 对话系统&#xff0c;提供流畅的用户体验和丰富的交互功能。 项目链接&#xff1a;即将开放… 功能特点 &#x1f916; 智能对话&#xff1a;支持与 AI 助手实时对话&#xff0c;流式输出回答&#x1f4c1; 文件处理&#xff…

经验证:将数据从索尼传输到Android的 4 种方法

概括 像Android Galaxy S20 这样的新型Android智能手机很酷&#xff0c;但除了将数据从索尼传输到Android之外。众所周知&#xff0c;旧的索尼手机上存储着大量的文件&#xff0c;因此将数据从旧的索尼手机传输到新的Android手机非常重要。为了解决这个问题&#xff0c;我们做…

IDEA 搭建 SpringBoot 项目之配置 Maven

目录 1?配置 Maven 1.1?打开 settings.xml 文件1.2?配置本地仓库路径1.3?配置中央仓库路径1.4?配置 JDK 版本1.5?重新下载项目依赖 2?配置 idea 2.1?在启动页打开设置2.2?配置 Java Compiler2.3?配置 File Encodings2.4?配置 Maven2.5?配置 Auto Import2.6?配置 C…

走方格(蓝桥杯2020年试题H)

【问题描述】在平面上有一些二维点阵。这些点的编号就像二维数组的编号一样&#xff0c;从上到下依次为第1~n行&#xff0c;从左到右依次为第1~m列&#xff0c;每个点可以用行号和列号表示。 现在有个人站在第1行第1列&#xff0c;他要走到第n行第m列&#xff0c;只能向右或者向…

python opencv的orb特征检测(Oriented FAST and Rotated BRIEF)

官方文档&#xff1a;https://docs.opencv.org/4.10.0/d1/d89/tutorial_py_orb.html SIFT/SURF/ORB对比 https://www.bilibili.com/video/BV1Yw411S7hH?spm_id_from333.788.player.switch&vd_source26bb43d70f463acac2b0cce092be2eaa&p80 ORB代码 import numpy a…

全面解析 Node-RED:功能、Docker 部署与实战示例

言简意赅的讲解Node-RED解决的痛点 Node-RED 是一个基于流的编程工具&#xff0c;专为物联网&#xff08;IoT&#xff09;应用而设计。它通过可视化的编程界面&#xff0c;使开发者能够轻松地连接各种硬件设备、API 以及在线服务&#xff0c;构建复杂的应用流程。本文将详细介…

使用 CSS 的 `::selection` 伪元素来改变 HTML 文本选中时的背景颜色

定义 ::selection 伪元素&#xff1a; 在你的 CSS 文件中&#xff0c;添加 ::selection 伪元素&#xff0c;并设置 background-color 属性来改变选中文本的背景颜色。 示例代码&#xff1a; ::selection {background-color: yellow; /* 你可以根据需要更改颜色 */color: black…