当前位置: 首页 > news >正文

Linux内存管理(Linux内存架构,malloc,slab的实现)

文章目录

  • 前言
  • 一、Linux进程空间内存分配
  • 二、malloc的实现机理
  • 三、物理内存与虚拟内存
    • 1.物理内存
    • 2.虚拟内存
  • 四、磁盘和物理内存区别
  • 五、页
      • 页的基本概念:
      • 分页管理的核心概念:
      • Linux 中分页的实现:
      • 总结:
  • 六、伙伴算法
      • 伙伴算法的核心概念:
      • 伙伴算法的工作原理:
      • 伙伴算法的优缺点:
        • 优点:
        • 缺点:
      • 伙伴算法的实现:
      • 例子:
      • 总结:


前言

本篇文章开始讲解Linux的内存管理,深入了解内存管理有助于我们深入Linux底层逻辑,加强对Linux的学习和了解。

一、Linux进程空间内存分配

下面是一个32位的Linux进程空间内存排布具体的情况:
在这里插入图片描述
内核空间(1GB):内核空间是操作系统核心代码和数据的所在区域。这部分空间由操作系统内核独占,用于执行操作系统的任务,如操作系统的调度、内存管理、驱动程序和系统调用等。在内核空间中,所有的内核模块和驱动程序都运行,并可以直接访问硬件资源。

用户空间(3GB):用户空间是给用户程序运行的区域,其中包含了应用程序的代码、数据和堆栈等。用户空间是被操作系统管理和分配的,用户程序在这个空间中运行,并通过系统调用与内核进行交互。用户空间中的进程无法直接访问内核空间中的数据和功能,而必须通过系统调用接口来请求内核的服务。

在用户空间中,可以进一步细分不同的段:

1.代码段:存储可执行程序的指令,即程序的二进制代码。

2.数据段:存储全局变量和静态变量等数据。

3.堆区:用于动态分配内存,通过调用malloc()、new等函数实现。

4.栈区:用于存储函数调用时的局部变量和函数调用的上下文。

这种地址空间的分配方式可以提供对操作系统和应用程序的良好隔离,确保系统的稳定性和安全性。操作系统内核在内核空间中运行,可以对硬件资源进行直接访问和管理,而用户程序在用户空间中运行,通过系统调用接口与内核进行通信,实现各种操作和功能。

二、malloc的实现机理

当使用malloc函数时会调用到brk系统调用。

brk系统调用用于改变进程的数据段的结束地址,从而调整进程的堆空间大小。当调用malloc函数时,它会向操作系统请求一块指定大小的内存空间。操作系统会检查当前的堆空间大小,并根据需要调整堆的结束地址,以供新的内存分配。

malloc函数会调用brk系统调用来增加堆空间的大小,以适应所需分配的内存大小。如果请求的内存大小超过了当前堆空间的剩余大小,则操作系统会扩展堆空间,使其满足请求。反之,如果请求的内存大小较小,当前剩余的堆空间可能会被重新分割,并返回相应大小的内存块给malloc函数。

调用brk系统调用将会调用到下面的函数:

这个函数的主要作用是根据用户空间的请求调整进程的数据段结束地址,从而动态调整进程的堆空间大小。

SYSCALL_DEFINE1(brk, unsigned long, brk)
{unsigned long retval;unsigned long newbrk, oldbrk;struct mm_struct *mm = current->mm;struct vm_area_struct *next;unsigned long min_brk;bool populate;if (down_write_killable(&mm->mmap_sem))return -EINTR;#ifdef CONFIG_COMPAT_BRK/** CONFIG_COMPAT_BRK can still be overridden by setting* randomize_va_space to 2, which will still cause mm->start_brk* to be arbitrarily shifted*/if (current->brk_randomized)min_brk = mm->start_brk;elsemin_brk = mm->end_data;
#elsemin_brk = mm->start_brk;
#endifif (brk < min_brk)goto out;/** Check against rlimit here. If this check is done later after the test* of oldbrk with newbrk then it can escape the test and let the data* segment grow beyond its set limit the in case where the limit is* not page aligned -Ram Gupta*/if (check_data_rlimit(rlimit(RLIMIT_DATA), brk, mm->start_brk,mm->end_data, mm->start_data))goto out;newbrk = PAGE_ALIGN(brk);oldbrk = PAGE_ALIGN(mm->brk);if (oldbrk == newbrk)goto set_brk;/* Always allow shrinking brk. */if (brk <= mm->brk) {if (!do_munmap(mm, newbrk, oldbrk-newbrk))goto set_brk;goto out;}/* Check against existing mmap mappings. */next = find_vma(mm, oldbrk);if (next && newbrk + PAGE_SIZE > vm_start_gap(next))goto out;/* Ok, looks good - let it rip. */if (do_brk(oldbrk, newbrk-oldbrk) < 0)goto out;set_brk:mm->brk = brk;populate = newbrk > oldbrk && (mm->def_flags & VM_LOCKED) != 0;up_write(&mm->mmap_sem);if (populate)mm_populate(oldbrk, newbrk - oldbrk);return brk;out:retval = mm->brk;up_write(&mm->mmap_sem);return retval;
}

