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

第30章 汇编语言--- 性能优化技巧

汇编语言是用于直接编程计算机硬件的低级语言,它几乎是一对一地映射到机器指令。因为汇编代码与特定处理器架构紧密相关,所以在讨论性能优化技巧时,通常需要考虑具体的CPU架构和指令集。

以下是一些通用的汇编语言性能优化技巧,并结合一些伪代码来说明这些概念:

  1. 循环展开(Loop Unrolling)

    • 减少循环控制指令的数量可以提高性能。
    • 例如,如果你有一个简单的循环:
      loop_start:; do something with index iinc icmp i, limitjl loop_start
      
    • 展开后可能看起来像这样:
      loop_start:; do something with index i; do something with index i+1; do something with index i+2; do something with index i+3add i, 4cmp i, limitjl loop_start
      
  2. 使用寄存器变量(Register Variables)

    • 尽量将频繁使用的变量保持在寄存器中,以减少内存访问次数。
    • 例如:
      mov eax, [memory_address]  ; Load into register once
      ; Use eax multiple times instead of accessing memory_address each time
      
  3. 避免不必要的分支(Branch Prediction Optimization)

    • 现代CPU有复杂的分支预测机制,但错误预测会带来显著的性能损失。
    • 通过重新组织代码逻辑,可以尽量减少难以预测的分支。
    • 例如,使用条件执行或条件移动指令代替条件跳转。
  4. 数据预取(Data Prefetching)

    • 提前加载可能会用到的数据到缓存中,可以减少等待时间。
    • 某些CPU架构支持显式的预取指令:
      prefetch [data_address]
      
  5. 指令调度(Instruction Scheduling)

    • 重排指令顺序以充分利用CPU的并行处理能力,比如让非依赖性的指令尽可能靠近执行。
    • 例如,如果有一系列独立的操作,可以交错安排它们以填充延迟:
      ; Original sequence
      mov eax, [ebx]
      add ecx, edx; Reordered for better performance
      add ecx, edx    ; Non-dependent instruction first
      mov eax, [ebx]  ; Memory access can be slower
      
  6. 使用SIMD指令(Single Instruction Multiple Data)

    • 如果你的CPU支持,使用SIMD指令可以同时对多个数据点进行操作。
    • 例如,使用SSE/AVX指令集处理向量运算。
  7. 局部性原则(Locality of Reference)

    • 确保代码和数据具有良好的空间和时间局部性,以便更好地利用CPU缓存。
  8. 减少函数调用(Inlining Functions)

    • 函数调用有额外的开销,包括保存和恢复寄存器状态等。
    • 对于小且频繁调用的函数,可以考虑将其内联展开。

请注意,上述示例为简化版,实际应用时需根据具体处理器架构调整。而且,随着技术的发展,某些传统上的优化方法可能不再适用或效果减弱,因此,在实践中总是应该测量和验证优化的效果。

循环展开(Loop Unrolling)

假设我们有一个简单的循环来累加一个数组中的值:

section .dataarray dd 1,2,3,4,5,6,7,8,9,10  ; 定义一个整数数组length dd 10                   ; 数组长度section .bsssum resd 1                     ; 用于存储结果的变量section .text
global _start_start:mov ecx, [length]              ; 将数组长度加载到ecx寄存器xor eax, eax                   ; 清零eax寄存器,用于累加lea ebx, [array]               ; 加载数组地址到ebxloop_start:add eax, [ebx + ecx*4 - 4]     ; 累加当前元素到eaxloop loop_start                ; 减少ecx并跳转到循环开始处,如果ecx不为0mov [sum], eax                 ; 将结果存储到sum变量

现在我们对这个循环进行展开:

section .dataarray dd 1,2,3,4,5,6,7,8,9,10  ; 定义一个整数数组length dd 10                   ; 数组长度section .bsssum resd 1                     ; 用于存储结果的变量section .text
global _start_start:mov ecx, [length]              ; 将数组长度加载到ecx寄存器xor eax, eax                   ; 清零eax寄存器,用于累加lea ebx, [array]               ; 加载数组地址到ebxloop_start:add eax, [ebx]                 ; 累加第一个元素到eaxadd eax, [ebx + 4]             ; 累加第二个元素到eaxadd eax, [ebx + 8]             ; 累加第三个元素到eaxadd eax, [ebx + 12]            ; 累加第四个元素到eaxadd ebx, 16                    ; 移动指针到下一个元素sub ecx, 4                     ; 每次减少4个元素jnz loop_start                 ; 如果还有元素,继续循环mov [sum], eax                 ; 将结果存储到sum变量

