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

大语言模型LLM的微调代码详解

代码的摘要说明

一、整体功能概述
这段 Python 代码主要实现了基于 Hugging Face Transformers 库对预训练语言模型(具体为 TAIDE-LX-7B-Chat 模型)进行微调(Fine-tuning)的功能,使其能更好地应用于生成唐诗相关内容的任务。整个流程涵盖了数据加载与预处理、模型配置、模型训练以及训练后模型的测试与结果保存等环节。

二、代码各部分详细说明
导入模块部分
导入了多个常用的 Python 库,如 os、sys、argparse 等基础库,以及和深度学习、自然语言处理相关的库,像 torch、transformers、peft 等,还导入了用于忽略警告信息、文本颜色控制等功能的相关模块,为后续代码实现做准备。
基础配置与参数设置部分
模型相关路径:定义了模型名称、模型所在目录、工作目录、输出目录以及检查点目录等路径相关的变量,确保后续文件操作(如加载模型、保存训练结果等)能找到正确的位置。
训练参数设置:设定了训练的总轮数(num_epoch)、学习率(LEARNING_RATE),还包括一系列影响模型训练和推理的超参数,例如用于训练的数据集数量(num_train_data)、文本截断长度(CUTOFF_LEN)、LORA 相关参数(LORA_R、LORA_ALPHA、LORA_DROPOUT)、验证集大小(VAL_SET_SIZE)、微批次大小(MICRO_BATCH_SIZE)、梯度累积步数(GRADIENT_ACCUMULATION_STEPS)等,这些参数将用于控制模型训练过程和微调效果。
分布式训练配置:根据环境变量 WORLD_SIZE 判断是否进行分布式训练,若需要则配置相应的设备映射(device_map)。
数据加载与预处理部分
创建目录:首先确保输出目录和检查点目录存在,若不存在则创建它们,用于后续保存训练结果和模型检查点。
加载数据集:从指定的 json 文件路径(dataset_dir)读取原始数据集,先将其加载为 Python 的字典数据结构(通过 json.load),然后从中截取指定数量(由 num_train_data 决定)的数据保存为临时数据集文件(tmp_dataset_path),再通过 load_dataset 函数以 json 格式重新加载这个临时数据集用于后续处理。
数据处理函数定义:定义了 generate_training_data 函数,用于将包含指令、输入和输出文本的数据点转换为适合模型训练的格式,具体是通过 tokenizer 将文本转换为令牌(token)形式,并处理好输入令牌、对应标签以及注意力掩码等信息,构建训练样本。
划分数据集(可选):根据 VAL_SET_SIZE 参数的值决定是否划分训练集和验证集,如果 VAL_SET_SIZE 大于 0,则从原始训练集中划分出验证集,对训练集和验证集的数据都应用 generate_training_data 函数进行预处理,使其格式符合模型训练要求;若 VAL_SET_SIZE 为 0,则整个数据集都作为训练集进行处理。
模型相关配置部分
创建模型和令牌器:从指定的模型路径(model_path)加载预训练的语言模型,并根据相关配置(如量化配置 BitsAndBytesConfig 用于以特定量化方式加载模型以节省内存等)进行初始化;同时加载对应的分词器(tokenizer),并设置其填充令牌(pad_token)与结束令牌(eos_token)相同,方便后续文本处理。
模型准备与微调配置:将加载的模型通过 prepare_model_for_int8_training 函数进行处理,使其适合 INT8 训练方式,接着使用 LoraConfig 配置 LORA(低秩适配,一种参数高效微调方法)相关参数,并通过 get_peft_model 函数应用 LORA 配置到模型上,以便后续进行参数高效的微调训练。
生成配置定义:定义了 GenerationConfig 对象,用于指定模型在生成文本时的一些解码参数,例如是否采样(do_sample)、温度参数(temperature)、束搜索的束数量(num_beams)等,这些参数会影响模型生成唐诗内容时的质量和多样性等特性。
模型训练部分
使用 transformers.Trainer 类来实例化训练器对象(trainer),传入模型、训练数据集、验证数据集(可为 None)以及训练相关的各种参数(如批次大小、学习率、训练轮数、梯度累积步数、日志记录和模型保存相关的配置等),同时指定了数据整理器(data_collator)用于整理训练数据的格式。在训练前还禁用了模型的缓存功能(model.config.use_cache = False),然后通过 try-except 语句块尝试执行模型训练过程,若训练出现异常则打印出错误信息。
模型保存部分
同样使用 try-except 语句块尝试将训练后的模型保存到指定的检查点目录(ckpt_dir)中,若保存过程出现异常则打印相应的错误提示。
模型测试与结果保存部分
从指定的测试数据文件(test_data_path)中加载测试数据集(通过 json.load),然后循环遍历每条测试数据,调用 evaluate 函数(该函数基于给定的指令、生成配置和输入等信息,利用已训练好的模型生成相应回复内容)获取模型针对测试数据的回复,将测试数据的输入以及模型生成的回复拼接后写入到指定的结果文件(output_path)中保存起来,同时也会打印出来方便查看,用于评估模型在测试集上的表现。

                        
原文链接:https://blog.csdn.net/chenchihwen/article/details/144000079

整体代码参照这篇文章

《生成式 AI》课程 作业6 大语言模型(LLM)的训练微调 Fine Tuning -- part2-CSDN博客文章浏览阅读872次,点赞26次,收藏8次。代码围绕一个主工作目录展开,在这个主工作目录下包含了多个子目录和相关文件,用于存放不同阶段的数据、模型以及输出结果等内容,各个部分分工明确,以支持整个预训练语言模型微调及测试的流程。这段 Python 代码主要实现了基于 Hugging Face Transformers 库对预训练语言模型(具体为模型)进行微调(Fine-tuning)的功能,使其能更好地应用于生成唐诗相关内容的任务。整个流程涵盖了数据加载与预处理、模型配置、模型训练以及训练后模型的测试与结果保存等环节。https://blog.csdn.net/chenchihwen/article/details/144000079?spm=1001.2014.3001.5501拆解说明如下

导入必要的库

import os
import sys
import argparse
import json
import warnings
import logging

  • 导入了多个 Python 标准库,os用于操作系统相关的操作(如文件路径处理、目录创建等),sys用于处理 Python 运行时环境相关的配置和参数,argparse常用于命令行参数解析(但此代码中未体现其使用),json用于处理 JSON 格式的数据读写,warnings用于控制警告信息的显示,logging用于记录日志信息。

# 忽略警告信息
warnings.filterwarnings("ignore")

  • 配置warnings模块,使其忽略所有警告信息,避免在程序运行过程中输出大量警告干扰正常的输出和分析。