三、物理内存与虚拟内存

1.物理内存

物理内存是指计算机中实际存在的硬件内存。它是由RAM(Random Access Memory,随机存取内存)或其他类似的硬件组成的,用于存储正在执行的程序和数据。物理内存是直接与计算机的中央处理器(CPU)相连,提供快速、随机访问数据的能力。

2.虚拟内存

虚拟内存是操作系统提供的一种抽象概念。它扩展了可用的内存空间,使得程序可以使用比物理内存更大的地址空间。虚拟内存将程序所需的内存分为连续的地址空间块,称为虚拟地址空间。每个程序都有自己的虚拟地址空间,这使得每个程序能够以独立的方式执行,而不会互相干扰。

虚拟内存的工作原理是将部分程序和数据存储在物理内存中,而将不常用的部分保存在磁盘上的交换文件中。当程序需要访问虚拟内存中的某个地址时,操作系统会根据某种映射机制将虚拟地址转换为物理地址。这个过程被称为虚拟内存管理。通过这种方式,系统可以运行更多的程序,即使物理内存有限。

虚拟内存的主要优势之一是提供了更大的地址空间,以支持大型程序和多任务操作系统。它还可以通过将不常用的数据存储在磁盘上来节省物理内存的使用,并提供更好的内存管理和数据保护机制。

总结而言,物理内存是计算机实际的硬件内存,而虚拟内存是通过操作系统提供的抽象层面扩展的内存概念,使得程序能够使用比物理内存更大的地址空间,并提供更好的内存管理和保护机制。

四、磁盘和物理内存区别

物理内存和磁盘是不同的存储介质,它们在计算机系统中扮演不同的角色和功能。物理内存用于临时存储当前正在执行的程序和数据,而磁盘用于长期存储文件和持久性数据。

五、页

在 Linux 操作系统中,“页”(Page)是内存管理中的一个重要概念,尤其是在虚拟内存管理中。操作系统通过页来管理物理内存和虚拟内存之间的映射,使得每个进程能够独立地使用其虚拟地址空间,而不直接操作物理内存。这种分页管理有助于实现内存保护、内存共享、内存映射文件等机制。

页的基本概念:

  1. 虚拟内存与物理内存

    • 虚拟内存:是操作系统为每个进程提供的虚拟地址空间。每个进程看到的内存地址是连续的,这与实际的物理内存地址无关。
    • 物理内存:是计算机硬件中的实际内存(RAM),其中存储了系统运行时的程序数据和操作系统的内核。
  2. 页的基本定义

    • 页(Page):虚拟内存的基本单位,操作系统将虚拟地址空间划分为大小相同的块,每个块叫做“页”。通常,一页的大小为 4KB(也可以更大,如 2MB 或 1GB),具体大小取决于体系结构(例如,x86 体系结构通常为 4KB)。
    • 页帧(Page Frame):物理内存的基本单位,物理内存被划分为大小相同的块,每个块叫做“页帧”,对应虚拟内存中的“页”。
  3. 页表(Page Table)

    • 页表是操作系统用于将虚拟地址转换为物理地址的结构。它存储了虚拟页到物理页帧的映射信息。每个进程都有自己的页表,操作系统通过页表来实现虚拟内存到物理内存的映射。
    • 页表的结构通常是多级的(如 2 级页表、4 级页表等),以提高映射效率并减少内存消耗。

分页管理的核心概念:

  1. 虚拟地址空间划分
    虚拟内存被划分成多个大小相等的页。每个进程的虚拟内存都由操作系统划分成若干个虚拟页,而每个虚拟页与物理内存中的页帧通过页表建立映射。

  2. 页面映射
    在一个分页系统中,每个虚拟地址空间的页都与物理内存的页帧相对应。操作系统使用页表来记录这种映射关系。

    • 当进程访问虚拟内存时,操作系统会通过查找页表来获取对应的物理内存地址。
    • 如果虚拟页没有映射到物理页帧(即页面不在内存中),操作系统会触发缺页异常(Page Fault),并将数据从磁盘加载到内存。
  3. 页替换
    由于物理内存的有限性,可能会发生内存不足的情况。这时,操作系统需要将某些不常用的虚拟页面交换到磁盘上的交换空间(Swap Space)中,将内存中的页面替换出来以腾出空间。当需要这些被替换的页面时,它们会重新加载回内存。

  4. 大页(Huge Pages)
    对于某些需要大量内存的应用程序(如数据库、高性能计算等),操作系统允许使用大页(如 2MB 或 1GB 页),以减少页表的开销。大页可以减少页表项的数量,提高内存访问效率。

  5. 页对齐
    在虚拟内存系统中,页是按照一定的对齐方式来管理的,通常是 4KB。页对齐确保了虚拟地址空间和物理内存的页边界一致性。

  6. 页面保护与权限
    每个页都有一定的访问权限,这些权限决定了该页面的访问行为:

    • 读权限(Read)
    • 写权限(Write)
    • 执行权限(Execute)

    这些权限通过页表项设置,并在进程访问该页面时由硬件控制。