使用寄存器变量(Register Variables)

如果我们有频繁访问的内存位置,可以将其保持在寄存器中:

; 假设我们需要频繁访问某个变量
mov eax, [variable_address]        ; 初始加载到寄存器
; 在接下来的操作中使用eax而不是直接访问内存
; ...
; 当我们完成时,再保存回内存
mov [variable_address], eax

数据预取(Data Prefetching)

对于支持数据预取指令的处理器,如Intel x86系列,你可以显式地预取数据:

prefetcht0 [data_address]          ; 预取数据到L1缓存

指令调度(Instruction Scheduling)

为了展示指令调度,让我们看一段代码,其中包含两个独立的操作:

; 原始序列
mov eax, [ebx]                     ; 内存读取可能较慢
add ecx, edx                       ; 这条指令不需要等待上面的读取完成; 调度后的序列
add ecx, edx                       ; 先执行这条快速指令
mov eax, [ebx]                     ; 内存读取可以在后台进行

以上只是几个例子,汇编语言的优化可以非常复杂,涉及到对硬件行为的深刻理解以及对特定程序逻辑的细致分析。每个优化决策都应基于实测数据,因为并非所有优化都能带来性能提升,有时甚至可能导致性能下降。

避免不必要的分支(Branch Prediction Optimization)

现代处理器通过预测分支方向来提高执行效率。如果分支预测错误,流水线需要清空并重新填充,这会带来性能损失。因此,尽量减少难以预测的分支或者用条件执行代替分支可以提高性能。

使用条件移动指令

假设我们有一个简单的最大值计算:

section .dataa dd 10b dd 20section .bssmax resd 1section .text
global _start_start:mov eax, [a]cmp eax, [b]       ; 比较a和bjge greater        ; 如果a >= b, 跳转到greater标签mov eax, [b]       ; 否则,将b的值赋给eax
greater:mov [max], eax     ; 将较大值存储在max变量中

我们可以使用条件移动指令(如cmovg)来避免显式的分支跳转:

section .dataa dd 10b dd 20section .bssmax resd 1section .text
global _start_start:mov eax, [a]mov ebx, [b]cmovg eax, ebx     ; 如果ebx > eax, 则eax = ebxmov [max], eax     ; 将较大值存储在max变量中

使用SIMD指令(Single Instruction Multiple Data)

SIMD允许我们在一个指令周期内对多个数据点进行操作,这对于向量运算或多媒体处理非常有用。这里以SSE2指令集为例,演示如何同时对两个32位整数求和:

section .datavector_a dd 1, 2, 3, 4          ; 定义两个向量vector_b dd 5, 6, 7, 8result dd 0, 0, 0, 0            ; 存储结果的向量section .text
global _start_start:movups xmm0, [vector_a]         ; 加载vector_a到xmm0寄存器movups xmm1, [vector_b]         ; 加载vector_b到xmm1寄存器addps xmm0, xmm1                ; 对两个向量中的元素求和movups [result], xmm0           ; 将结果存储回内存

请注意,addps用于浮点加法,对于整数加法应该使用paddd指令。另外,movups是未对齐的数据加载指令;如果你的数据是16字节对齐的,那么可以使用更高效的movaps指令。

函数调用内联(Inlining Functions)

函数调用有额外的开销,包括保存和恢复寄存器状态等。对于小且频繁调用的函数,可以考虑将其展开为内联代码,以减少这些开销。例如,假设有一个简单函数inc,它只增加一个值:

; 原始版本,包含函数调用
call inc_function                   ; 调用函数
inc_function:inc eax                         ; 增加eaxret                             ; 返回调用者; 内联版本
inc eax                             ; 直接在原地增加eax

在实际的程序中,你可能不会直接编写这样的汇编代码,而是依赖于编译器自动优化。然而,在写汇编时,你可以选择不使用函数调用而是在需要的地方直接编写相应的指令。

性能测量与验证

无论应用哪种优化技术,最重要的是要测量和验证其效果。可以通过分析工具(如性能计数器)来监控CPU周期、缓存命中率、分支预测准确性等指标,确保你的优化确实带来了预期的效果。此外,不同的输入数据可能会导致不同的行为,所以测试应覆盖各种可能的况。