import torch
import torch.nn as nn
import bitsandbytes as bnb
from datasets import load_dataset, load_from_disk
import transformers
from peft import PeftModel
from colorama import Fore, Back, Style

  • 导入了torch(深度学习常用的张量计算库)及其nn模块(用于构建神经网络),bitsandbytes(可能用于模型量化等相关功能),load_datasetload_from_disk函数用于加载数据集,transformers库(用于自然语言处理相关的预训练模型、工具等),PeftModel(可能是某种特定的模型扩展相关类),colorama(用于在终端输出添加颜色等样式,不过此处代码后续未看到其使用)。

# 以下是根据Hugging Face Transformers库的相关功能进行的设置
from transformers import AutoTokenizer, AutoConfig, AutoModelForCausalLM, BitsAndBytesConfig
from transformers import GenerationConfig
from peft import (prepare_model_for_int8_training,LoraConfig,get_peft_model,get_peft_model_state_dict,prepare_model_for_kbit_training
)

  • transformers库中导入了多个用于创建模型、配置模型以及处理特定训练相关设置的类和函数,比如AutoTokenizer用于自动加载合适的文本令牌化器,AutoConfig用于自动加载模型配置,AutoModelForCausalLM用于加载适用于因果语言建模的预训练模型,BitsAndBytesConfig用于配置模型量化相关的参数;从peft模块导入的函数主要用于对模型进行特定的微调(如int8训练准备、配置LORA相关参数、获取基于LORA配置后的模型等)。

模型和路径相关的参数设置

# 模型名称
model_name = "TAIDE-LX-7B-Chat"
# 模型所在目录
model_path = os.path.join(os.path.dirname(os.path.abspath(__file__)), "TAIDE-LX-7B-Chat")

  • 定义了模型的名称以及模型所在的本地文件系统路径,通过os.path.join结合当前脚本文件所在目录(os.path.dirname(os.path.abspath(__file__))获取)来确定完整的模型路径。

# 设置工作目录
work_dir = os.path.dirname(os.path.abspath(__file__))
# 输出目录
output_dir = os.path.join(work_dir, "output")
# 检查点目录
ckpt_dir = os.path.join(work_dir, "checkpoint")

  • 确定了工作目录(即当前脚本所在目录),并基于此设置了后续用于存储输出结果的目录和保存模型检查点的目录路径。

# 训练的总Epoch数
num_epoch = 1
# 学习率
LEARNING_RATE = 3e-4

  • 定义了模型训练的关键超参数,总训练轮数为1轮,学习率设置为3e-4,学习率决定了模型在训练过程中参数更新的步长大小。

# 配置模型和训练参数
cache_dir = os.path.join(work_dir, "cache")
from_ckpt = False
ckpt_name = None

  • 设置了模型缓存目录路径,以及两个与是否从检查点加载模型、具体检查点名称相关的变量,这里表示不从检查点加载(from_ckptFalse)且没有指定具体检查点名称(ckpt_nameNone)。

# 以下是补充的超参数设置
# 用于训练的数据集数量
num_train_data = 1040  # 可设置的最大值为5000,数据量越多,模型性能可能越好,但训练时间也会增加
# 以下参数影响模型的训练和推理
CUTOFF_LEN = 256  # 文本截断的最大长度
LORA_R = 8  # LORA的R值
LORA_ALPHA = 16  # LORA的Alpha值
LORA_DROPOUT = 0.05  # LORA的Dropout率
VAL_SET_SIZE = 0  # 验证集的大小,0表示不使用验证集
TARGET_MODULES = ["q_proj", "up_proj", "o_proj", "k_proj", "down_proj", "gate_proj", "v_proj"]  # 目标模块,用于LORA训练
#
MICRO_BATCH_SIZE = 4  # 微批次大小,这里设置为和原示例类似的值,你可根据实际情况调整
GRADIENT_ACCUMULATION_STEPS = 4  # 计算每个微批次累积的梯度步数,示例中为16//4,这里假设为4,可按需改
logging_steps = 20  # 定義訓練過程中每隔多少步驟輸出一次訓練誌
save_steps = 65  # 定義訓練過程中每隔多少步驟保存一次模型
save_total_limit = 3  # 控制最多保留幾個模型checkpoint
report_to = ["tensorboard"]  # 設定上報實驗指標的目標,預設為無 # 可以根据需求调整报告的对象,比如添加其他日志记录工具等

  • 这些都是详细的模型训练相关超参数设置:
    • num_train_data指定了用于训练模型的数据集样本数量,这里限制为1040个,更多数据理论上有助于提升模型性能但会增加训练时长。
    • CUTOFF_LEN用于限制输入文本的最大长度,超过这个长度会截断,有助于控制输入规模。
    • LORA_RLORA_ALPHALORA_DROPOUTLORA(低秩适应)方法相关的参数,用于调整模型微调时的结构和特性。
    • VAL_SET_SIZE决定是否划分出验证集以及验证集的大小,这里设置为0表示不划分验证集用于评估训练过程中的模型性能。
    • TARGET_MODULES明确了在LORA训练时针对模型的哪些模块进行操作。
    • MICRO_BATCH_SIZEGRADIENT_ACCUMULATION_STEPS一起用于控制每次训练时实际处理的数据量和梯度更新的频率。
    • logging_steps规定了每隔多少训练步骤记录一次训练日志,save_steps定义了每隔多少步骤保存一次模型检查点,save_total_limit控制最多保留的检查点数量,report_to指定了向哪些目标(如tensorboard)上报实验指标用于可视化等分析。

分布式训练相关配置

world_size = int(os.environ.get("WORLD_SIZE", 1))
ddp = world_size!= 1
if ddp:device_map = {"": int(os.environ.get("LOCAL_RANK") or 0)}

  • 首先获取环境变量WORLD_SIZE(通常用于分布式训练,表示总的进程数),如果获取不到则默认值为1。根据world_size是否等于1来判断是否是分布式训练(ddp变量标记),如果是分布式训练(ddpTrue),则通过环境变量LOCAL_RANK(通常表示当前进程在本地的序号)来配置设备映射(这里简单地将其映射到一个整数,用于指定设备,比如 GPU 设备编号等)。

创建目录

# 创建输出目录和检查点目录(如果不存在)
os.makedirs(output_dir, exist_ok=True)
os.makedirs(ckpt_dir, exist_ok=True)

  • 使用os.makedirs函数创建输出目录和检查点目录,如果目录已经存在(exist_ok=True参数表示存在时不报错)则不会重复创建,确保后续保存输出结果和模型检查点时有对应的目录可用。

数据集加载与处理

# 加载数据集
dataset_dir = os.path.join(work_dir, "data", "training_data", "Tang_training_data.json")
with open(dataset_dir, "r", encoding="utf-8") as f:data_json = json.load(f)

  • 构建数据集文件的完整路径(假设数据集是 JSON 格式存储在特定目录下),然后使用json.load函数从打开的文件中读取 JSON 数据并解析为 Python 对象(这里是字典或列表等结构)存储在data_json变量中。