Linux 中分页的实现:

在 Linux 操作系统中,虚拟内存管理采用了分页机制。具体来说,Linux 中有以下几个关键组件与分页管理密切相关:

  1. 内存管理单元(MMU)

    • MMU 是硬件部分,它负责执行虚拟地址到物理地址的转换。MMU 利用页表来进行地址转换,确保虚拟内存和物理内存的隔离。
  2. 页表(Page Table)

    • 在 Linux 中,页表是用来维护虚拟地址到物理地址映射的结构。每个进程有自己的页表,操作系统会维护内核的页表。页表结构通常是多级的,包括顶级页表、二级页表等。
  3. 进程虚拟地址空间

    • 每个进程拥有自己的虚拟地址空间,Linux 会将进程的虚拟内存划分为多个区段,如代码段、数据段、堆、栈等。每个区段的虚拟内存都有对应的页表条目。
  4. 缺页异常处理(Page Fault Handling)

    • 当进程访问未映射的虚拟内存时,MMU 会触发缺页异常。Linux 内核会处理缺页异常,判断是通过磁盘交换加载数据还是分配新的物理页面来解决问题。
  5. 交换空间(Swap Space)

    • 当物理内存不足时,Linux 会使用交换空间(swap)将一些不常用的页存储到磁盘中,从而腾出内存空间。交换空间的存在允许系统使用更多的内存,尽管这会降低性能。

总结:

  • 是内存管理的基本单位,操作系统将虚拟内存分成多个大小相同的页,并通过页表映射到物理内存的页帧。
  • 分页可以有效地隔离进程的虚拟内存,提供内存保护、共享、以及懒加载等机制。
  • Linux 使用分页机制来管理虚拟内存,硬件的 MMU 和操作系统的页表配合工作来完成虚拟地址到物理地址的转换。

分页和虚拟内存技术使得操作系统能够更高效、安全地管理内存,提高了系统的可伸缩性和稳定性。

六、伙伴算法

伙伴算法(Buddy Allocation System)是一种内存分配算法,主要用于动态内存管理,特别是在操作系统中用来管理物理内存的分配和释放。它通过将内存块分成具有2的幂大小的区域,并通过将相邻的空闲块配对来实现高效的内存分配。

伙伴算法的核心概念:

  1. 内存分配单位

    • 伙伴算法将内存划分为大小为2的幂的块(例如 2KB、4KB、8KB 等)。每个内存块的大小是2的幂,因此,内存块的大小总是是 2, 4, 8, 16, 32, 64 … 等。
  2. 分区

    • 内存被划分为多个固定大小的块(例如 4KB、8KB、16KB),这些块可以根据需要进行分配。当请求内存时,算法会从大块开始寻找能够满足请求的内存块。
  3. 伙伴

    • 内存中的每一块内存有一个“伙伴”,这个伙伴是与该块内存相邻且大小相同的块。如果一个块的内存被分配,另一个块(它的伙伴)会保持空闲状态,直到其伙伴被释放。
  4. 合并与分裂

    • 当一个内存块被请求时,伙伴算法会根据请求的大小找到最小的、适合的空闲块。如果所需的内存块比当前空闲块大,则需要分割一个更大的块,并将其分成两个“伙伴”块。这个过程会一直递归下去,直到找到合适的块为止。
    • 当释放内存时,算法会检查是否有空闲块的伙伴,如果有,则将它们合并成一个更大的块。这个合并过程会递归进行,直到没有更多的伙伴可以合并为止。

伙伴算法的工作原理:

  1. 分配内存

    • 当程序请求分配大小为 size 的内存时,伙伴算法会在空闲内存块中寻找足够大的块。
    • 如果没有一个空闲块的大小与 size 精确匹配,则从更大的块开始分配,直到找到一个合适的块。对于大于请求的空闲块,系统会将其分裂成两个相同大小的伙伴,继续进行分配。
  2. 释放内存

    • 当程序释放一个内存块时,系统会检查它的伙伴是否也空闲。如果伙伴是空闲的,两个伙伴块就会合并成一个更大的块。
    • 合并后的内存块会继续与它的伙伴合并,直到没有更多的伙伴可以合并为止。合并过程是递归进行的,可能会发生多次合并。

伙伴算法的优缺点:

优点:
  1. 快速分配和释放

    • 由于内存块的大小是2的幂,因此可以通过简单的计算快速找到合适的内存块。这使得内存分配和释放的速度比较快。
  2. 避免外部碎片

    • 由于内存块大小的限制(2的幂次),即使内存中出现了一些“空闲”空间,也能够通过合并操作将碎片整理成较大的块,减少外部碎片问题。
  3. 内存合并

    • 伙伴算法能够在释放内存时进行合并操作,将相邻的空闲内存块合并成一个更大的内存块,从而有效地利用了内存资源。
缺点:
  1. 内部碎片

    • 虽然伙伴算法可以避免外部碎片,但由于内存块大小是2的幂次,可能会出现内部碎片问题。即一个块比实际需要的内存大,导致浪费一部分内存。
    • 例如,如果请求的内存大小为 6KB,但最小可分配的块为 8KB,那么分配的内存将比实际需要的多出 2KB。
  2. 固定块大小

    • 由于内存块的大小是2的幂次,无法处理一些非常小的内存请求(例如 1KB 或 3KB),可能导致某些内存的浪费。
  3. 管理复杂度

    • 伙伴算法的合并操作需要一定的管理工作,尤其是在大量的内存块需要管理时。每次分配和释放内存时都需要更新伙伴关系。

伙伴算法的实现:

在实现伙伴算法时,通常使用一个位图(Bitmap)或者链表来管理每个内存块的空闲和占用状态。常见的实现方式如下:

  1. 位图管理

    • 使用一个位图来记录每个内存块的状态,每个位表示一个内存块的状态(空闲或已分配)。对于每个内存块,还会记录它的伙伴的位置。
  2. 链表管理

    • 将每个大小相同的内存块放入一个链表中,并通过合并操作将它们连接在一起。每个内存块都可以存储它的伙伴信息。
  3. 分配策略

    • 当请求内存时,算法会从最大的块开始查找,直到找到能够满足请求的最小块。如果找到了更大的块,则将它分割为两个相同大小的伙伴,直到获得合适的块。
  4. 合并策略

    • 在释放内存时,算法会检查释放的块是否和它的伙伴是相邻的,如果是相邻的并且空闲,则将这两个块合并成一个更大的块,并将新的块插入到链表中。

例子:

假设内存的最小分配块为 16 字节,并且内存总量为 64 字节。在内存管理开始时,内存将被划分为多个 16 字节的块,形成一个空闲块链表。假设用户请求了 24 字节的内存:

  1. 由于请求是 24 字节,而最小块是 16 字节,伙伴算法将选择一个 32 字节的块进行分配。
  2. 如果请求的内存已经分配出去,系统会将剩余的 8 字节(32 - 24)与相邻的伙伴块进行合并,形成一个 16 字节的块。

总结:

伙伴算法是一个高效的内存分配和释放算法,通过将内存块分成 2 的幂大小的块,并通过合并相邻的空闲块来减少外部碎片。尽管它存在一定的内部碎片和管理复杂度,但在许多操作系统和内存管理系统中,它仍然是一种常用的内存分配策略。

相关文章:

Linux内存管理(Linux内存架构,malloc,slab的实现)

文章目录 前言一、Linux进程空间内存分配二、malloc的实现机理三、物理内存与虚拟内存1.物理内存2.虚拟内存 四、磁盘和物理内存区别五、页页的基本概念&#xff1a;分页管理的核心概念&#xff1a;Linux 中分页的实现&#xff1a;总结&#xff1a; 六、伙伴算法伙伴算法的核心…...

【机器学习实战中阶】比特币价格预测

比特币价格预测项目介绍 比特币价格预测项目是一个非常有实用价值的机器学习项目。随着区块链技术的快速发展&#xff0c;越来越多的数字货币如雨后春笋般涌现&#xff0c;尤其是比特币作为最早的加密货币&#xff0c;其价格波动备受全球投资者和研究者的关注。本项目的目标是…...

【JVM-9】Java性能调优利器:jmap工具使用指南与应用案例

在Java应用程序的性能调优和故障排查中&#xff0c;jmap&#xff08;Java Memory Map&#xff09;是一个不可或缺的工具。它可以帮助开发者分析Java堆内存的使用情况&#xff0c;生成堆转储文件&#xff08;Heap Dump&#xff09;&#xff0c;并查看内存中的对象分布。无论是内…...

使用vscode在本地和远程服务器端运行和调试Python程序的方法总结

1 官网下载 下载网址&#xff1a;https://code.visualstudio.com/Download 如下图所示&#xff0c;可以分别下载Windows,Linux,macOS版本 历史版本下载链接: https://code.visualstudio.com/updates 2 安装Python扩展工具 打开 VS Code&#xff0c;安装 Microsoft 提供的官…...

AI 编程工具—Cursor 对话模式详解 Chat、Composer 与 Normal/Agent 模式