内存访问模式优化

1. 数据对齐(Data Alignment)

确保数据在内存中的对齐可以显著提高访问速度。大多数处理器对齐的数据访问更快,因为它们可以在单个总线周期内读取或写入数据。

section .data; 确保vector是16字节对齐的align 16vector dd 1, 2, 3, 4            ; 定义一个向量section .text
global _start_start:movaps xmm0, [vector]           ; 使用对齐指令加载向量到xmm0寄存器
2. 减少缓存未命中(Cache Miss Reduction)

尝试将频繁访问的数据放置在一起,以利用缓存行的优势。此外,避免使用过大的结构体或数组,因为它们可能导致缓存污染。

; 假设我们有一个频繁访问的变量和一个偶尔访问的大数组
section .datafrequently_used dd 0            ; 频繁访问的小变量; 其他代码...; 将大数组放在不同的部分,避免缓存冲突
section .bsslarge_array resd 1024           ; 较大的数组section .text
global _start_start:; 访问frequently_usedmov eax, [frequently_used]; ...; 在需要时才访问large_arraymov ebx, [large_array + ecx*4]

并行处理

1. 多线程编程(Multithreading)

虽然汇编语言不是多线程编程的最佳选择,但你可以编写支持多线程的代码。这通常涉及到操作系统API调用或者使用特定的库函数来创建和管理线程。

2. 超线程(Hyper-Threading)和多核(Multi-Core)

如果你的目标平台支持超线程或多核处理,尽量设计你的算法,使得不同线程或进程可以独立工作而不相互干扰。

利用现代CPU特性

1. AVX-512 指令集

对于最新的Intel CPU,可以考虑使用AVX-512指令集,它提供了512位宽的寄存器,能够一次性处理更多数据点。

section .data; 确保vector是64字节对齐的align 64vector_zmm dq 8                 ; 定义一个适合ZMM寄存器的向量section .text
global _start_start:vmovups zmm0, [vector_zmm]      ; 加载向量到zmm0寄存器vaddpd zmm0, zmm0, [vector_zmm] ; 对两个向量中的元素求和(双精度浮点)
2. FMA (Fused Multiply-Add) 指令

FMA指令可以在单个时钟周期内完成乘法和加法操作,这对于数值计算密集型应用非常有用。

section .dataa dq 2.0                        ; 定义三个双精度浮点数b dq 3.0c dq 4.0section .text
global _start_start:vbroadcastsd ymm0, [a]          ; 广播a到YMM寄存器的所有元素vbroadcastsd ymm1, [b]vbroadcastsd ymm2, [c]vfmadd213pd ymm0, ymm1, ymm2    ; 执行a * b + c的操作

性能测量与验证

为了确保你的优化确实有效,必须进行性能测量。可以使用性能分析工具(如Linux下的perf命令)来收集关于CPU周期、缓存命中率、分支预测准确性等信息。

# 使用 perf 工具进行性能分析
perf stat ./your_program

也可以编写简单的基准测试代码来比较优化前后的执行时间:

section .datastart_time dq 0end_time   dq 0section .text
global _start_start:; 获取开始时间rdtscmov [start_time], eax; ... 运行待测代码 ...; 获取结束时间rdtscmov [end_time], eax; 计算差值sub eax, [start_time]; eax 现在包含执行时间

请记住,实际的性能提升取决于具体的硬件环境和应用程序的工作负载。因此,总是应该基于实测数据来调整和验证你的优化策略。

相关文章:

第30章 汇编语言--- 性能优化技巧

汇编语言是用于直接编程计算机硬件的低级语言,它几乎是一对一地映射到机器指令。因为汇编代码与特定处理器架构紧密相关,所以在讨论性能优化技巧时,通常需要考虑具体的CPU架构和指令集。 以下是一些通用的汇编语言性能优化技巧,并…...

HTB:Paper[WriteUP]

目录 连接至HTB服务器并启动靶机 信息收集 使用rustscan对靶机TCP端口进行开放扫描 将靶机TCP开放端口号提取并保存 使用nmap对靶机TCP开放端口进行脚本、服务扫描 使用nmap对靶机TCP开放端口进行漏洞、系统扫描 使用nmap对靶机常用UDP端口进行开放扫描 对靶机进行子域…...