# 保存处理后的数据集
tmp_dataset_path = os.path.join(work_dir, "tmp_dataset.json")
# 获取文件所在目录路径
dir_path = os.path.dirname(tmp_dataset_path)
# 如果目录不存在,则创建目录
if not os.path.exists(dir_path):os.makedirs(dir_path)# 明确使用 'w' 模式打开文件用于写入,同时指定 encoding 为 'utf-8'
with open(tmp_dataset_path, 'w', encoding='utf-8') as f:json.dump(data_json[:num_train_data], f, indent=2, ensure_ascii=False)

  • 先确定临时数据集文件的路径以及其所在目录路径,若目录不存在就创建它。然后从原始读取的数据集数据(data_json)中选取前面num_train_data个样本,使用json.dump将这部分数据以缩进为2、不强制转换 ASCII 编码(ensure_ascii=False,便于处理中文等非 ASCII 字符)的方式写入到临时数据集文件中。

data = load_dataset('json', data_files=tmp_dataset_path, download_mode="force_redownload")

  • 使用load_dataset函数以 JSON 格式加载刚刚处理并保存的临时数据集文件(data_files参数指定),并设置下载模式为强制重新下载(不过对于本地文件这里可能没实际的下载行为,只是按照对应的加载逻辑处理),将加载好的数据集对象存储在data变量中。

创建模型和令牌器

model = AutoModelForCausalLM.from_pretrained(model_path,cache_dir=cache_dir,quantization_config=BitsAndBytesConfig(load_in_4bit=True,bnb_4bit_quant_type="nf4",bnb_4bit_use_double_quant=True,bnb_4bit_compute_dtype=torch.bfloat16),low_cpu_mem_usage=True
)

  • 通过AutoModelForCausalLM类从指定的model_path路径加载预训练的因果语言模型,使用cache_dir指定缓存目录,同时配置了模型量化相关参数(通过BitsAndBytesConfig),这里设置为以4位量化加载(load_in_4bit=True),采用nf4量化类型、启用双重量化以及指定计算的数据类型为torch.bfloat16,并且设置为低 CPU 内存使用模式,以优化模型加载和运行时的内存占用情况。

tokenizer = AutoTokenizer.from_pretrained(model_name,add_eos_token=True,cache_dir=cache_dir,quantization_config=BitsAndBytesConfig(load_in_4bit=True,bnb_4bit_quant_type="nf4",bnb_4bit_use_double_quant=True,bnb_4bit_compute_dtype=torch.bfloat16)
)
tokenizer.pad_token = tokenizer.eos_token

  • 类似地,使用AutoTokenizer从指定的model_name加载对应的文本令牌化器,添加结束符令牌(add_eos_token=True),配置缓存目录和量化相关参数(与模型配置的量化参数一致)。最后将令牌化器的填充令牌(pad_token)设置为结束符令牌(eos_token),方便后续处理文本序列长度不一致等情况。

生成训练数据的函数