Cursor AI 对话模式详解:Chat、Composer 与 Normal/Agent 模式 一、简介 Cursor 是一个强大的 AI 辅助编程工具,它提供了多种对话模式来满足不同的开发需求。主要包括: Chat 模式:直接对话交互Composer 模式:结构化编程助手Normal/Agent 模式:不同的 AI 响应策略打开Ch…...

【MySQL】数据库基础知识

欢迎拜访&#xff1a;雾里看山-CSDN博客 本篇主题&#xff1a;【MySQL】数据库基础知识 发布时间&#xff1a;2025.1.21 隶属专栏&#xff1a;MySQL 目录 什么是数据库为什么要有数据库数据库的概念 主流数据库mysql的安装mysql登录使用一下mysql显示数据库内容创建一个数据库创…...

ChatGPT开发教程指南

ChatGPT开发教程指南 一、ChatGPT 概述二、开发环境搭建&#xff08;一&#xff09;硬件要求&#xff08;二&#xff09;软件要求 三、开发流程&#xff08;一&#xff09;数据处理&#xff08;二&#xff09;模型选择与训练&#xff08;三&#xff09;接口开发 四、示例代码 随…...

OpenEuler学习笔记(四):OpenEuler与CentOS的区别在那里?

OpenEuler与CentOS的对比 一、基本信息 起源与背景&#xff1a; OpenEuler&#xff1a;由华为发起&#xff0c;后捐赠给开放原子开源基金会&#xff0c;旨在构建一个开放、多元化的云计算和边缘计算平台&#xff0c;以满足华为及其他企业的硬件和软件需求。CentOS&#xff1a;…...

spring cloud如何实现负载均衡

在Spring Cloud中&#xff0c;实际上并没有直接支持lb:\\这样的URL前缀来自动解析为负载均衡的服务地址。lb:\\这样的表示可能是在某些特定框架、文档或示例中自定义的&#xff0c;但它并不是Spring Cloud官方API或规范的一部分。 Spring Cloud实现负载均衡的方式通常依赖于服…...

LeetCode:37. 解数独

跟着carl学算法&#xff0c;本系列博客仅做个人记录&#xff0c;建议大家都去看carl本人的博客&#xff0c;写的真的很好的&#xff01; 代码随想录 LeetCode&#xff1a;37. 解数独 编写一个程序&#xff0c;通过填充空格来解决数独问题。 数独的解法需 遵循如下规则&#xff…...

如何在idea中搭建SpringBoot项目

如何在idea中快速搭建SpringBoot项目 目录 如何在idea中快速搭建SpringBoot项目前言一、环境准备&#xff1a;搭建前的精心布局 1.下载jdk &#xff08;1&#xff09;安装JDK&#xff1a;&#xff08;2&#xff09;运行安装程序&#xff1a;&#xff08;3&#xff09;设置安装…...

STM32补充——FLASH

目录 1.内部FLASH构成&#xff08;F1&#xff09; 2.FLASH读写过程&#xff08;F1&#xff09; 2.1内存的读取 2.2闪存的写入 2.3FLASH接口寄存器&#xff08;写入 & 擦除相关&#xff09; 3.FLASH相关HAL库函数简介&#xff08;F1/F4/F7/H7&#xff09; 4.编程实战 …...

ASP.NET Core 中的 JWT 鉴权实现

在当今的软件开发中&#xff0c;安全性和用户认证是至关重要的方面。JSON Web Token&#xff08;JWT&#xff09;作为一种流行的身份验证机制&#xff0c;因其简洁性和无状态特性而被广泛应用于各种应用中&#xff0c;尤其是在 ASP.NET Core 项目里。本文将详细介绍如何在 ASP.…...

Docker配置国内镜像源

访问docker hub需要科学上网 在 Docker 中配置镜像地址&#xff08;即镜像加速器&#xff09;可以显著提升拉取镜像的速度&#xff0c;尤其是在国内访问 Docker Hub 时。以下是详细的配置方法&#xff1a; 1. 配置镜像加速器 Docker 支持通过修改配置文件来添加镜像加速器地址…...

qiankun+vite+vue3