数据库中的 DDL、DML 和 DCL

数据库中的 DDL、DML 和 DCL 在数据库的定义与操作中,DDL、DML 和 DCL 是三个核心概念,分别用于不同层面的数据库管理与操作。 1. DDL(Data Definition Language) - 数据定义语言 定义 DDL 用于定义和管理数据库的结构或模式。…...

OKR 极简史及理解

大家读完觉得有帮助记得点赞和关注!!! 目录 MBO SMART 和 KPI OKR 1. 什么是 OKR? 1.1 Objectives(目标) 1.2 Key Results(关键成果) KR 应当是困难的,但并非不可…...

电商项目-基于ElasticSearch实现商品搜索功能(四)

一、 高亮显示 1.1 高亮分析 高亮显示是指根据商品关键字搜索商品的时候,显示的页面对关键字给定了特殊样式,让它显示更加突出,如商品搜索中,关键字变成了红色,其实就是给定了红色样式。 1.2 高亮搜索实现步骤解析 …...

TCP封装数据帧

void *send_data(void *arg) //这是一个发送数据的线程 {int sockfd init_tcp_cli("192.168.0.148",50000) //传ip和port&#xff0c;port 50000是因为大概前五万都被其它服务所占用&#xff0c;50000后是私人ipif(sockfd < 0){return NULL;}unsigned char …...

数据结构与算法之二叉树: LeetCode 515. 在每个树行中找最大值 (Ts版)

在每个树行中找最大值 https://leetcode.cn/problems/find-largest-value-in-each-tree-row/description/ 描述 给定一棵二叉树的根节点 root &#xff0c;请找出该二叉树中每一层的最大值 示例1 输入: root [1,3,2,5,3,null,9] 输出: [1,3,9]示例2 输入: root [1,2,3]…...

百度视频搜索架构演进

导读 随着信息技术的迅猛发展&#xff0c;搜索引擎作为人们获取信息的主要途径&#xff0c;其背后的技术架构也在不断演进。本文详细阐述了近年来视频搜索排序框架的重大变革&#xff0c;特别是在大模型技术需求驱动下&#xff0c;如何从传统的多阶段级联框架逐步演变为更加高…...

构造函数的原型原型链

代码示例 // 定义一个构造函数 Test function Test() {this.name 张三 }; //向构造函数的原型添加一个属性 age18 Test.prototype.age 18;//使用构造函数 Test 来实例化一个新对象 const test new Test();//向 Object.prototype 添加了一个名为 sex 的属性&#xff0c;其值…...

nginx反向代理及负载均衡

华子目录 nginx反向代理功能http反向代理反向代理配置参数proxy_pass的注意事项案例&#xff1a;反向代理单台后端服务器案例&#xff1a;反向代理实现动静分离案例&#xff1a;反向代理的缓存功能非缓存场景下测压准备缓存缓存场景下测压验证缓存文件 反向代理负载均衡&#x…...

单片机实物成品-011 火灾监测

火灾监测&#xff08;20个版本&#xff09; 版本20&#xff1a; oled显示温湿度烟雾浓度火焰传感器天然气浓度窗户风扇水泵排气系统声光报警语音播报按键WIFI模块 ----------------------------------------------------------------------------- https://www.bilibili.com…...

使用 Docker 在 Alpine Linux 下部署 Caddy 服务器

简介 在现代 web 开发中&#xff0c;选择合适的 web 服务器至关重要。Caddy 是一个功能强大的现代化 HTTP/2 服务器&#xff0c;支持自动 HTTPS&#xff0c;配置简单&#xff0c;适合开发和生产环境。Docker 则为我们提供了一种轻量级的容器化技术&#xff0c;使得应用程序的部…...

每日十题八股-2025年1月12日

1.为什么四次挥手之后要等2MSL? 2.服务端出现大量的timewait有哪些原因? 3.TCP和UDP区别是什么&#xff1f; 4.TCP为什么可靠传输 5.怎么用udp实现http&#xff1f; 6.tcp粘包怎么解决&#xff1f; 7.TCP的拥塞控制介绍一下&#xff1f; 8.描述一下打开百度首页后发生的网络过…...

Python中定位包含特定文本信息的元素

