《深入理解计算机系统(CSAPP)》第9章虚拟内存 - 学习笔记
写在前面的话:此系列文章为笔者学习CSAPP时的个人笔记,分享出来与大家学习交流,目录大体与《深入理解计算机系统》书本一致。因是初次预习时写的笔记,在复习回看时发现部分内容存在一些小问题,因时间紧张来不及再次整理总结,希望读者理解。
《深入理解计算机系统(CSAPP)》第3章 程序的机器级表示 - 学习笔记_友人帐_的博客-CSDN博客
《深入理解计算机系统(CSAPP)》第5章 优化程序性能 - 学习笔记_友人帐_的博客-CSDN博客
《深入理解计算机系统(CSAPP)》第6章 存储器层次结构 - 学习笔记_友人帐_的博客-CSDN博客
《深入理解计算机系统(CSAPP)》第7章 链接- 学习笔记_友人帐_的博客-CSDN博客
《深入理解计算机系统(CSAPP)》第8章 异常控制流 - 学习笔记_友人帐_的博客-CSDN博客
《深入理解计算机系统(CSAPP)》第9章虚拟内存 - 学习笔记_友人帐_的博客-CSDN博客
第九章 虚拟内存
内存管理单元(Memory Management Unit, MMU):专用硬件,利用存放在主存中的查询表来动态翻译虚拟地址,该表的内容由操作系统管理。
1. 地址空间
-
地址空间(address space):一个非负整数地址的有序集合:{0, 1, 2, …}
-
线性地址空间(linear address space):地址空间中的整数是连续的,则称为线性地址空间
-
物理地址空间(physical address sapce): M = 2 m M=2^m M=2m个物理地址的集合{0, 1, 2, 3, …, M-1}
-
虚拟地址空间(virtual address space): N = 2 n N=2^n N=2n个虚拟地址的集合{0, 1, 2, 3,…, N-1}
虚拟地址的思想:允许每个数据对象有多个独立的地址,其中每个地址都选自一个不同的地址空间。
为什么要使用虚拟内存Virtual Memory(VM)?
- 有效使用主存:使用DRAM作为部分虚拟地址空间的缓存
- 简化内存管理:每个进程都使用统一的线性地址空间
- 独立地址空间:个进程不能影响其他进程的内存;用户程序无法获取特权内核信息和代码
2. 虚拟内存作为缓存的工具
虚拟内存:存放在磁盘上、有N个连续字节的数组。
磁盘上这个数组的内容被缓存在物理内存中(DRAM cache),缓存块被称为页(页面大小为 P = 2 p P=2^p P=2p)。
虚拟页分类:
- 未分配的:VM系统还未分配(或者创建)的页。未分配的块没有任何数据和它们相关联,因此也就不占用任何磁盘空间。
- 缓存的:当前已缓存在物理内存中的已分配页。
- 未缓存的:未缓存在物理内存中的已分配页。
2.1 DRAM缓存的组织结构
DRAM若不命中,会产生巨大的不命中开销,因此采用:
- 大的虚拟页面。标准4KB,可达到4MB/页。
- DRAM缓存使用全相联映射:任何虚拟页都可以放置在任何物理页中。
- 不命中时使用了更复杂精密的替换算法。
- DRAM缓存使用写回法(磁盘访问时间长)。
2.2 页表(Page Table, PT)
存放**页表条目(Page Table Entry, PTE)**的数组,将虚拟页地址映射到物理页地址。DRAM中的每个进程都有自己的页表。
- 如果设置了有效位,那么地址字段就表示DRAM中相应的物理页的起始位置,这个物理页中缓存了该虚拟页。
- 如果没有设置有效位,
- 那么一个空地址表示这个虚拟页还未被分配。
- 否则,这个地址就指向该虚拟页在磁盘上的起始位置。
2.3 页命中
要访问的虚拟内存中的内容存在于物理内存中,即DRAM缓存命中。
2.4 缺页
DRAM缓存不命中称为缺页(page fault)。
缺页处理:
以访问VP3,选择VP4作为牺牲页为例:
CPU引用了VP3中的一个字,地址翻译硬件从内存中读取PTE3,有效位为0推断出VP3未被缓存,并且触发一个缺页异常。
缺页异常调用内核中的缺页异常处理程序,该程序会选择一个牺牲页,在此例中就是存放在PP3中的VP4。如果VP4已经被修改了,那么内核就会将它复制回磁盘。接着内核将VP4的页表条目有效位置0。
接下来,内核从磁盘复制VP3到内存中的PP3,更新PTE3,随后异常处理程序返回,重新执行导致缺页的指令,然后页命中。
使用按需页面调度:只有当不命中时,才换入页面。
2.5 分配页面
内核在磁盘上分配一个新的虚拟内存页,并且将某个PTE指向这个新的位置(该虚拟页在磁盘上的起始位置)。
仅是分配,未载入内存,有效位是0。
2.6 虚拟内存效率高的原因
尽管在整个运行过程中程序引用的不同页面的总数可能超出物理内存总的大小,但是局部性原则保证了在任意时刻,程序将趋向于在一个较小的活动页面(active page)集合上工作,这个集合叫做工作集(working set)或者常驻集合(resident set)。在初始开销,也就是将工作集页面调度到内存中之后,接下来对这个工作集的引用将导致命中,而不会产生额外的磁盘流量。
如果工作集的大小超出了物理内存的大小,这时页面将不断地换进换出,叫做抖动(thrashing),性能暴跌。
3. 虚拟内存作为内存管理的工具
核心思想:每个进程都拥有一个独立的虚拟地址空间。
页表将虚拟地址映射到物理地址,多个虚拟页面可以映射到同一个共享物理页面上。
- 简化链接
独立的地址空间允许每个进程的内存映像使用相同的基本格式,而不管代码和数据实际存放在物理内存的何处(结构一致)。
- 简化加载
使向内存中加载可执行文件和共享对象文件更容易。
要把目标文件中.text和.data节加载到一个新创建的进程中,Linux加载器为代码和数据段分配虚拟页,把它们标记为无效的(即未被缓存的),将页表条目指向目标文件中适当的位置。
加载器从不从磁盘到内存实际复制任何数据。而是在每个页初次被引用时,由虚拟内存系统会按照需要自动地调入数据页。
- 简化共享
将不同进程中适当的虚拟页面映射到相同的物理页面,使得进程共享代码和数据,而不必在各进程私有区域内重复复制。
- 简化内存分配
当一个运行在用户进程中的程序要求额外的堆空间时(如调用ma1loc的结果),操作系统分配适当多个连续的虚拟内存页面,并且将它们映射到物理内存中任意位置的物理页面。
4. 虚拟内存作为内存保护的工具
通过在PTE上扩展许可位以对访问控制做权限限制。
(内核模式才可访问;可读;可写;可执行)
内存管理单元(MMU)每次访问数据都要检查许可位。如果一条指令违反了这些许可条件,那么CPU就触发一个一般保护故障,将控制传递给一个内核中的异常处理程序。Linux shell一般将这种异常报告为"段错误(segmentation fault)"。
5. 地址翻译
地址翻译就是由一个虚拟地址A获得其物理地址(DRAM)的过程。若结果未空,则说明虚拟地址A是无效的地址,或其对应的内容存储在磁盘上。
5.1 基于页表的地址翻译
- 页表基址寄存器CR3(Page Table Base Register, PTBR):CR3控制寄存器指向第一级页表(L1)的起始位置。CR3的值是每个进程上下文的一部分,每次上下文切换时,CR3的值都会被恢复。
- 由VPN在页表中匹配PTE,获取PPN,与页偏移量PO拼接得到物理地址。
- VPO与PPO是相同的。
(1)页面命中时硬件执行步骤:
第1步:处理器生成一个虚拟地址,并把它传送给MMU。
第2步:MMU生成PTE地址(PTEA),并从高速缓存/主存请求得到PTE。
第3步:高速缓存/主存向MMU返回PTE。
第4步:MMU构造物理地址,并把它传送给高速缓存/主存。
第5步:高速缓存/主存返回所请求的数据字给处理器。
- 整个过程完全由硬件处理。
- 需要访存两次(在高速缓存/主存中获取PTE以构造虚拟地址、由物理地址在高速缓存/主存找数据)。
(2)缺页异常时执行步骤:
第1步:处理器生成一个虚拟地址,并将其传送给MMU。
第2步:MMU生成PTE地址(PTEA),并从高速缓存/主存请求得到PTE。
第3步:高速缓存/主存向MMU返回PTE。
第4步:PTE中的有效位是零,所以MMU触发缺页异常,传递CPU中的控制到操作系统内核中的缺页异常处理程序。
第5步:缺页处理程序确定出物理内存中的牺牲页,如果这个页面已经被修改了,则把它换出到磁盘。
第6步:缺页处理程序页面调入新的页面,并更新内存中的PTE。
第7步:缺页处理程序返回到原来的进程,再次执行导致缺页的指令。
- 由硬件、OS内核协作完成。
- 2x2次访存,(1次内存写入磁盘+1次磁盘写入内存)。
5.2 结合高速缓存和虚拟内存
高速缓存采用物理寻址,多个进程同时在高速缓存中有存储块和共享来自相同虚拟页面的块。
注意,页表条目可以缓存,就像其他的数据字一样。
5.3 利用快表TLB加速地址翻译
后备缓冲器(Translation Lookaside Buffer, TLB)。
目的:为了减少寻找PTE的开销。
TLB是MMU中一个小的、具有高相联度的缓存,实现虚拟页号VPN向物理页号PPN的映射,页数很少的页表可以完全放在TLB中。
(1)访问TLB
TLB的每行都保存着一个由单个PTE组成的块。MMU使用虚拟地址的VPN部分来访问TLB:将VPN划分为TLB的组选择和行匹配的标记字段。
(2)TLB的命中与不命中操作
注意:不命中时MMU从L1缓存中取出相应的PTE,并同时存放在TLB中、提供给MMU。
5.4 多级页表
目的:压缩页表的大小。
思想:虚拟地址空间中每个虚拟页不一定全部都分配,也即都还未被使用,也就没必要保存一条PTE在页表中占用空间。
(1)二级页表示例
- **基本情况:**假设32位虚拟地址空间被分为4KB的页,每个页表条目都是4字节。分配情况:内存的前2K个页面分配给了代码和数据,接下来的6K个页面还未分配,再接下来的1023个页面也未分配,接下来的1个页面分配给了用户栈。
**使用一级页表:**需要有 2 32 2 12 = 2 20 = 1 M \frac{2^{32}}{2^{12}}=2^{20}=1M 212232=220=1M个PTE。
使用二级页表:
一级页表中的每个PTE负责映射虚拟地址空间中一个4MB的片(chunk),这里每一片都是由1024个连续的页面组成的。一级页表中仅需要1K个PTE。
如果片 i i i中的每个页面都未被分配,那么一级 P T E i PTE_i PTEi就为空。如果在片 i i i中至少有一个页是分配了的,那么一级 P T E i PTE_i PTEi就指向一个二级页表的基址。二级页表中的每个PTE都负责映射一个4KB的虚拟内存页面。
- 为什么二级页表可以减少内存要求:
①如果一级页表中的一个PTE是空的,那么相应的二级页表就根本不会存在。
②只有一级页表才需要总是在主存中(因为使用最频繁);虚拟内存系统可以在需要时创建、页面调入或调出二级页表,这就减少了主存的压力;只有最经常使用的二级页表才需要缓存在主存中。
(2)K级页表的地址翻译
-
虚拟地址被划分成为k个VPN和1个VPO,每个 V P N i VPN_i VPNi都是一个到第 i i i级页表的索引。
-
前k-1级页表中的每个PT都指向下一级的某个页表的基址。
-
第k级页表中的每个PTE包含某个物理页面的PPN,或者一个磁盘块的地址。
为了构造物理地址,在能够确定PPN之前,MMU必须访问k个PTE。对于只有一级的页表结构,PPO和VPO是相同的。
通过将不同层次上页表的PTE缓存起来,带多级页表的地址翻译并不比单级页表慢很多。
5.5 一个端到端的地址翻译示例
(1)基本假设
- 内存是按字节寻址的。
- 虚拟地址是14位长的(n=14)。
- 物理地址是12位长的(m=12)。
- 页面大小是64字节(P=64)。
- TLB是四路组相联的,总共有16个条目。
- L1d-cache是物理寻址、直接映射的,行大小为4字节,而总共有16个组。
(2)虚拟地址和物理地址的格式
每个页面大小为64B,需要6位地址做页内偏移量。故低6位为VPO、PPO,其余的作为VPN和PPN。
(3)TLB的格式
TLB是四路组相联的,总共有16个条目。故共有4组,需要2位作为组索引TLBI,其余作为标记TLBT。
(TLB是利用VPN的位进行虚拟寻址的)
(4)页表格式
采用单级页表。共需要 2 14 2 6 = 2 8 = 256 \frac{2^{14}}{2^{6}}=2^{8}=256 26214=28=256条PTE(虚拟页面大小/页面大小)
使用VPN来进行标识,VPN并不是页表的一部分,也不存储在内存中。
(5)Cache格式
由每行4字节,需要块内偏移量2位;
直接映射(1行就是1组),共16个组,需要4位组索引。
使用物理地址寻址。
(6)读取示例
- TLB读取命中示例
假设CPU读取0x03d4处的1个字节:
- CPU给出的即虚拟地址,故写出VPN和VPO,在VPN中划分出TLBT和TLBI:
-
先上TLB中去寻找第0x3组中有无标记为0x03的块,发现有且valid为1。故此时TLB命中,不存在缺页故障,找到PPN为0x0D
-
MMU将来自PTE的PPN和来自虚拟地址的VPO连接起来,形成物理地址0x354。
-
MMU将物理地址发给高速缓存L1,缓存从物理地址中划分出块内偏移CO(0x0)、组索引CI(0x5)和缓存表及CT(0x0D)。在Cache中找到对应块,且valid有效,读出在偏移量CO处的数据字节0x36返回给MMU,由MMU传递给CPU。
- TLB不命中示例
如果TLB不命中,那么MMU必须从页表中的PTE中取出PPN。如果得到的PTE是无效的,那么就产生一个缺页,内核必须调入合适的页面,重新运行这条加载指令。
- TLB命中但是Cache不命中
另一种可能性是PTE是有效的,但是所需要的内存块在缓存中不命中。
6. Core i7/Linux内存系统
6.1 虚拟内存系统
(1)四级页表层次结构
(2)地址翻译概况
为了简化,没有显示i-cache、i-TLB和L2统一TLB。
(3)各级页表中条目格式
第1~3级
每个条目引用一个4KB子页表。注意PS位
当P=1时:地址字段包含一个40位,对于的下一级页表的基地址。
当P=0时:前面保存的都是磁盘上的页表位置。
第4级
注意D位
P=1时:地址字段包括一个40位PPN,指向物理内存中某一页的基地址。
当P=0时:前面保存的都是磁盘上的页表位置。
(4)页表翻译过程
当MMU翻译每一个虚拟地址时:
- 每次访问一个页时,MMU都会设置A位,称为引用位(reference bit)。内核可以用这个引用位来实现它的页替换算法。
- 每次对一个页进行了写之后,MMU都会设置D位,又称修改位或脏位(dirty bit)。修改位告诉内核在复制替换页之前是否必须写回牺牲页
- 内核可以通过调用一条特殊的内核模式指令来清除引用位或修改位。
下图给出了Core i7MMU如何使用四级的页表来将虚拟地址翻译成物理地址。
36位VPN被划分成四个9位的片,每个片被用作到一个页表的偏移量。CR3寄存器包含L1页表的物理地址。VPN1提供到一个L1PET的偏移量,这个PTE包含L2页表的基地址。VPN2提供到一个L2PTE的偏移量,以此类推。
6.2 单个进程的虚拟地址空间
物理内存:方便内核访问物理内存中任何特定的位置。
(1)Linux虚拟内存区域
任务结构中的一个条目指向mm_struct,它描述了虚拟内存的当前状态:
- pgd指向第一级页表(页全局目录)的基址;
- mmap指向一个vm_area_structs(区域结构)的链表,其中每个vm_area_structs都描述了当前虚拟地址空间的一个区域。当内核运行这个进程时,就将pgd存放在CR3控制寄存器中。
其中,每个vm_area_structs包含:
- vm_start:指向这个区域的起始处。
- vm_end:指向这个区域的结束处。
- vm_prot:描述这个区域内包含的所有页的读写许可权限。
- vm_flags:描述这个区域内的页面是与其他进程共享的,还是这个进程私有的(还描述了其他一些信息)。
- vm_next:指向链表中下一个区域结构。
(2)Linux缺页异常处理
缺页处理程序检查:
- 地址是否合法?搜索区域链表,确认地址在(合法的某个区域内?否则,非法->段错误
- 访问是否合法?有读、写或执行区域内页面的权限。否则,违反许可,触发保护异常->段错误
7. 内存映射
Linux通过将一个虚拟内存区域与一个磁盘上的对象(object)关联起来,以初始化这个虚拟内存区域的内容,这个过程称为内存映射(memory mapping)。
虚拟内存区域可以映射到两种类型的对象中的一种:
- 磁盘上的普通文件(eg,一个可执行目标文件)
- 文件区被分成页大小的片,对虚拟页面初始化
- 匿名文件(内核创建,全是二进制零)
- 首次访问该区域的虚拟页会引发缺页异常->分配一个全零的物理页(demand-zero pagei请求二进制零的页)
- 一旦该页面被修改,即和其他页面一样
无论在哪种情况中,一旦一个虚拟页面被初始化了,它就在一个由内核维护的专门的交换文件(swap file),之间换来换去。交换文件也叫做交换空间(swap space)或者交换区域。
7.1 再看共享对象
两个进程映射了同一个共享对象,两个进程的虚拟地址可以是不同的:
对于每个映射私有对象的进程,相应私有区域的页表条目都被标记为只读,并且区域结构被标记为私有的写时复制。只要没有进程试图写它自己的私有区域,它们就可以继续共享物理内存中对象的一个单独副本。
然而,只要有一个进程试图写私有区域内的某个页面,那么这个写操作就会触发一个保护故障。当故障处理程序注意到保护异常是由于进程试图写私有的写时复制区域中的一个页面而引起的,它就会在物理内存中创建这个页面的一个新副本,更新页表条目指向这个新的副本,然后恢复这个页面的可写权限,当故障处理程序返回时,CPU重新执行这个写操作,现在在新创建的页面上这个写操作就可以正常执行了。
7.2 再看fork函数
当fork函数被当前进程调用时,内核为新进程创建各种数据结构,并分配给它一个唯一的PID。
为了给这个新进程创建虚拟内存,它创建了当前进程的mm_struct、区域结构和页表的原样副本。它将两个进程中的每个页面都标记为只读,并将两个进程中的每个区域结构都标记为私有的写时复制。
当fork在新进程中返回时,新进程现在的虚拟内存刚好和调用fork时存在的虚拟内存相同。
当这两个进程中的任一个后来进行写操作时,写时复制机制就会创建新页面,因此,也就为每个进程保持了私有地址空间的抽象概念。
即:
- 完全copy,标为只读,也使得能够共享虚拟内存对应的物理空间
- 写操作时,写时复制机制就会创建新页面,保持了每个进程的私有地址空间
7.3 再看execve函数
假设调用execve("a.out", NULL, NULL)
execve函数在当前进程中加载并运行包含在可执行目标文件a.out中的程序,用a.out程序有效地替代了当前程序。加载并运行a.out需要以下几个步骤:
- 删除当前进程虚拟地址的用户部分中的已存在的区域结构(页表、结构体、vm_area_strcut链表)。
- 映射私有区域(创建自己的新的区域结构)。为新程序的代码、数据、bss和栈区域创建新的区域结构,所有这些新的区域都是私有的、写时复制的。
- 代码和数据区域被映射为a.out文件中的.text和.data区。
- bss区域是请求二进制零的,映射到匿名文件,其大小包含在a.out中。
- 栈和堆区域也是请求二进制零的,初始长度为零。
- 映射共享区域。将共享对象动态链接到这个程序,然后再映射到用户虚拟地址空间中的共享区域内。
- 设置程序计数器(PC)。设置当前进程上下文中的PC,使之指向代码区域的入口点。
7.4 使用mmap函数的用户级内存映射
Linux进程可以使用rmap函数来创建新的虚拟内存区域,并将对象映射到这些区域中。
void *mmap(void *start, int len, int prot, int flags, int fd, int offset)
从fd指定磁盘文件的offset处,映射len个字节到一个新创建的虚拟内存区域,该区域从地址stat处开始。
- start:虚拟内存的起始地址,通常定义为NULL
- prot:虚拟内存区域的访问权限
- PROT_READ(可读)
- PROT_WRITE(可写)
- PROT_EXEC(可执行)
- PROT_NONE(不能被访问)
- flags:被映射对象的类型
- MAP_ANON(匿名对象)
- MAP_PRIVATE(私有的写时复制对象)
- MAP_SHARED(共享对象)
- 返回值:指向映射区域开始处的指针
相关文章:

《深入理解计算机系统(CSAPP)》第9章虚拟内存 - 学习笔记
写在前面的话:此系列文章为笔者学习CSAPP时的个人笔记,分享出来与大家学习交流,目录大体与《深入理解计算机系统》书本一致。因是初次预习时写的笔记,在复习回看时发现部分内容存在一些小问题,因时间紧张来不及再次整理…...

信息论与编码 SCUEC DDDD 期末复习
1.证明熵的可加性 2.假设一帧视频图像可以认为是由3*10的五次方个像素组成(每像素均独立变化),如果每个像素可取128个不同的等概率亮度表示。请计算出每帧图像含多少信息量?若有一口述者在约12000个汉字的字汇中选400个字来口述此…...

windows安装python开发环境
最近因工作需要,要学习一下python,所以先安装一下python的开发环境,比较简单 下载和安装Python 首先,在浏览器中打开Python的官方网站(https://www.python.org/downloads/) 然后,从该网站下载与你的操…...

java idea常用的快捷方式
文章目录 java idea常用的快捷方式快速复制选多行改变代码格式化 快速代码编辑psvmsout5.forarr.for快速死循环快速补全代码当方法还没创建的时候抽取具有一定功能的代码变成方法 java idea常用的快捷方式 快速复制 c t r l d \color{red}{ctrld} ctrld 选多行改变 A l t 鼠…...

lwIP 开发指南
目录 lwIP 初探TCP/IP 协议栈是什么TCP/IP 协议栈架构TCP/IP 协议栈的封包和拆包 lwIP 简介lwIP 源码下载lwIP 文件说明 MAC 内核简介PHY 芯片介绍YT8512C 简介LAN8720A 简介 以太网接入MCU 方案软件TCP/IP 协议栈以太网接入方案硬件TCP/IP 协议栈以太网接入方案 lwIP 无操作系…...

RabbitMQ消息属性详解
content-type属性 如同各种标准化的HTTP规范,content-type传输消息体的MIME类型。例如,如果你的应用程序正在发送JSON序列化的数据值,那么将content-type属性设置为application/json将允许尚待开发的消费者应用程序在收到消息时检查消息类型…...

shader 混合模式
在所有着色器执行完毕,所有纹理都被应用,所有像素准备被呈现到屏幕之后,使用Blend命令来操作这些像素进行混合。 3.2 blend的语法 BlendOff:关闭blend混合(默认值) BlendSrcFactor DstFactor :配置并启动混…...

【大数据工具】Hive 安装
Hive 环境搭建与基本使用 Hive 安装包下载地址:https://dlcdn.apache.org/hive/ 注:安装 Hive 前要先安装好 MySQL 1. MySQL 安装 MySQL 安装包下载地址:https://dev.mysql.com/downloads/mysql/archives/community/MySQL%20::%20Downloa…...

Android9.0 iptables用INetd实现app某个时间段禁止上网的功能实现
1.前言 在9.0的系统rom定制化开发中,在system中netd网络这块的产品需要中,会要求设置app某个时间段禁止上网的功能,liunx中iptables命令也是比较重要的,接下来就来在INetd这块实现app某个时间段禁止上网的的相关功能,就是在系统中只能允许某个app某个时间段禁止上网,就是…...

webpack.config.js基础配置(五大核心属性)
在上一节webpack零基础入门中我们在安装完webpack 和 webpack-cli依赖之后,直接通过npx webpack ./src/main.js --modedevelopment的方式对src下的js文件进行了打包。 其中的 ./src/main.js: 指定 Webpack 从 main.js 文件开始打包,不但会打包 main.js&a…...

【华为OD机试】阿里巴巴找黄金宝箱(IV)【2023 B卷|200分】
【华为OD机试】-真题 !!点这里!! 【华为OD机试】真题考点分类 !!点这里 !! 题目描述 一贫如洗的樵夫阿里巴巴在去砍柴的路上,无意中发现了强盗集团的藏宝地, 藏宝地有编号从0-N的箱子,每个箱子上面有一个数字,箱子排列成一个环, 编号最大的箱子的下一个是编号为0的箱…...

Qt6 C++基础入门2 文件结构与信号和槽
目录 标准文件结构widget.hwidget.cppmain.cpppro 文件 信号与槽自定义信号connect 的两种方式 标准文件结构 widget.h widget 对象的头文件 一般会直接在头文件导入所有后续在 cpp 文件内用到的类,所以 include 基本都会写在这里 // 头文件标志起始 #ifndef WID…...

常用模拟低通滤波器的设计——契比雪夫II型滤波器
常用模拟低通滤波器的设计——契比雪夫II型滤波器 切比雪夫 II 型滤波器的振幅平方函数为: 式中,为有效带通截止频率, 是与通带波纹有关的参量, 大,波纹大,; 为 N 阶契比雪夫多项式。 在 Matl…...

SSM 如何使用 Redis 实现缓存?
SSM 如何使用 Redis 实现缓存? Redis 是一个高性能的非关系型数据库,它支持多种数据结构和多种操作,可以用于缓存、队列、计数器等场景。在 SSM(Spring Spring MVC MyBatis)开发中,Redis 可以用来实现数…...

uin-app如何获取微信昵称和头像的博客
在很多应用中都会使用到微信登录功能,这样可以方便用户快速地完成注册、登录等操作。本文将介绍如何通过uin-app获取微信用户的昵称和头像信息。 第一步:准备开发环境 首先,需要下载并安装QQ精简版开发工具(uin-app)…...

第六十七天学习记录:对陈正冲编著《C 语言深度解剖》中关于变量命名规则的学习
最近开始在阅读陈正冲编著的《C 语言深度解剖》,还没读到十分之一就感觉收获颇多。其中印象比较深刻的是其中的变量的命名规则。 里面提到的不允许使用拼音正是我有时候会犯的错。 因为在以往的工作中,偶尔会遇到时间紧迫的情况。 而对于新增加的变量不知…...

matlab 计算点云的线性指数
目录 一、算法原理二、代码实现三、结果展示一、算法原理 选取当前点 p i ( x , y , z ) p_{i}(x,y,z) p<...

SpringBoot集成ElasticSearch
文章目录 前言一、ElasticSearch本地环境搭建二、SpringBoot整合ElasticSearch1.pom中引入ES依赖2.application.yaml配置elasticsearch3.ElasticSearchClientConnect连接ES客户端工具类4.ElasticSearchResult封装响应结果5.Person实体类6.Person实体类7.ElasticsearchControlle…...

分治入门+例题
目录 🥇2.3.2 合并排序 🥇2.3.3 快速排序 🌼P1010 [NOIP1998 普及组] 幂次方 🌳总结 形象点,分治正如“凡治众如治寡,分数是也”,管理少数几个人,即可统领全军 本质ÿ…...

剑指offer打卡
这里写目录标题 day1 二叉树和为某一路径day2复杂链表的复刻day3二叉搜索树与双向链表day4数字排列day5找出出现次数超过一半的次数day6 二进制中1的个数day7 二叉树的最近公共祖先day8 字符串转换为整数day9 构建乘积数组day10不用加减乘除的加法day11求12....nday11 股票的最…...

运维实用脚本整理
运维实用脚本整理 本文脚本仅供参考运维排查问题思路运维排查问题的方法和命令(1)尽可能搞清楚问题的前因后果(2)有谁在?(3)之前发生了什么?(4) 现在在运行的进程是啥?࿰…...

INT8 中的稀疏性:加速的训练工作流程和NVIDIA TensorRT 最佳实践
INT8 中的稀疏性:加速的训练工作流程和NVIDIA TensorRT 最佳实践 文章目录 INT8 中的稀疏性:加速的训练工作流程和NVIDIA TensorRT 最佳实践结构稀疏量化在 TensorRT 中部署稀疏量化模型的工作流程案例研究:ResNet-34要求第 1 步:…...

隧道模式HTTP代理使用代码示例
以下是使用Python实现隧道模式HTTP代理的代码示例: python import socket def handle_client(client_socket): # 接收客户端请求 request client_socket.recv(4096) # 解析请求头,获取目标主机和端口号 host request.split(b\r\n)[1].sp…...

翻筋斗觅食海鸥优化算法-附代码
翻筋斗觅食海鸥优化算法 文章目录 翻筋斗觅食海鸥优化算法1.海鸥优化算法2. 改进海鸥优化算法2.1 非线性参数 A 策略2.2 翻筋斗觅食策略 3.实验结果4.参考文献5.Matlab代码6.python代码 摘要:针对基本海鸥优化算法(SOA)在处理复杂优化问题中存在低精度、…...

K8S常见应用场景(六)
Kubernetes 是一个可移植的、可扩展的开源平台,用于管理容器化的工作负载和服务,可促进声明式配置和自动化。 Kubernetes 拥有一个庞大且快速增长的生态系统。Kubernetes 的服务、支持和工具广泛可用。 Kubernetes 这个名字源于希腊语,意为“…...

《不抱怨的世界》随记
*不抱怨的世界 * 1.天才只有三件事:我的事,他的事,老天的事。抱怨自己的的人,应该试着学习接纳自己;抱怨他人的人,应该试着把抱怨转成请求;抱怨老天的人么,请试着用祈祷的方式来诉求…...

2.2 利用MyBatis实现CRUD操作
一、准备工作 打开MyBatisDemo项目 二、查询表记录 1、在映射器配置文件里引入结果映射元素 在UserMapper.xml文件里创建结果映射元素 将UserMapper接口里抽象方法上的注解暂时注释掉 运行TestUserMapper测试类里的testFindAll()测试方法,查看结果 2、添加…...

自动缩放Kubernetes上的Kinesis Data Streams应用程序
想要学习如何在Kubernetes上自动缩放您的Kinesis Data Streams消费者应用程序,以便节省成本并提高资源效率吗?本文提供了一个逐步指南,教您如何实现这一目标。 通过利用Kubernetes对Kinesis消费者应用程序进行自动缩放,您可以从其…...

介绍js各种事件
目录 一、点击事件 二、鼠标移动事件 三、键盘事件 四、滚轮事件 五、拖放事件 六、窗口大小改变事件 一、点击事件 点击事件是指当用户单击页面上的某个元素时触发的事件。这是最常见和基础的事件之一,也是Web应用程序中最常用的交互之一。 以下是如何使用…...

Python 将 CSV 分割成多个文件
文章目录 使用 Pandas 在 Python 中创建 CSV 文件在 Python 中将 CSV 文件拆分为多个文件根据行拆分 CSV 文件根据列拆分 CSV 文件 总结 在本文中,我们将学习如何在 Python 中将一个 CSV 文件拆分为多个文件。 我们将使用 Pandas 创建一个 CSV 文件并将其拆分为多个…...