基座与子应用代码示例 本示例中,基座为Vue3,子应用也是Vue3,由于qiankun不支持Vite构建的项目,这里还要引入 vite-plugin-qiankun 插件 基座(主应用) 加载qiankun依赖 npm i qiankun -S qiankun配置(src/qiankun) src/qiankun/config.ts export default {subApp…...

如何使用AI工具cursor(内置ChatGPT 4o+claude-3.5)

⚠️温馨提示&#xff1a; 禁止商业用途&#xff0c;请支持正版&#xff0c;充值使用&#xff0c;尊重知识产权&#xff01; 免责声明&#xff1a; 1、本教程仅用于学习和研究使用&#xff0c;不得用于商业或非法行为。 2、请遵守Cursor的服务条款以及相关法律法规。 3、本…...

Linux内核编程(二十一)USB驱动开发-键盘驱动

一、驱动类型 USB 驱动开发主要分为两种&#xff1a;主机侧的驱动程序和设备侧的驱动程序。一般我们编写的都是主机侧的USB驱动程序。 主机侧驱动程序用于控制插入到主机中的 USB 设备&#xff0c;而设备侧驱动程序则负责控制 USB 设备如何与主机通信。由于设备侧驱动程序通常与…...

vue3+ts watch 整理

watch() 一共可以接受三个参数&#xff0c;侦听数据源、回调函数和配置选项 作用&#xff1a;监视数据的变化&#xff08;和Vue2中的watch作用一致&#xff09; 特点&#xff1a;Vue3中的watch只能监视以下四种数据&#xff1a; ref定义的数据。 reactive定义的数据。 函数返…...

2025年最新深度学习环境搭建:Win11+ cuDNN + CUDA + Pytorch +深度学习环境配置保姆级教程

本文目录 一、查看驱动版本1.1 查看显卡驱动1.2 显卡驱动和CUDA对应版本1.3 Pytorch和Python对应的版本1.4 Pytorch和CUDA对应的版本 二、安装CUDA三、安装cuDANN四、安装pytorch五、验证是否安装成功 一、查看驱动版本 1.1 查看显卡驱动 输入命令nvidia-smi可以查看对应的驱…...

USART_串口通讯轮询案例(HAL库实现)

引言 前面讲述的串口通讯案例是使用寄存器方式实现的&#xff0c;有利于深入理解串口通讯底层原理&#xff0c;但其开发效率较低&#xff1b;对此&#xff0c;我们这里再讲基于HAL库实现的串口通讯轮询案例&#xff0c;实现高效开发。当然&#xff0c;本次案例需求仍然和前面寄…...

CAN 网络介绍

背景 在T-Box 产品开发过程中&#xff0c;我们离不开CAN总线&#xff0c;因为CAN总线为我们提供了车身的相关数据&#xff0c;比如&#xff0c;车速、油耗、温度等。用于上报TSP平台&#xff0c;进行国标认证&#xff1b;也帮助我们进行车身控制&#xff0c;比如车门解锁/闭锁…...

pytorch 多机多卡训练方法

在深度学习训练中&#xff0c;使用多机多卡&#xff08;多台机器和多块 GPU&#xff09;可以显著加速模型训练过程。 PyTorch 提供了多种方法来实现多机多卡训练&#xff0c;以下是一些常用的方法和步骤&#xff1a; 1. 使用 torch.distributed 包 PyTorch 的 torch.distribut…...

【智能控制】年末总结,模糊控制,神经网络控制,专家控制,遗传算法

关注作者了解更多 我的其他CSDN专栏 毕业设计 求职面试 大学英语 过程控制系统 工程测试技术 虚拟仪器技术 可编程控制器 工业现场总线 数字图像处理 智能控制 传感器技术 嵌入式系统 复变函数与积分变换 单片机原理 线性代数 大学物理 热工与工程流体力学 …...

Linux系统 C/C++编程基础——使用make工具和Makefile实现自动编译

ℹ️大家好&#xff0c;我是练小杰&#xff0c;今天周二了&#xff0c;距离除夕只有&#xff16;天了&#xff0c;新的一年就快到了&#x1f606; 本文是有关Linux C/C编程的make和Makefile实现自动编译相关知识点&#xff0c;后续会不断添加相关内容 ~~ 回顾:【Emacs编辑器、G…...

kafka学习笔记7 性能测试 —— 筑梦之路

kafka 不同的参数配置对 kafka 性能都会造成影响&#xff0c;通常情况下集群性能受分区、磁盘和线程等影响因素&#xff0c;因此需要进行性能测试&#xff0c;找出集群性能瓶颈和最佳参数。 # 生产者和消费者的性能测试工具 kafka-producer-perf-test.sh kafka-consumer-perf-t…...

C#与AI的共同发展

C#与人工智能(AI)的共同发展反映了编程语言随着技术进步而演变&#xff0c;以适应新的挑战和需要。自2000年微软推出C#以来&#xff0c;这门语言经历了多次迭代&#xff0c;不仅成为了.NET平台的主要编程语言之一&#xff0c;还逐渐成为构建各种类型应用程序的强大工具。随着时…...

multus使用教程

操作步骤如下&#xff1a; 1.在vmware vsphere上配置所有主机使用的端口组安全项 Forged transmits 设置为&#xff1a; Accept Promiscuous Mode 设置为&#xff1a;Accept Promiscuous Mode&#xff08;混杂模式&#xff09;和Forged Transmits&#xff08;伪传输&#xff09…...

用JAVA写算法之输入输出篇

本系列适合原来用C语言或其他语言写算法&#xff0c;但是因为找工作或比赛的原因改用JAVA语言写算法的同学。当然也同样适合初学算法&#xff0c;想用JAVA来写算法题的同学。 常规方法&#xff1a;使用Scanner类和System.out 这种方法适用于leetcode&#xff0c;以及一些面试手…...

场馆预定平台高并发时间段预定实现V2

&#x1f3af; 本文档介绍了场馆预订系统接口V2的设计与实现&#xff0c;旨在解决V1版本中库存数据不一致及性能瓶颈的问题。通过引入令牌机制确保缓存和数据库库存的最终一致性&#xff0c;避免因服务器故障导致的库存错误占用问题。同时&#xff0c;采用消息队列异步处理库存…...

(1)STM32 USB设备开发-基础知识

开篇感谢&#xff1a; 【经验分享】STM32 USB相关知识扫盲 - STM32团队 ST意法半导体中文论坛 单片机学习记录_桃成蹊2.0的博客-CSDN博客 USB_不吃鱼的猫丿的博客-CSDN博客 1、USB鼠标_哔哩哔哩_bilibili usb_冰糖葫的博客-CSDN博客 USB_lqonlylove的博客-CSDN博客 USB …...

Spring Boot 整合 ShedLock 处理定时任务重复执行的问题

&#x1f337; 古之立大事者&#xff0c;不惟有超世之才&#xff0c;亦必有坚忍不拔之志 &#x1f390; 个人CSND主页——Micro麦可乐的博客 &#x1f425;《Docker实操教程》专栏以最新的Centos版本为基础进行Docker实操教程&#xff0c;入门到实战 &#x1f33a;《RabbitMQ》…...

缓存之美:万文详解 Caffeine 实现原理(上)

由于社区最大字数限制&#xff0c;本文章将分为两篇&#xff0c;第二篇文章为缓存之美&#xff1a;万文详解 Caffeine 实现原理&#xff08;下&#xff09; 大家好&#xff0c;我是 方圆。文章将采用“总-分-总”的结构对配置固定大小元素驱逐策略的 Caffeine 缓存进行介绍&…...

PHP语言的网络编程

PHP语言的网络编程 网络编程是现代软件开发中不可或缺的一部分&#xff0c;尤其是在日益发展的互联网时代。PHP&#xff08;Hypertext Preprocessor&#xff09;是一种广泛使用的开源脚本语言&#xff0c;专门用于Web开发。它的灵活性、易用性以及强大的社区支持使得PHP在网络…...

【技巧】优雅的使用 pnpm+Monorepo 单体仓库构建一个高效、灵活的多项目架构

单体仓库&#xff08;Monorepo&#xff09;搭建指南&#xff1a;从零开始 单体仓库&#xff08;Monorepo&#xff09;是一种将多个相关项目集中管理在一个仓库中的开发模式。它可以帮助开发者共享代码、统一配置&#xff0c;并简化依赖管理。本文将通过实际代码示例&#xff0…...

算法项目实时推流

1、搭建流媒体服务器 下载mediamtx 2、视频流直推 ffmpeg -stream_loop -1 -i DJI_20250109112715_0002_W.MP4 -r 30 -c:v libx264 -preset ultrafast -f flv rtmp://192.168.100.20:1935/live/test_chengdu1 3、硬件加速 如果硬件支持&#xff0c;可以使用硬件加速编码器&am…...

软件测试—— 接口测试(HTTP和HTTPS)

软件测试—— 接口测试&#xff08;HTTP和HTTPS&#xff09; HTTP请求方法GET特点使用场景URL结构URL组成部分URL编码总结 POST特点使用场景请求结构示例 请求标头和响应标头请求标头&#xff08;Request Headers&#xff09;示例请求标头 响应标头&#xff08;Response Header…...

PCL K4PCS算法实现点云粗配准【2025最新版】

目录 一、算法原理1、算法概述2、算法流程3、参考文献二、 代码实现1、原始版本2、2024新版三、 结果展示本文由CSDN点云侠原创,原文链接,首发于:2020年4月27日。如果你不是在点云侠的博客中看到该文章,那么此处便是不要脸的抄袭狗。 博客长期更新,本文最近一次更新时间为…...

Docker 学习总结(85)—— docker cp 使用总结

前言 在现代软件开发中,Docker 已成为一种流行的容器化技术。无论是在开发、测试还是生产环境中,管理容器内的文件都是一项常见且重要的任务。本文将详细介绍如何使用 docker cp 命令在 Docker 容器与宿主机之间拷贝文件和目录,并结合一些实际使用场景,帮助您更高效地管理…...

《FMambaIR:一种基于混合状态空间模型和频域的方法用于图像恢复》学习笔记

paper&#xff1a;(PDF) FMambaIR: A Hybrid State Space Model and Frequency Domain for Image Restoration 目录 摘要 一、引言 二、相关工作 1、图像恢复 2、频率学习 3、状态空间模型&#xff08;SSM&#xff09; 三、框架 1、基本知识 2、整体框架 3、F-Mamba…...

PyQt5 超详细入门级教程上篇

PyQt5 超详细入门级教程 上篇&#xff1a;1-3部分&#xff1a;PyQt5基础与常用控件 第1部分&#xff1a;初识 PyQt5 和安装 1.1 什么是 PyQt5&#xff1f; PyQt5 是 Python 的图形用户界面 (GUI) 框架&#xff0c;它基于强大的 Qt 库。Qt 是一个跨平台的 C 框架&#xff0c;用…...

通信协议—WebSocket

一、WebSocket编程概念 1.1 什么是WebSocket WebSocket 是一种全双工通信协议&#xff0c;允许在客户端&#xff08;通常是浏览器&#xff09;和服务器之间建立持久连接&#xff0c;以实现实时的双向通信。它是 HTML5 标准的一部分&#xff0c;相比传统的 HTTP 请求&#xff…...

FFmpeg音视频采集

文章目录 音视频采集音频采集获取设备信息录制麦克风录制声卡 视频采集摄像机画面采集 音视频采集 DirectShow&#xff08;简称DShow&#xff09;是一个Windows平台上的流媒体框架&#xff0c;提供了高质量的多媒体流采集和回放功能&#xff0c;它支持多种多样的媒体文件格式&…...

【微机原理与接口技术】定时控制接口

文章目录 8253的引脚和工作方式内部结构和引脚工作方式方式0&#xff1a;计数结束中断方式1&#xff1a;可编程单稳脉冲方式2&#xff1a;周期性负脉冲输出方式3&#xff1a;方波发生器方式4&#xff1a;软件触发的单次负脉冲输出方式5&#xff1a;硬件触发的单次负脉冲输出各种…...

AG32 FPGA 的 Block RAM 资源:M9K 使用

1. 概述 AG32 FPGA 包含了 4 个 M9K 块&#xff0c;每个 M9K 块的容量为 8192 bits&#xff0c;总计为 4 个 M9K&#xff08;4K bytes&#xff09;。这使得 AG32 的内部存储非常适合嵌入式应用&#xff0c;能够有效地利用片上资源。 M9K 参数 参考自《AGRV2K_Rev2.0.pdf》。…...

第3天:阿里巴巴微服务解决方案概览

一、阿里巴巴微服务解决方案概述 阿里巴巴在微服务领域贡献了多个开源项目&#xff0c;形成了完整的微服务解决方案&#xff0c;广泛应用于分布式系统开发。其中&#xff0c;Spring Cloud Alibaba 是基于 Spring Cloud 构建的一站式微服务解决方案&#xff0c;集成了多个阿里巴…...

在Ubuntu上安装RabbitMQ教程

1、安装erlang 因为rabbitmq是基于erlang开发的&#xff0c;所以要安装rabbitmq&#xff0c;首先需要安装erlang运行环境 apt-get install erlang执行命令查是否安装成功&#xff1a;erl&#xff0c;疯狂 Ctrlc 就能退出命令行 2、安装rabbitmq 1、查看erlang与rabbitmq版本…...

WPF 引发类型为“System.Windows.Forms.AxHost+InvalidActiveXStateException”的异常 解决办法

本章讲述&#xff1a;引发类型为“System.Windows.Forms.AxHostInvalidActiveXStateException”的异常 解决办法。 这几天在做一个WPF功能时&#xff0c;因为引用了第三方的OCX控件&#xff0c;一般来说一个对象只要实例化就行了, 但是在引用这个控件时就报引发类型为“System.…...

Vue 3中导航守卫(Navigation Guard)结合Axios实现token认证机制

在Vue 3中&#xff0c;导航守卫&#xff08;Navigation Guard&#xff09;用于拦截路由的变化&#xff0c;可以在用户访问页面前进行检查。结合Axios进行token认证机制时&#xff0c;我们可以通过导航守卫在路由跳转时&#xff0c;检查用户的认证状态&#xff0c;确保用户有有效…...

代码随想录算法【Day28】

Day28 122.买卖股票的最佳时机 II 最终利润是可以分解的 假如第 0 天买入&#xff0c;第 3 天卖出&#xff0c;那么利润为&#xff1a;prices[3] - prices[0]。 相当于(prices[3] - prices[2]) (prices[2] - prices[1]) (prices[1] - prices[0])。 所以把利润分解为每天…...

【21】Word:德国旅游业务❗

目录 题目 NO1.2.3 NO4 NO5.6 NO7 NO8.9.10.11 题目 NO1.2.3 F12&#xff1a;另存为布局→页面设置→页边距&#xff1a;上下左右选中“德国主要城市”→开始→字体对话框→字体/字号→文本效果&#xff1a;段落对话框→对齐方式/字符间距/段落间距 NO4 布局→表对话框…...