目录 一、为什么需要定位包含文本信息的元素 二、使用Selenium定位包含文本的元素 1. 使用find_element_by_link_text 2. 使用find_element_by_partial_link_text 3. 使用XPath定位包含文本的元素 4. 使用CSS选择器定位包含文本的元素 三、使用BeautifulSoup定位包含文本…...

uniapp实现H5页面内容居中与两边留白,打造类似微信公众号阅读体验

在 UniApp 中&#xff0c;由于需要兼容多端应用&#xff0c;我们通常使用 rpx 作为尺寸单位。然而&#xff0c;在某些情况下&#xff0c;如需要实现内容居中且两边留白时&#xff0c;直接使用 rpx 可能会带来一些限制。这时&#xff0c;我们可以考虑使用 px 或 rem 等单位&…...

极品飞车6里的赛道简介

极品飞车里有很多赛道,赛道分为前向赛道Forward、后向赛道Backward。前向赛道Forward是从A点到B点;后向赛道Backward是前向赛道的逆过程,即从B点到A点。这里介绍极品飞车6的赛道长度、中英文名称翻译、难度等级。 序号赛道英文名赛道中文名总长(km)急弯难度等级1Alpine Trai…...

SAP推出云端ERP解决方案,加速零售行业数字化转型

2025年1月9日&#xff0c;SAP发布了一款专为零售行业设计的云端ERP行业解决方案——S/4HANA Cloud Public Edition&#xff0c;进一步推动企业向云端迁移。这款解决方案旨在集中运营数据&#xff0c;整合财务、采购和商品管理流程&#xff0c;以帮助零售企业优化运营效率。 核…...

Python爬虫进阶——案例:模拟bilibili登录)

主要内容&#xff1a;模拟bilibili账号密码登录&#xff0c;不要实现的的实现功能是单击登录按钮&#xff0c;切换登录方式&#xff0c; 输入账号和密码&#xff0c;然后完成图片点击验证&#xff0c;最后单击立即登录按钮。 1、第一步&#xff1a;通过selenium模块访问bilibi…...

什么是数据分析?

什么是数据分析&#xff1f; 数据分析&#xff08;Data Analysis&#xff09;是指通过对数据进行收集、整理、处理、建模和解读&#xff0c;以揭示数据中的有用信息、支持决策和解决实际问题的过程。它是一门将数据转化为知识的学科&#xff0c;广泛应用于商业、科学研究、医疗…...

基于springboot的课程作业管理系统源码(springboot+vue+mysql)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于springboot的课程作业管理系统。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 可以管理首页、个人中心…...

多线程之旅:属性及其基本操作

上次分享到了&#xff0c;多线程中是是如何创建的&#xff0c;那么接下来&#xff0c;小编继续分享下多线程的相关知识。 多线程中的一些基本属性。 基本属性 属性获取方法IDgetId()名称getName()状态getState()优先级getPriority()是否后台线程isDemo()是否存活isAlive()是…...

数据表中的数据插入、更新和删除

文章目录 一、表的插入二、更新表中的数据记录三、删除表中的数据记录 一、表的插入 插入数据记录是常见的数据操作&#xff0c;可以显示向表中增加的新的数据记录。在MySQL中可以通过“INSERT INTO”语句来实现插入数据记录&#xff0c;该SQL语句可以通过如下4种方式使用&…...

Q_OBJECT宏报错的问题

在Qt中继承QObject&#xff0c;并且加上Q_OBJECT宏&#xff0c;有时候会报错&#xff0c;比如我的错误&#xff1a; error: debug/httpmgr.o:httpmgr.cpp:(.rdata$.refptr._ZTV7HttpMgr[.refptr._ZTV7HttpMgr]0x0): undefined reference to vtable for HttpMgr 意思是没有虚…...

提升性能300ms:深入解析Spring多表联接查询优化与SQL调优实战

优化所需知识点&#xff08;必须掌握&#xff09; 索引篇 explain命令 重点&#xff1a;这是后续分析是否使用索引以及使用是否恰当的工具 作用&#xff1a;查看sql的执行计划&#xff0c;可以看sql语句是否使用了索引&#xff0c;索引的使用情况&#xff0c;以及sql的性能。 …...

增量导入和全量导入的区别是什么?

定义 全量导入&#xff1a;是指将数据源中的所有数据一次性全部导入到目标系统中。例如&#xff0c;一个电商公司要将其旧数据库中的所有商品信息&#xff08;包括商品名称、价格、库存等&#xff09;全部迁移到新的数据库系统中&#xff0c;这个过程就是全量导入。这种方式会覆…...

