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

Lora原理及实现浅析

Lora

什么是Lora

Lora的原始论文为《LoRA: Low-Rank Adaptation of Large Language Models》,翻译为中文为“大语言模型的低秩自适应”。最初是为了解决大型语言模在进行任务特定微调时消耗大量资源的问题;随后也用在了Diffusion等领域,用于对模型进行微调。总而言之,Lora是一种微调技术。

Lora是怎么实现的

原理

Lora的思想很简单,在原有权重的旁边加一个的分支。训练时冻结原有权重,只训练这个分支。然后将分支输出的结果与原有权重的输出结果相加即可。这么做的理论依据是:预训练模型拥有极小的内在维度(instrisic dimension),即存在一个极低维度的参数,微调它和在全参数空间中微调能起到相同的效果

Lora简要示意图

伪代码实现

假设我们有一个简单的模型,仅包含一个线性层:

import torch
import torch.nn as nn# 原始 SimpleModel 定义
class SimpleModel(nn.Module):def __init__(self):super().__init__()self.linear = nn.Linear(100, 50)  # 原始权重形状: (50, 100)def forward(self, x):return self.linear(x)

那么我们用最原始的手段添加Lora:

import torch
import torch.nn as nnclass SimpleModelWithLoRA(nn.Module):def __init__(self, rank=4):super().__init__()# 原始预训练层(冻结)self.linear = nn.Linear(100, 50)# 冻结原始参数for param in self.linear.parameters():param.requires_grad = False# LoRA 参数self.lora_A = nn.Parameter(torch.randn(50, rank))   # (out_dim, rank)self.lora_B = nn.Parameter(torch.randn(rank, 100))  # (rank, in_dim)def forward(self, x):# 原始输出original_output = self.linear(x)# LoRA 分支输出lora_output = x @ self.lora_B.t() @ self.lora_A.t()  # [batch, 100] -> [batch, 50]# 输出为两者的和return original_output + lora_output

可以看到,Lora的代码跟原始代码的主要区别仅在于:

在权重层面:

  1. 冻结原始参数
  2. 添加了两个线性层,分别用于将输出映射到rank维的中间结果,和将中间结果映射到输出

在推理层面:

  1. 计算Lora的输出结果
  2. 将两个计算结果相加

这就是Lora的实际计算过程了,就是在原有模型旁边加了个轻量化的分支而已。此外,由于矩阵运算的性质,我们还可以直接把训练好的权重加到原始权重上以提高计算效率:

import torch
import torch.nn as nn# 应用了 LoRA 的 SimpleModel 定义
class SimpleModelWithLoRA(nn.Module):def __init__(self, rank=4):super().__init__()# 初始化原始线性层self.linear = nn.Linear(100, 50)  # 原始权重形状: (50, 100)# 冻结原始线性层的参数for param in self.linear.parameters():param.requires_grad = False# 添加 LoRA 矩阵 A 和 Bself.A = nn.Parameter(torch.randn(50, rank))  # 形状: (50, rank)self.B = nn.Parameter(torch.randn(rank, 100)) # 形状: (rank, 100)def forward(self, x):# 计算 LoRA 修正项lora_term = self.A @ self.B  # 形状: (50, 100)# 将 LoRA 修正项加到原始权重上adapted_weight = self.linear.weight.data + lora_term# 使用适应后的权重进行计算output = nn.functional.linear(x, adapted_weight, self.linear.bias)return output

值得注意的是,是否将Lora权重与原始权重融合取决于是否需要更新lora权重。因为融合后的权重虽然能够提高计算效率,但是无法单独更新Lora了。

其他细节说明

其他微调方法

除了Lora,还有其他微调手段,下面简单列出,不再做进一步说明:

方法描述参数规模灵活性应用场景
全量微调(Full Fine-tuning)修改全部参数所有任务
Adapter Tuning在层之间插入小型神经网络模块模型嵌套结构支持时
Prefix Tuning / Prompt Tuning修改输入提示向量极小NLP
LoRA引入低秩矩阵进行参数增量更新广泛适用

rank选择与初始化

rank 越大:模型表达能力越强,但消耗资源越多。

rank 越小:更轻量,但可能无法捕捉复杂的任务特征。

通常尝试 r=8, r=16, r=64 等值,根据验证集效果选择最优配置。

在原始的 LoRA 论文中A和两个矩阵通常是随机初始化。但为了确保刚开始微调的时候不对模型输出产生影响,一些方法会其中一个矩阵初始化为零。

Q&A

Q: LoRA 是否会影响模型推理速度?

A: 不会显著影响,融合权重后模型层面没有增加任何多余的计算量

Q: LoRA 是否只能用于语言模型?

A: 不是,也可以用于图像模型(如 Diffusion)、语音模型等。

Q: 我可以用多个 LoRA 吗?

A: 可以,有些框架支持加载多个 LoRA,并按需组合使用。

http://www.xdnf.cn/news/5299.html

相关文章:

  • 【C++】特殊类设计
  • 支持向量机与逻辑回归的区别及 SVM 在图像分类中的应用
  • matlab中的积分函数
  • 【Java学习日记34】:this关键字和成员变量
  • armv7 backtrace
  • LoRA(Low-Rank Adaptation)原理详解
  • 【ajax基础】
  • 深入理解深度Q网络DQN:基于python从零实现
  • OB Cloud 云数据库V4.3:SQL +AI全新体验
  • redis主从同步于对象模型
  • 【基于 LangChain 的异步天气查询2】GeoNames实现地区实时气温查询
  • EDITPLUS配置CTags实现函数跳转
  • 技术方案模型需要兼顾战略规划、技术实现与落地可行性
  • 《操作系统真象还原》第十三章——编写硬盘驱动程序
  • SQL注入问题
  • powerbuilder9.0中文版
  • 7、系统开发
  • 计算机网络 4-2-1 网络层(IPv4)
  • 每日算法-250510
  • 深入理解Embedding技术-什么是Embedding?
  • 使用fdisk 、gdisk管理分区
  • Satori:元动作 + 内建搜索机制,让大模型实现超级推理能力
  • python:ASCII-generator 实用教程
  • $\int_{0}^{1} x \arcsin \sqrt{4x - 4x^2}dx$
  • LintCode第366题-斐波那契数列
  • 各种环境测试
  • 解释器和基于规则的系统比较
  • 【Linux基础】文件和目录管理指令
  • 对日开发 TeraTerm ttl脚本开发环境配置
  • python04——条件判断(选择结构)