def generate_training_data(data_point):"""此函数用于将数据点(包含指令、输入和输出文本)转换为模型可读取的令牌形式参数:data_point (dict): 包含 "instruction"、"input" 和 "output" 字段的字典,所有字段均为字符串返回:dict: 包含模型的输入令牌、注意力掩码和相应输出目标的字典"""# 构建完整的输入提示prompt = f"""\
[INST] <<SYS>>
You are a helpful assistant and good at writing Tang poem. 你是一个乐于助人的助手且擅长写唐诗。
<</SYS>>
{data_point["instruction"]}
{data_point["input"]}
[/INST]"""# 计算输入令牌的数量len_user_prompt_tokens = (len(tokenizer(prompt,truncation=True,max_length=CUTOFF_LEN + 1,padding="max_length",)["input_ids"]) - 1)# 将输入提示转换为令牌full_tokens = tokenizer(prompt + " " + data_point["output"] + "</s>",truncation=True,max_length=CUTOFF_LEN + 1,padding="max_length",)["input_ids"][:-1]return {"input_ids": full_tokens,"labels": [-100] * len_user_prompt_tokens + full_tokens[len_user_prompt_tokens:],"attention_mask": [1] * (len(full_tokens)),}

  • 这个函数用于将输入的包含指令、输入文本和输出文本的数据点转换为适合模型训练的格式:
    • 首先构建完整的输入提示文本(按照特定的格式拼接指令、输入等内容),形成prompt字符串。
    • 然后通过令牌化器对prompt进行处理(截断到CUTOFF_LEN + 1长度并进行填充等操作),计算出用户提示部分的令牌数量(减去1可能是去除某个特定的标记等原因)。
    • 接着将包含输出文本的完整提示再进行令牌化处理,并去掉最后一个令牌(具体原因可能与模型训练时的输入输出格式要求有关),得到full_tokens
    • 最后构建并返回一个字典,包含模型输入的令牌序列(input_ids),对应的标签序列(labels,用户提示部分设为-100表示不计算这部分的损失,后面接上实际输出对应的令牌)以及注意力掩码(全1表示都参与注意力计算,长度与令牌序列长度一致)。

评估模型的函数

def evaluate(instruction, generation_config, max_len, input="", verbose=True):"""此函数用于根据输入的指令、生成配置和最大长度,获取模型的输出参数:instruction (str): 描述模型要执行的操作的字符串generation_config (transformers.GenerationConfig): 用于指定与模型推理相关的解码参数的对象max_len (int): 模型输出的最大长度input (str, 可选): 模型需要解决指令的输入字符串,默认值为 ""verbose (bool, 可选): 是否打印模型的输出,默认值为True返回:str: 模型根据指令和输入生成的响应"""# 构建完整的输入提示prompt = f"""\
[INST] <<SYS>>
You are a helpful assistant and good at writing Tang poem. 你是一个乐于助人的助手且擅长写唐诗。
<</SYS>>
{instruction}
{input}
[/INST]"""# 将提示文本转换为模型所需的数字表示形式inputs = tokenizer(prompt, return_tensors="pt")input_ids = inputs["input_ids"].cuda() if torch.cuda.is_available() else inputs["input_ids"]# 使用模型进行生成回复generation_output = model.generate(input_ids=input_ids,generation_config=generation_config,return_dict_in_generate=True,output_scores=True,max_new_tokens=max_len,)# 将生成的回复解码并打印for s in generation_output.sequences:output = tokenizer.decode(s)output = output.split("[/INST]")[1].replace("</s>", "").replace("<s>", "").replace("Assistant:", "").replace("Assistant", "").strip()if verbose:print(output)return output

  • 该函数用于对模型进行评估,根据给定的指令、生成配置以及最大输出长度等参数获取模型的输出:
    • 先按照特定格式构建完整的输入提示文本prompt,包含系统描述、指令和可选的输入内容。
    • 接着使用令牌化器将提示文本转换为张量形式(return_tensors="pt"表示返回 PyTorch 张量),如果 GPU 可用则将输入的令牌input_ids移到 GPU 上,方便后续加速计算。

evaluate函数(续)

    generation_config=generation_config,return_dict_in_generate=True,output_scores=True,max_new_tokens=max_len,
)

  • 调用模型的generate方法来生成回复内容,传递了几个重要参数:
    • generation_config:这是一个之前定义好的配置对象(类型为transformers.GenerationConfig),里面包含了如采样策略(do_sample等参数)、温度参数(temperature)、束搜索相关参数(num_beams等)等诸多与生成文本过程相关的设置,用于控制模型生成回复的方式。
    • return_dict_in_generate=True:表示让generate方法以字典形式返回生成的结果,方便后续提取和处理相关信息,比如包含生成的序列、对应的分数等信息。
    • output_scores=True:指示模型在生成过程中同时输出每个生成步骤的分数信息(例如概率得分等,虽然代码后续没有明显体现对这些分数的进一步处理,但在一些更详细的分析场景中可能会用到)。
    • max_new_tokens=max_len:指定了模型最多生成的新令牌数量,也就是限制了生成回复的最大长度,避免生成过长的无意义文本。

for s in generation_output.sequences:output = tokenizer.decode(s)output = output.split("[/INST]")[1].replace("</s>", "").replace("<s>", "").replace("Assistant:", "").replace("Assistant", "").strip()if verbose:print(output)

  • 这段代码对生成的结果进行处理和输出(如果verboseTrue):
    • 首先通过循环遍历generation_output.sequences(生成结果中的序列,可能存在多个候选序列等情况,不过这里代码逻辑上似乎只处理了其中一个,没有更复杂的选择逻辑体现)。
    • 使用tokenizer.decode将生成的令牌序列s解码为文本形式,得到原始的回复文本。
    • 接着对解码后的文本进行一系列字符串处理操作:
      • split("[/INST]")[1]:按照[/INST]这个特定标记进行分割,并取分割后的第二部分内容,可能是为了去除前面的一些指令相关的固定格式内容,提取出真正的回复部分。
      • 然后通过replace方法依次去除一些多余的标记如</s>(结束符标记)、<s>(起始符标记)、Assistant:(可能是回复开头的特定前缀标记)以及Assistant(类似的前缀情况),并通过strip方法去除文本两端的空白字符,最终得到比较干净整洁的回复文本内容。
    • 如果verboseTrue,就将处理后的回复文本打印输出,方便查看模型生成的结果情况;如果verboseFalse,则只是进行上述处理,不进行打印,用于后续可能的静默式评估或者将结果进一步传递使用等场景。
    • 最后,函数返回处理后的回复文本(无论是否打印),使得调用该函数的地方可以获取到模型生成的结果用于后续的操作,比如和真实结果对比评估模型性能等。

划分训练集和验证集(如果需要)

if VAL_SET_SIZE > 0:train_val = data["train"].train_test_split(test_size=VAL_SET_SIZE, shuffle=True, seed=42)train_data = train_val["train"].shuffle().map(generate_training_data)val_data = train_val["test"].shuffle().map(generate_training_data)
else:train_data = data['train'].shuffle().map(generate_training_data)val_data = None

  • 这部分代码根据之前定义的验证集大小(VAL_SET_SIZE)来决定是否划分训练集和验证集:
    • 如果VAL_SET_SIZE大于0,表示需要划分出验证集用于在训练过程中评估模型在未见过的数据上的性能。通过调用数据集对象data["train"](假设data数据集有train这个子集表示训练数据部分)的train_test_split方法,按照给定的测试集大小比例(test_size=VAL_SET_SIZE)、进行随机打乱(shuffle=True)并设置随机种子(seed=42,保证每次划分结果可复现)来划分出训练集和验证集两部分,分别存储在train_val这个包含traintest两个子集的对象中。
    • 然后对划分后的训练集和验证集进一步处理,通过调用shuffle方法再次打乱顺序(进一步打乱确保数据随机性,可能是为了后续训练更稳定等原因),并使用map方法结合generate_training_data函数将数据集中的每个样本转换为适合模型训练的格式(如前面generate_training_data函数中所做的将文本数据转换为包含输入令牌、标签、注意力掩码等的字典形式),最终得到处理后的训练数据train_data和验证数据val_data
    • 如果VAL_SET_SIZE等于0,则表示不需要划分验证集,直接将整个训练数据集(data['train'])进行打乱(shuffle)并通过map方法应用generate_training_data函数处理每个样本,得到训练数据train_data,同时将验证数据val_data设为None,表明没有单独的验证集用于后续评估。

将模型准备好以使用 INT8 训练

model = prepare_model_for_int8_training(model)

  • 调用prepare_model_for_int8_training函数对已加载的模型进行预处理,使其能够适应以INT8量化精度进行训练的要求,这个函数内部可能会对模型的一些层结构、参数初始化等方面进行调整,以便在后续训练过程中能够基于INT8量化方式正确地更新参数、计算梯度等,确保训练的顺利进行和模型性能的合理优化。

使用 LoraConfig 配置 LORA 模型

config = LoraConfig(r=LORA_R,lora_alpha=LORA_ALPHA,target_modules=TARGET_MODULES,lora_dropout=LORA_DROPOUT,bias="none",task_type="CAUSAL_LM",
)
model = get_peft_model(model, config)

  • 首先创建一个LoraConfig配置对象,传入之前定义好的相关LORA超参数:
    • r=LORA_R:设置LORA方法中的秩(rank)参数,它决定了低秩矩阵分解的维度等特性,影响模型微调的效果和参数量等。
    • lora_alpha=LORA_ALPHA:与LORA的自适应调整等相关的参数,配合r等参数一起控制模型训练过程中的权重更新等情况。
    • target_modules=TARGET_MODULES:指定了在模型中哪些模块上应用LORA技术进行微调,即明确了操作的具体对象范围。
    • lora_dropout=LORA_DROPOUT:设置LORA中的丢弃率(Dropout)参数,用于防止过拟合等情况。
    • bias="none":表示不处理偏置项(可能是不针对偏置进行LORA相关的调整等情况)。
    • task_type="CAUSAL_LM":表明任务类型是因果语言建模,用于让LORA配置知晓模型的任务特性以便进行针对性的适配。
  • 接着调用get_peft_model函数,传入原始模型和LoraConfig配置对象,对模型进行基于LORA配置的转换和调整,返回一个应用了LORA技术后的新模型,替换原来的model变量,使得后续训练能够基于LORA微调的方式进行,在尽量少增加参数量的情况下提升模型在特定任务上的性能。

定义 nf4_config

nf4_config = BitsAndBytesConfig(load_in_4bit=True,bnb_4bit_quant_type="nf4",bnb_4bit_use_double_quant=True,bnb_4bit_compute_dtype=torch.bfloat16
)

  • 创建一个BitsAndBytesConfig配置对象,重新配置了模型量化相关的参数(和之前模型加载时配置的量化参数类似,但这里单独定义了一个配置对象,可能后续有其他使用场景或者方便修改统一的量化设置等):
    • load_in_4bit=True:明确表示以4位量化的方式加载模型,减少模型的内存占用等。
    • bnb_4bit_quant_type="nf4":指定量化类型为nf4,这是一种特定的量化技术选择。
    • bnb_4bit_use_double_quant=True:启用双重量化机制,进一步优化量化效果和模型存储、计算效率等。
    • bnb_4bit_compute_dtype=torch.bfloat16:设定在量化计算过程中使用的数据类型为torch.bfloat16,用于在量化情况下兼顾计算精度和速度等方面的平衡。

定义生成配置

generation_config = GenerationConfig(do_sample=True,temperature=0.1,num_beams=1,top_p=0.3,no_repeat_ngram_size=3,pad_token_id=2
)

  • 创建一个GenerationConfig对象用于配置模型生成文本时的各种参数:
    • do_sample=True:表示采用采样的方式生成文本,而不是确定性的生成(例如对比于基于束搜索等确定性的生成策略),这样可以增加生成文本的多样性,但也可能带来一定的不确定性和结果的波动性。
    • temperature=0.1:温度参数,用于控制采样的随机性程度,较小的值(如这里的0.1)会让采样更偏向于概率高的词汇,生成的文本相对更保守、更符合常规;较大的值则会增加随机性,使生成的文本更加多样化但可能质量参差不齐。
    • num_beams=1:束搜索相关参数,这里设置为1表示不使用束搜索(束搜索通常会保留多个候选路径并综合评估选择最优路径,当num_beams大于1时启用,这里仅设为1相当于就是普通的逐个生成词汇的方式)。
    • top_p=0.3:一种概率截断采样的参数,叫核采样(nucleus sampling),表示只从概率累计和达到0.3的那部分词汇中进行采样,有助于避免生成一些概率极低但可能出现的不合理词汇,同时保证一定的词汇选择范围和文本多样性。
    • no_repeat_ngram_size=3:设置生成文本时避免重复出现的n元语法(连续的n个词汇组成的单元)的大小为3,即尽量不让连续的3个词汇重复出现,以提高生成文本的流畅性和多样性,防止重复啰嗦的表述。
    • pad_token_id=2:指定填充令牌对应的id2,用于在处理批次数据等场景下对长度不一致的文本序列进行填充对齐操作时的标识。

训练模型

trainer = transformers.Trainer(model=model,train_dataset=train_data,eval_dataset=val_data,args=transformers.TrainingArguments(per_device_train_batch_size=4,  # 微批次大小,可根据需要调整gradient_accumulation_steps=16 // 4,  # 计算每个微批次累积的梯度步数warmup_steps=50,num_train_epochs=num_epoch,learning_rate=LEARNING_RATE,fp16=True,  # 使用混合精度训练logging_steps=20,save_strategy="steps",save_steps=65,output_dir=ckpt_dir,save_total_limit=3,ddp_find_unused_parameters=False if ddp else None,),data_collator=transformers.DataCollatorForLanguageModeling(tokenizer, mlm=False),
)

  • 创建一个transformers.Trainer对象用于训练模型,传入了多个关键参数:
    • model=model:指定要训练的模型,即前面经过INT8训练准备、LORA配置后的模型。
    • train_dataset=train_data:传入训练数据集,是经过前面处理(如转换为合适格式、划分等操作)后的用于训练的样本数据集合。
    • eval_dataset=val_data:传入验证数据集(如果有,即val_data不为None时),用于在训练过程中定期评估模型性能,若val_dataNone则表示训练过程中不进行基于验证集的评估。
    • args=transformers.TrainingArguments(...):创建一个TrainingArguments对象来配置训练相关的各种超参数:
      • per_device_train_batch_size=4:设置每个设备(如每个 GPU)上每次训练的微批次大小为4,即每次处理4个样本数据,这个大小可以根据硬件资源、模型复杂度等情况进行调整,影响训练时的内存占用和梯度更新频率等。
      • gradient_accumulation_steps=16 // 4:计算每个微批次累积的梯度步数,这里实际设置为4(通过16 // 4计算得出),意味着每处理4个微批次后才进行一次梯度更新,用于在内存有限等情况下模拟更大的批次大小训练效果,同时控制梯度更新的节奏。
      • warmup_steps=50:在训练开始阶段设置的热身步数,在这个阶段学习率会从初始值逐渐上升到设定的正常学习率,有助于模型在训练初期更稳定地收敛,避免一开始就使用较大学习率导致训练不稳定等情况。
      • num_train_epochs=num_epoch:指定训练的总轮数,这里使用之前定义的num_epoch变量,即训练1轮(根据前面的定义)。
      • learning_rate=LEARNING_RATE:使用之前定义好的学习率(3e-4)来控制模型参数更新的步长大小。
      • fp16=True:启用混合精度训练,即使用半精度(float16)的数据类型来存储和计算部分张量,减少内存占用同时在支持的硬件上可以加速计算,又能在一定程度上保证计算精度不至于损失太多,适合深度学习模型训练场景。
      • logging_steps=20:按照每训练20步记录一次训练日志,方便查看训练过程中的各项指标变化情况,如损失值、学习率等。
      • save_strategy="steps":表示按照训练步数来决定何时保存模型检查点,与之对应的还有按训练轮数等保存策略,这里选择基于步数保存。
      • save_steps=65:结合save_strategy="steps",意味着每隔65个训练步骤就保存一次模型检查点,用于后续可以恢复训练或者选择不同阶段的模型进行评估等。
      • output_dir=ckpt_dir:指定保存模型检查点的目录为之前定义的ckpt_dir,确保模型文件保存到正确的位置。
      • save_total_limit=3:控制最多保留3个模型检查点,避免过多的检查点占用大量磁盘空间,当保存的检查点数量超过这个限制时,可能会自动删除较早的检查点。
      • ddp_find_unused_parameters=False if ddp else None:在分布式训练(ddpTrue)的场景下,设置是否查找未使用的参数,这里设置为False表示不进行查找(具体是否查找可能根据实际训练需求和模型特点等决定,不查找可以节省一定的计算资源和时间),如果不是分布式训练(ddpFalse)则设为None,可能按照默认行为处理。
    • data_collator=transformers.DataCollatorForLanguageModeling(tokenizer, mlm=False):指定数据整理器(Data Collator),用于将训练数据整理成合适的批次格式,这里使用的是适用于语言建模任务的整理器,传入了令牌化器tokenizer,并且设置mlm=False表示不是做掩码语言建模(而是因果语言建模任务对应的整理方式),它会对批次内的文本序列进行填充、截断等操作,使其符合模型输入的格式要求。

禁用模型的 cache 功能

model.config.use_cache = False

  • 将模型配置中的use_cache属性设置为False,禁用模型的缓存功能。在某些情况下,禁用缓存可以避免一些潜在的内存问题或者不正确的结果,尤其是在训练过程中或者模型结构、输入等有动态变化的场景下,确保每次计算都是基于最新的输入和模型状态进行,而不会依赖之前缓存的中间结果。

训练过程

try:trainer.train()
except Exception as e:print(f"Training failed: {e}")

  • 使用try-except语句块来执行模型训练过程,调用trainer.train()方法启动训练,让模型根据前面配置的训练数据集、超参数等进行训练。如果在训练过程中出现任何异常(比如梯度爆炸、内存不足、运行时错误等),就会捕获该异常并打印出错误信息(Training failed: {e}),方便排查问题,了解训练失败的原因。

保存训练后的模型

try:model.save_pretrained(ckpt_dir)
except Exception as e:print(f"Saving model failed: {e}

 整体代码与工作目录,必要的 package

参照这篇文章

《生成式 AI》课程 作业6 大语言模型(LLM)的训练微调 Fine Tuning -- part2-CSDN博客文章浏览阅读872次,点赞26次,收藏8次。代码围绕一个主工作目录展开,在这个主工作目录下包含了多个子目录和相关文件,用于存放不同阶段的数据、模型以及输出结果等内容,各个部分分工明确,以支持整个预训练语言模型微调及测试的流程。这段 Python 代码主要实现了基于 Hugging Face Transformers 库对预训练语言模型(具体为模型)进行微调(Fine-tuning)的功能,使其能更好地应用于生成唐诗相关内容的任务。整个流程涵盖了数据加载与预处理、模型配置、模型训练以及训练后模型的测试与结果保存等环节。https://blog.csdn.net/chenchihwen/article/details/144000079?spm=1001.2014.3001.5501

训练json数据摘要如下:

Training data:sample, 你可以试着补充
Tang Poem dataseticon-default.png?t=O83Ahttps://github.com/chinese-poetry/chinese-poetry/tree/master/%E5%85%A8%E5%94%90%E8%AF%97?fbclid=IwAR2bM14S42T-VtrvMi3wywCqKfYJraBtMl7QVTo0qyPMjX9jj9Vj3JepFBA

Tang_training_data.json

[

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "秦川雄帝宅,函谷壯皇居。",

    "output": "綺殿千尋起,離宮百雉餘。連甍遙接漢,飛觀迥凌虛。雲日隱層闕,風煙出綺疎。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "巖廊罷機務,崇文聊駐輦。",

    "output": "玉匣啓龍圖,金繩披鳳篆。韋編斷仍續,縹帙舒還卷。對此乃淹留,欹案觀墳典。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "移步出詞林,停輿欣武宴。",

    "output": "琱弓寫明月,駿馬疑流電。驚雁落虛弦,啼猿悲急箭。閱賞誠多美,於茲乃忘倦。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "鳴笳臨樂館,眺聽歡芳節。",

    "output": "急管韻朱絃,清歌凝白雪。彩鳳肅來儀,玄鶴紛成列。去茲鄭衛聲,雅音方可悅。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "芳辰追逸趣,禁苑信多奇。",

    "output": "橋形通漢上,峰勢接雲危。煙霞交隱映,花鳥自參差。何如肆轍跡?萬里賞瑤池。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "白日依山盡,黃河入海流。",

    "output": "欲窮千里目,更上一層樓。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "落日雙闕昏,回輿九重暮。",

    "output": "長煙散初碧,皎月澄輕素。搴幌翫琴書,開軒引雲霧。斜漢耿層閣,清風搖玉樹。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "歡樂難再逢,芳辰良可惜。",

    "output": "玉酒泛雲罍,蘭殽陳綺席。千鍾合堯禹,百獸諧金石。得志重寸陰,忘懷輕尺璧。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "建章歡賞夕,二八盡妖妍。",

    "output": "羅綺昭陽殿,芬芳玳瑁筵。珮移星正動,扇掩月初圓。無勞上懸圃,即此對神仙。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "新豐停翠輦,譙邑駐鳴笳。",

    "output": "園荒一徑斷,苔古半階斜。前池消舊水,昔樹發今花。一朝辭此地,四海遂爲家。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "春蒐馳駿骨,總轡俯長河。",

    "output": "霞處流縈錦,風前瀁卷羅。水花翻照樹,堤蘭倒插波。豈必汾陰曲,秋雲發棹歌。"

  },

  {

    "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

    "input": "重巒俯渭水,碧嶂插遙天。",

    "output": "出紅扶嶺日,入翠貯巖煙。疊松朝若夜,複岫闕疑全。對此恬千慮,無勞訪九仙。"

  },

testing data

 Tang_testing_data.json

[

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "雪霽銀妝素,桔高映瓊枝。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "夫子何爲者?栖栖一代中。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "飛蓋去芳園,蘭橈遊翠渚。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "條風開獻節,灰律動初陽。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "昨夜星辰昨夜風,畫樓西畔桂堂東。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "三日入廚下,洗手作羹湯。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "嵩雲秦樹久離居,雙鯉迢迢一紙書。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "慨然撫長劒,濟世豈邀名。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "乘興南遊不戒嚴,九重誰省諫書函。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "猿鳥猶疑畏簡書,風雲常爲護儲胥。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "君問歸期未有期,巴山夜雨漲秋池。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "相見時難別亦難,東風無力百花殘。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "雲母屏風燭影深,長河漸落曉星沈。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "高閣客竟去,小園花亂飛。"

    },

    {

        "instruction": "以下是一首唐詩的第一句話,請用你的知識判斷並完成整首詩。",

        "input": "瑤池阿母綺窗開,黃竹歌聲動地哀。"

    }

  ]

相关文章:

大语言模型LLM的微调代码详解

代码的摘要说明 一、整体功能概述 这段 Python 代码主要实现了基于 Hugging Face Transformers 库对预训练语言模型&#xff08;具体为 TAIDE-LX-7B-Chat 模型&#xff09;进行微调&#xff08;Fine-tuning&#xff09;的功能&#xff0c;使其能更好地应用于生成唐诗相关内容的…...

钉钉与企业微信机器人:助力网站定时任务高效实现

钉钉、企业微信机器人在网站定时任务中的应用&#xff0c;主要体现在自动化通知、提醒以及数据处理等方面。 以下是一些具体的应用场景&#xff1a; 1. 自动化通知 项目进度提醒&#xff1a;在蒙特网站所负责的软件开发或网站建设项目中&#xff0c;可以利用机器人设置定时任…...

自然语言处理工具-广告配音工具用于语音合成助手/自媒体配音/广告配音/文本朗读-已经解锁了 全功能的 apk包

Android -「安卓端」 广告配音工具用于语音合成助手/自媒体配音/广告配音/文本朗读。 广告配音工具&#xff1a;让您的文字“说话”&#xff0c;在这个快速发展的数字时代&#xff0c;广告配音工具为各种语音合成需求提供了一站式解决方案。无论是自媒体配音、商业广告配音、…...

深入解析注意力机制

引言随着深度学习的快速发展&#xff0c;注意力机制&#xff08;Attention Mechanism&#xff09;逐渐成为许多领域的关键技术&#xff0c;尤其是在自然语言处理&#xff08;NLP&#xff09;和计算机视觉&#xff08;CV&#xff09;中。其核心思想是赋予模型“关注重点”的能力…...

Unity图形学之雾Fog

1.设置雾化&#xff1a; 2.雾化变化曲线&#xff1a;FogMode &#xff08;1&#xff09;线性&#xff1a; &#xff08;2&#xff09;一次指数&#xff1a; &#xff08;3&#xff09;二次指数&#xff1a; Shader "Custom/FogTest" {Properties{_Color ("Color…...

【大数据学习 | Spark-Core】详解Spark的Shuffle阶段

1. shuffle前言 对spark任务划分阶段&#xff0c;遇到宽依赖会断开&#xff0c;所以在stage 与 stage 之间会产生shuffle&#xff0c;大多数Spark作业的性能主要就是消耗在了shuffle环节&#xff0c;因为该环节包含了大量的磁盘IO、序列化、网络数据传输等操作。 负责shuffle…...

如何启动 Docker 服务:全面指南

如何启动 Docker 服务:全面指南 一、Linux 系统(以 Ubuntu 为例)二、Windows 系统(以 Docker Desktop 为例)三、macOS 系统(以 Docker Desktop for Mac 为例)四、故障排查五、总结Docker,作为一种轻量级的虚拟化技术,已经成为开发者和运维人员不可或缺的工具。它允许用…...

使用client-go在命令空间test里面对pod进行操作

目录 一、获取使用restApi调用的token信息 二、client-go操作pod示例 1、获取到客户端 2、创建pod 3、获取test命令空间的所有pod 4、获取某个具体pod的详细信息 5、更新pod 6、删除pod 三、总结 官方参考地址&#xff1a;https://kubernetes.io/docs/reference/kuber…...

Linux中网络文件系统nfs使用

一、nfs服务 NFS&#xff08;Network File System&#xff09; 是一种用于在网络中共享文件的协议&#xff0c;允许不同操作系统&#xff08;如 Linux、Unix、MacOS 等&#xff09;之间进行文件共享。 NFS 的工作原理基于客户端-服务器模型&#xff0c;服务器提供共享文件系统…...

气膜建筑:打造全天候安全作业空间,提升工程建设效率—轻空间

在现代建筑工程中&#xff0c;施工环境的管理和作业效率是决定项目进度和质量的关键因素。然而&#xff0c;施工过程中常常会受到天气变化的影响&#xff0c;诸如大风、雨雪、沙尘等恶劣天气常常延误工期&#xff0c;增加施工难度。为了解决这一问题&#xff0c;气膜建筑以其独…...

【HarmonyOS学习日志(10)】一次开发,多端部署之功能级一多开发,工程级一多开发

功能级一多开发 SysCap机制介绍 HarmonyOS使用SysCap机制&#xff08;即SystemCapability&#xff09;&#xff0c;可以帮助开发者仅关注设备的系统能力&#xff0c;而不用考虑成百上千种具体的设备类型。 在过去&#xff0c;开发不同设备上的应用就用不同设备的SDK进行开发&…...

dmdba用户资源限制ulimit -a 部分配置未生效

dmdba用户资源限制ulimit -a 部分配置未生效 1 环境介绍2 数据库实例日志报错2.1 mpp01 实例日志报错2.2 mpp02 实例日志报错 3 mpp02 服务器资源限制情况4 关闭SELinux 问题解决4.1 临时关闭 SELinux4.2 永久关闭 SELinux 5 达梦数据库学习使用列表 1 环境介绍 Cpu x86 Os Ce…...

【Code First】.NET开源 ORM 框架 SqlSugar 系列

.NET开源 ORM 框架 SqlSugar 系列 【开篇】.NET开源 ORM 框架 SqlSugar 系列【入门必看】.NET开源 ORM 框架 SqlSugar 系列【实体配置】.NET开源 ORM 框架 SqlSugar 系列【Db First】.NET开源 ORM 框架 SqlSugar 系列【Code First】.NET开源 ORM 框架 SqlSugar 系列 &#x1f…...

如何在谷歌浏览器中切换DNS服务器

在浏览网页时&#xff0c;DNS&#xff08;域名系统&#xff09;服务器的作用是将您输入的网址转换为计算机可以理解的IP地址。有时&#xff0c;您可能需要更改默认的DNS服务器以提升网络速度或解决访问问题。本文将详细介绍如何在谷歌浏览器中切换DNS服务器&#xff0c;并在此过…...

Spring Cloud Stream实现数据流处理

1.什么是Spring Cloud Stream&#xff1f; Spring Cloud Stream的核心是Stream&#xff0c;准确来讲Spring Cloud Stream提供了一整套数据流走向&#xff08;流向&#xff09;的API&#xff0c; 它的最终目的是使我们不关心数据的流入和写出&#xff0c;而只关心对数据的业务处…...

列表上移下移功能实现

后台管理某列表需实现上移下移功能&#xff0c;并与前端展示列表排序相关。 现将开发完成过程笔记记录下来。 目录 列表增加属性 JQuery脚本 服务端 控制器 服务层 总结 列表增加属性 在循环渲染时&#xff0c;在table表格的tr上增加id和排序的属性值&#xff0c;以便传…...

升级智享 AI 直播三代:领航原生直播驶向自动化运营新航道

在瞬息万变的数字商业世界&#xff0c;直播行业恰似一艘破浪前行的巨轮&#xff0c;原生直播作为初始 “航船”&#xff0c;在历经风雨后&#xff0c;终于迎来智享 AI 直播三代这股强劲 “东风”&#xff0c;校准航向&#xff0c;开启自动化运营的全新航道&#xff0c;驶向一片…...

Llmcad: Fast and scalable on-device large language model inference

题目&#xff1a;Llmcad: Fast and scalable on-device large language model inference 发表于2023.09 链接&#xff1a;https://arxiv.org/pdf/2309.04255 声称是第一篇speculative decoding边缘设备的论文&#xff08;不一定是绝对的第一篇&#xff09;&#xff0c;不开源…...

Hbase2.2.7集群部署

环境说明 准备三台服务器&#xff0c;分别为&#xff1a;bigdata141&#xff08;作为Hbase主节点&#xff09;、bigdata142、bigdata143确保hadoop和zookeeper集群都先启动好我这边的hadoop版本为3.2.0&#xff0c;zookeeper版本为3.5.8 下载安装包 下载链接&#xff1a;In…...

【青牛科技】D1671 75Ω 带4级低通滤波的单通道视频放大电 路芯片介绍

概 述 &#xff1a; D1671是 一 块 带 4级 低 通 滤 波 的 单 通 道 视 频 放 大 电 路 &#xff0c; 可 在3V或5V的 低 电 压 下 工 作 。 该 电 路 用 在 有 TV影 象 输 出 功 能 的 产 品 上 面&#xff0c;比如 机 顶 盒 &#xff0c;监 控 摄 象 头 &#xff0c;DVD&#…...

[NeurIPS 2022] Leveraging Inter-Layer Dependency for Post-Training Quantization

Contents IntroductionMethodExperimentsReferences Introduction 作者提出一种端到端的 PTQ 训练策略 Network-Wise Quantization (NWQ)&#xff0c;并通过 Annealing Softmax (ASoftmax) 和 Annealing Mixup (AMixup) 改进了 AdaRound&#xff0c;降低了训练收敛难度 Metho…...

ubuntu+ROS推视频流至网络

目录 概述 工具 ros_rtsp 接受流 web_video_server 源码安装 二进制安装 ros接收rtsp视频流 总结 概述 ros_rtsp功能包可以将ros视频流以rtsp形式推送 web_video_server功能包可以将ros视频话题推HTTP流 rocon_rtsp_camera_relay可以接受同一网段下的rtsp视频流输出为…...

PHP 去掉特殊不可见字符 “\u200e“

描述 最近在排查网站业务时&#xff0c;发现有数据匹配失败的情况 肉眼上完全看不出问题所在 当把字符串 【M24308/23-14F‎】复制出来发现 末尾有个不可见的字符 使用删除键或左右移动时才会发现 最后测试通过 var_dump 打印 发现这个"空字符"占了三个长度 &#xf…...

深度学习—BP算法梯度下降及优化方法Day37

梯度下降 1.公式 w i j n e w w i j o l d − α ∂ E ∂ w i j w_{ij}^{new} w_{ij}^{old} - \alpha \frac{\partial E}{\partial w_{ij}} wijnew​wijold​−α∂wij​∂E​ α为学习率 当α过小时&#xff0c;训练时间过久增加算力成本&#xff0c;α过大则容易造成越过最…...

elasticsearch8.16 docker-compose 多机器集群安装

在网上找了一圈, 发现要么就是单机版的部署了多个节点, 很少有多台机器部署集群的, 有些就拿官网的例子写一写, 没有实战经验, 下面分享一个教程, 实实在在的多台机器, 每台机器部署2个节点的例子 先上.env , docker-compose.yml文件, 这个文件是核心, 里面掺杂太多坑, 已经帮你…...

Flink--API 之 Source 使用解析

目录 一、Flink Data Sources 分类概览 &#xff08;一&#xff09;预定义 Source &#xff08;二&#xff09;自定义 Source 二、代码实战演示 &#xff08;一&#xff09;预定义 Source 示例 基于本地集合 基于本地文件 基于网络套接字&#xff08;socketTextStream&…...

uniapp在小程序连接webScoket实现余额支付

webScoket文档&#xff1a;uni.connectSocket(OBJECT) | uni-app官网 /plugins/event.js const Dep function() {this.Evens Object.create(null); } class Event {constructor({dep new Dep()} {}) {if (dep.constructor Object && Object.keys(dep).length 0…...

Spring Boot【三】

自动注入 xml中可以在bean元素中通过autowire属性来设置自动注入的方式&#xff1a; <bean id"" class"" autowire"byType|byName|constructor|default" /> byName&#xff1a;按照名称进行注入 byType&#xff1a;按类型进行注入 constr…...

R 因子

R 因子 引言 在金融领域&#xff0c;风险管理和投资策略的优化一直是核心议题。传统的风险度量工具&#xff0c;如波动率、Beta系数等&#xff0c;虽然在一定程度上能够帮助投资者理解市场的波动和资产的相对风险&#xff0c;但它们往往无法全面捕捉到市场动态的复杂性。因此…...

【博主推荐】C# Winform 拼图小游戏源码详解(附源码)

文章目录 前言摘要1.设计来源拼图小游戏讲解1.1 拼图主界面设计1.2 一般难度拼图效果1.3 普通难度拼图效果1.4 困难难度拼图效果1.5 地域难度拼图效果1.6 内置五种拼图效果 2.效果和源码2.1 动态效果2.2 源代码 源码下载结束语 前言 在数字浪潮汹涌澎湃的时代&#xff0c;程序开…...

万网网站备案多久/百度广告商

如今智能手机大部分已经得到广泛的应用&#xff0c;可玩性的东西也是越来越多&#xff0c;但是手机时间用久了以后便会发现很卡&#xff0c;于是很多机友们都很想知道手机很卡怎么办&#xff0c;如何保持手机系统的流畅性呢&#xff1f;下面小编就为智能手机使用中占多数的安卓…...

网站优化建设绵阳/长春网站优化页面

442.数组中重复的数据442.数组中重复的数据题解代码442.数组中重复的数据 442.数组中重复的数据 题解 题目&#xff1a;给一个数组&#xff0c;返回数组中出现两次的数&#xff0c;数是1-n的&#xff0c;并且只出现一次或两次&#xff0c;要求常数的时间复杂度 思路&#x…...

wordpress is电影主题/整合营销方案案例

前言 继续翻览《程序是怎样跑起来的》 本节是第九章 操作系统和应用的关系 1、操作系统功能的历史 最早是仅具有加载和运行功能的 监控程序 后来&#xff0c;基本的输入输出部分的程序被追加到了监控程序中&#xff0c;初期的操作系统诞生 2、系统调用和高级编程语言的移植性…...

做网站有哪些费用/网站建设黄页视频

1.Python的注释注释的目的是让阅读人员能够轻松读懂每一行代码的意义&#xff0c;同时也为后期代码维护提供便利。在python中&#xff0c;单行注释以#开头&#xff0c;如下所示.#第一个注释 print(Hello,Wold!)#第二个注释Python的多行注释用两个三引号 包含起来&#xff0c;如…...

研究生做网站开发/100%能上热门的文案

之前在windows下写了hello world&#xff0c;终归是不够用啊&#xff0c;因为开发环境是Linux&#xff0c;怎么办呢~~~学习学习再学习 写在前面的话&#xff1a;我从百度文库的一个文章里摘出来的&#xff0c;原文章名称《Linux下安装ApachePHPMySql 搭建PHP运行环境》 http://…...

苏州公司建站/网络推广需要多少钱

Http1.0 、Http1.1 、Http2.0区别 长连接和短链接 在解释区别之前&#xff0c;我们要先理解TCP的长连接和短链接&#xff1a; **短链接&#xff1a;**你朋友给你发信息拉你打游戏&#xff08;发送连接请求&#xff09;&#xff0c;你说行&#xff08;应答&#xff09;&#xf…...