【百度智能云客悦智能客服】搭建AI agent智能对话 - 购车推荐

前期准备 平台链接&#xff1a;https://keyue.cloud.baidu.com/ 一、开始创建 二、会话流程配置 我们以购车推荐的案例&#xff0c;来进行 AI agent 配置演示 1.添加开场白 在 起始主题 画布中&#xff0c;我们可以配置 AI agent 的开场白&#xff0c;画布左侧默认有 开始 …...

【HTML+CSS+JS+VUE】web前端教程-3-标题标签

标题介绍与应用 标题是通过<h1>-<h6>标签进行定义的 <h1>定义最大的标题 <h6>定义最小的标题<h1...

逐笔成交逐笔委托Level2高频数据下载和分析:20250102

level2逐笔成交逐笔委托下载 链接: https://pan.baidu.com/s/1p7OOj5p-QGFrWkt6KKoYng?pwd7f4g 提取码: 7f4g Level2逐笔成交逐笔委托数据分享下载 通过Level2逐笔成交和逐笔委托这种每一笔的毫秒级别的数据可以分析出很多有用的点&#xff0c;包括主力意图&#xff0c;虚假动…...

JavaEE之线程池

前面我们了解了多个任务可以通过创建多个线程去处理&#xff0c;达到节约时间的效果&#xff0c;但是每一次的线程创建和销毁也是会消耗计算机资源的&#xff0c;那么我们是否可以将线程进阶一下&#xff0c;让消耗计算机的资源尽可能缩小呢&#xff1f;线程池可以达到此效果&a…...

java 中 main 方法使用 KafkaConsumer 拉取 kafka 消息如何禁止输出 debug 日志

pom 依赖&#xff1a; <dependency><groupId>org.springframework.kafka</groupId><artifactId>spring-kafka</artifactId><version>2.5.14.RELEASE</version> </dependency> 或者 <dependency><groupId>org.ap…...

app在线开发制作/排名优化关键词公司

环境:centOS 6.5 X86 64位 nagios-4.08 步骤&#xff1a; 1、 最小化安装系统 2、 修改安全特性 关闭SELINUX SELINUXdisabled 清除iptables防火墙规则&#xff0c;开机不启动该服务 3、 安装所必须的软件 yum install gcc mysql httpd php gd openssl openssl-devel my…...

wordpress 每日替换/网站优化推广培训

关注、星标嵌入式客栈&#xff0c;精彩及时送达[导读] 你是否被要求写的代码需要0 erros, 0 warnings? 或者你的项目是否需要做静态代码检测&#xff1f;0 erros是一定要做到&#xff0c;而0 wanring有时候会让你抓狂。前面转了一篇数组内包含头文件的写法&#xff0c;看起来对…...

诸城网站制作/朋友圈广告投放价格表

能不能在flash动画中给asp传递变量&#xff1f;faq-it.org/vb/---------------------------------------------------------------数据库结构&#xff1a;第一步要做的是建立数据库.例子中使用Access数据库&#xff0c;表名为tblStaff有三个字段&#xff1a;strID (自动编号),…...

做网站开发多少钱/湖南网站建设推荐

阿里 阿里的开源项目很多&#xff0c;这也跟淘宝正明的开源态度密不可分。有很多重量级的项目&#xff0c;例如LVS、Tengine&#xff0c;或者很有实践价值的中间件&#xff0c;例如 MetaQ(分布式消息系统)、dubbo(RPC框架)、cobar(数据库中间件)&#xff0c;或者是Java世界的工…...

北京软件研发公司/牛排seo系统

在網上搜了很多的關於表格排序的Js代碼&#xff0c;看起來相當晦澀&#xff0c;爲了以後便於查找&#xff0c;方便自己也方便大家。 1.对单列的表格排序&#xff1a; <html> <head> <title>Table Sort Example</title> <script type"…...

做网站的故意给中病毒/软文推广的优点

文章目录MyBatis-Plus特性快速开始项目整体目录结构数据库准备配置文件简单CRUD通用service源码下载MyBatis-Plus 官网 个人白话解释&#xff1a;简单的CRUD直接通过方法调用&#xff0c;无需多写接口和xml&#xff0c;像jpa那样调用 愿景 我们的愿景是成为 MyBatis 最好的搭档…...