从这篇开始,我们要进入“炼丹术”的下一阶段——进阶实战。

在这一新的阶段里,我们将结合开源的轻量级大语言模型MiniMind,针对《大模型炼丹术》系列中没有涉及或者浅浅带过的进阶知识点进行详细讲解。


具体来说,我们将对MiniMind的源码进行逐行解读,并结合张量(也就是数据)在模型中shape的变化,一步步看清楚它是怎么运作的。

正文马上开始~

一、导入训练tokenizer所需库

tokenizers是Hugging Face出的一个高性能、可定制的子词分词器库,主要用于训练和使用像BPE、WordPiece、Unigram等子词模型,是训练 LLM(如GPT/BERT)时常用的工具。

import random
import json
from tokenizers import (
    Tokenizer,
    decoders,
    models,
    pre_tokenizers,
    trainers,
)
import os

Tokenizer是核心分词器对象,控制整个分词、编码、解码过程,可以与不同的模型、预处理器和解码器配合使用。

models包含各种子词分词模型(如 BPE、WordPiece、Unigram),定义了如何对文本进行分割与映射成token IDs。

pre_tokenizers定义了文本的预处理方式,负责在真正的分词前对文本进行初步分割,如按空格、字节或其他规则分割。

trainers用于训练分词模型的工具,包括设置词表大小、特殊符号等的参数配置,常用的有 BpeTrainerWordPieceTrainer 等。

decoders用于将分词后的 token IDs 转回原始文本(解码),支持不同的解码策略,如 ByteLevelMetaspace

二、初始化tokenizer

tokenizer = Tokenizer(models.BPE())
tokenizer.pre_tokenizer = pre_tokenizers.ByteLevel(add_prefix_space=False)

代码首先定义了一个使用BPE模型的分词器,该分词器使用BPE算法将文本拆分成子词单元,以增强模型对未登录词和低频词的处理能力。

然后将预处理器设置为ByteLevel,这一步将文本转换为字节级别的单位,允许更细粒度的文本处理,且add_prefix_space=False控制是否在每个单词前加空格,由于处理的是中文,因此将其设置为False。

三、设置训练器trainer并添加特殊token

# 定义特殊token
special_tokens = ["<|endoftext|>", "<|im_start|>", "<|im_end|>"]

# 设置训练器并添加特殊token
trainer = trainers.BpeTrainer(
    vocab_size=6400,
    special_tokens=special_tokens,  # 确保这三个token被包含
    show_progress=True,
    initial_alphabet=pre_tokenizers.ByteLevel.alphabet()
)

首先来看这些特殊tokens。

它们是在训练过程中专门定义的,用于标识特定的文本模式或结构,通常用于控制生成文本的结构。

  • "<|endoftext|>"用于表示文本的结束,通常在模型生成任务中用来指示生成文本的终止。
  • "<|im_start|>"标识对话或任务的开始,可以用于标记输入文本的起始。
  • "<|im_end|>"标识对话或任务的结束,用来标记输入文本的结束。

这些特殊tokens会在训练过程中作为词表的一部分,确保它们在分词和生成过程中能被正确处理。

接下来解释trainer的参数。

  • vocab_size=6400表示模型训练过程中会生成最多 6400 个子词(包括特殊 tokens)。一般情况下,词汇表的大小会在模型训练之前根据训练数据的大小和复杂度来决定。
  • special_tokens=special_tokens用于标记特殊tokens,使得它们不会在BPE训练过程中被拆分或合并,因为这些特殊标记一般用于标记对话的开始或者结束等特殊意义,而没有明显的对话语义信息。具体来说,在 BPE 训练过程中,算法会找到文本中最频繁的字节对(或字符对),并将它们合并成新的词汇项。例如,BPE会将两个常见的字符组合,如’a’和’b’,合并成一个新的词汇项’ab’。然而,特殊tokens是不参与这一过程的。它们是已经定义好的、固定的标记,不会被拆分或进行BPE 合并。所以,当训练过程中遇到 “<|im_start|>” 这样的特殊token时,它会被保留为一个整体,BPE不会再进一步拆分它。
  • show_progress=True用于在训练过程中会展示当前的训练进度,以便跟踪训练的进度和耗时。
  • initial_alphabet=pre_tokenizers.ByteLevel.alphabet()指定了BPE模型的初始字母表。在这里使用的是ByteLevel的字母表,ByteLevel是一个字符级的预处理方法,它将文本拆解为字节级的子单元(包括字母、标点符号、空格等)。这一设置的作用是让训练器初始化时使用ByteLevel分词器的默认字母表,从而使得 ByteLevel分词器能够适应不同的字符和符号类型,特别是在处理包含非标准字符的文本时非常有用。

四、读取文本数据

使用预训练数据集训练tokenizer,为了便于演示,这里只读取前100条数据。

# 读取JSONL文件并提取文本数据
defread_texts_from_jsonl(file_path, max_samples=100):
with open(file_path, 'r', encoding='utf-8') as f:
for i, line in enumerate(f):
if i >= max_samples:
break
            data = json.loads(line)
yield data['text']

data_path = r'D:\MyFile\github\minimind-master\minimind_dataset\pretrain_hq.jsonl'
texts = read_texts_from_jsonl(data_path)

查看数据示例:

print(list(texts)[1])
<|im_start|>根据输入的内容,编写一个类别标签。
这是一篇介绍如何阅读心电图的文章类别标签: 医学/心电图阅读指南<|im_end|> <|im_start|>帮我搜索一下最近的天气情况。当然,我可以帮您搜索最新的天气情况。请问您需要查询哪个城市的天气情况呢?<|im_end|> <|im_start|>帮我讲一个令人开心的笑话。好的,我帮您讲一个关于细菌的笑话。为什么细菌不会上网?因为连接总是断开了!<|im_end|> <|im_start|>现在给我生成一首关于大海的五言诗。碧波万顷月满天,海天相接处天地间。波涛滚滚江山美,海鸟翱翔日月闲。<|im_end|> <|im_start|>谢谢你,这篇文章很有用。不客气,我很高兴能够为您提供帮助。如果您还有其他问题或需求,随时可以对我说。<|im_end|> <|im_start|>你好,我想下载一个视频编辑软件,你有什么推荐吗?您好!当然,有很多选择。您想要免费软件还是愿意付费?<|im_end|> <|im_start|>为什么我的程序不输出正确结果?可能是代码逻辑有误,或者输入数据有误,需要仔细调试代码逻辑和输入数据。<|im_end|> <|im_start|>谢谢你的回答。现在我想知道这场比赛的具体时间和地点。这场比赛的时间是北京时间10月4日,地点是上海。<|im_end|>

五、开始训练tokenizer

直接调用tokenizertrain_from_iterator方法即可开始训练

# 训练tokenizer
tokenizer.train_from_iterator(texts, trainer=trainer)

六、设置解码器

为分词器设置一个 ByteLevel 解码器,让其在将 token ID 序列转换回原始文本时,能够正确还原被分词器按字节切分的内容。

tokenizer.decoder = decoders.ByteLevel()

七、保存训练好的tokenizer

tokenizer_dir = r"./model"
os.makedirs(tokenizer_dir, exist_ok=True)
tokenizer.save(os.path.join(tokenizer_dir, "tokenizer.json"))
tokenizer.model.save(tokenizer_dir)

八、手动创建并保存配置文件

# 手动创建配置文件
config = {
"add_bos_token": False,
"add_eos_token": False,
"add_prefix_space": False,
"added_tokens_decoder": {
"0": {
"content": "<|endoftext|>",
"lstrip": False,
"normalized": False,
"rstrip": False,
"single_word": False,
"special": True
        },
"1": {
"content": "<|im_start|>",
"lstrip": False,
"normalized": False,
"rstrip": False,
"single_word": False,
"special": True
        },
"2": {
"content": "<|im_end|>",
"lstrip": False,
"normalized": False,
"rstrip": False,
"single_word": False,
"special": True
        }
    },
"additional_special_tokens": [],
"bos_token": "<|im_start|>",
"clean_up_tokenization_spaces": False,
"eos_token": "<|im_end|>",
"legacy": True,
"model_max_length": 32768,
"pad_token": "<|endoftext|>",
"sp_model_kwargs": {},
"spaces_between_special_tokens": False,
"tokenizer_class": "PreTrainedTokenizerFast",
"unk_token": "<|endoftext|>",
"chat_template": "{% if messages[0]['role'] == 'system' %}{% set system_message = messages[0]['content'] %}{{ '<|im_start|>system\\n' + system_message + '<|im_end|>\\n' }}{% else %}{{ '<|im_start|>system\\nYou are a helpful assistant<|im_end|>\\n' }}{% endif %}{% for message in messages %}{% set content = message['content'] %}{% if message['role'] == 'user' %}{{ '<|im_start|>user\\n' + content + '<|im_end|>\\n<|im_start|>assistant\\n' }}{% elif message['role'] == 'assistant' %}{{ content + '<|im_end|>' + '\\n' }}{% endif %}{% endfor %}"
}

# 保存配置文件
with open(os.path.join(tokenizer_dir, "tokenizer_config.json"), "w", encoding="utf-8") as config_file:
    json.dump(config, config_file, ensure_ascii=False, indent=4)
字段名 解释
add_bos_token 是否自动在文本开头添加 bos_token(如 `<
add_eos_token 是否自动在文本末尾添加 eos_token(如 `<
add_prefix_space Byte-level 分词时是否在文本前加空格。通常英文中启用(True)更好,中文中设为 False。
added_tokens_decoder 特殊 token 的详细配置。包括 token 内容、是否为特殊 token、是否仅限单词等。key 是内部 token ID。
additional_special_tokens 除了 bos/eos/pad/unk 外,额外声明的特殊 token 列表。当前为空。
bos_token 起始 token,通常用于语言模型的开头控制符,这里设为 `<
clean_up_tokenization_spaces 解码时是否清理 token 化带来的空格冗余。False 表示不清理。
eos_token 结束 token,通常用于语言模型输出结束的标记,这里设为 `<
legacy 设置为 True 兼容旧版本 tokenizer 行为。推荐保持默认。
model_max_length 模型支持的最大 token 长度。超过将触发截断或报错。这里为 32768。
pad_token 用于对齐 padding 的特殊 token。此处为 `<
sp_model_kwargs SentencePiece 模型的额外配置参数(当前为 BPE,未使用,故为空)。
spaces_between_special_tokens 是否在特殊 token 之间自动添加空格。设置为 False。
tokenizer_class 指定 tokenizer 类型。Hugging Face 使用 "PreTrainedTokenizerFast" 支持 Rust 实现加速。
unk_token 用于标记未知词(out-of-vocabulary)的 token,这里也设为 `<
chat_template Jinja2 模板字符串,用于格式化对话数据为模型输入格式。适用于 Chat 模型(如 LLaMA2-Chat、ChatGPT)。

九、测试训练好的tokenizer

defeval_tokenizer():
from transformers import AutoTokenizer

# 加载预训练的tokenizer
    tokenizer = AutoTokenizer.from_pretrained(r"D:\MyFile\github\minimind-master\mm")

    messages = [
        {"role": "system", "content": "你是一个优秀的聊天机器人,总是给我正确的回应!"},
        {"role": "user", "content": '你来自哪里?'},
        {"role": "assistant", "content": '我来自地球'}
    ]
    new_prompt = tokenizer.apply_chat_template(
        messages,
        tokenize=False
    )

# 获取实际词汇表长度(包括特殊符号)
    actual_vocab_size = len(tokenizer)
    print('tokenizer实际词表长度:', actual_vocab_size)

    model_inputs = tokenizer(new_prompt)
    print('encoder长度:', len(model_inputs['input_ids']))

    input_ids = model_inputs['input_ids']
    response = tokenizer.decode(input_ids, skip_special_tokens=False)
    print('decoder和原始文本是否一致:', response == new_prompt)

    print('\n输入文本:\n',new_prompt,'\n')
    print('解码文本:\n',response,'\n')

eval_tokenizer()
tokenizer实际词表长度: 259
encoder长度: 133
decoder和原始文本是否一致: True

输入文本:
 <|im_start|>system
你是一个优秀的聊天机器人,总是给我正确的回应!<|im_end|>
<|im_start|>user
你来自哪里?<|im_end|>
<|im_start|>assistant
我来自地球<|im_end|>

解码文本:
 <|im_start|>system
你是一个优秀的聊天机器人,总是给我正确的回应!<|im_end|>
<|im_start|>user
你来自哪里?<|im_end|>
<|im_start|>assistant
我来自地球<|im_end|>

至此,关于MiniMind中的tokenizer训练部分就解读完成了。


(如果你也想从0到1看懂大语言模型的底层实现,欢迎关注我,系列文章将持续更新,我们一起把大模型“拆干净”)

大模型算是目前当之无愧最火的一个方向了,算是新时代的风口!有小伙伴觉得,作为新领域、新方向人才需求必然相当大,与之相应的人才缺乏、人才竞争自然也会更少,那转行去做大模型是不是一个更好的选择呢?是不是更好就业呢?是不是就暂时能抵抗35岁中年危机呢?

答案当然是这样,大模型必然是新风口!

那如何学习大模型 ?

由于新岗位的生产效率,要优于被取代岗位的生产效率,所以实际上整个社会的生产效率是提升的。但是具体到个人,只能说是:

最先掌握AI的人,将会比较晚掌握AI的人有竞争优势。
这句话,放在计算机、互联网、移动互联网的开局时期,都是一样的道理。

但现在很多想入行大模型的人苦于现在网上的大模型老课程老教材,学也不是不学也不是,基于此我用做产品的心态来打磨这份大模型教程,深挖痛点并持续修改了近100余次后,终于把整个AI大模型的学习路线完善出来!

在这里插入图片描述

在这个版本当中:

您只需要听我讲,跟着我做即可,为了让学习的道路变得更简单,这份大模型路线+学习教程已经给大家整理并打包分享出来, 😝有需要的小伙伴,可以 扫描下方二维码领取🆓↓↓↓

👉CSDN大礼包🎁:全网最全《LLM大模型学习资源包》免费分享(安全咨料,放心领取)👈

一、大模型经典书籍(免费分享)

AI大模型已经成为了当今科技领域的一大热点,那以下这些大模型书籍就是非常不错的学习资源

在这里插入图片描述

二、640套大模型报告(免费分享)

这套包含640份报告的合集,涵盖了大模型的理论研究、技术实现、行业应用等多个方面。无论您是科研人员、工程师,还是对AI大模型感兴趣的爱好者,这套报告合集都将为您提供宝贵的信息和启示。(几乎涵盖所有行业)
在这里插入图片描述

三、大模型系列视频教程(免费分享)

在这里插入图片描述

四、2025最新大模型学习路线(免费分享)

我们把学习路线分成L1到L4四个阶段,一步步带你从入门到进阶,从理论到实战。

img

L1阶段:启航篇丨极速破界AI新时代

L1阶段:了解大模型的基础知识,以及大模型在各个行业的应用和分析,学习理解大模型的核心原理、关键技术以及大模型应用场景。

img

L2阶段:攻坚篇丨RAG开发实战工坊

L2阶段:AI大模型RAG应用开发工程,主要学习RAG检索增强生成:包括Naive RAG、Advanced-RAG以及RAG性能评估,还有GraphRAG在内的多个RAG热门项目的分析。

img

L3阶段:跃迁篇丨Agent智能体架构设计

L3阶段:大模型Agent应用架构进阶实现,主要学习LangChain、 LIamaIndex框架,也会学习到AutoGPT、 MetaGPT等多Agent系统,打造Agent智能体。

img

L4阶段:精进篇丨模型微调与私有化部署

L4阶段:大模型的微调和私有化部署,更加深入的探讨Transformer架构,学习大模型的微调技术,利用DeepSpeed、Lamam Factory等工具快速进行模型微调,并通过Ollama、vLLM等推理部署框架,实现模型的快速部署。

img

L5阶段:专题集丨特训篇 【录播课】

img

全套的AI大模型学习资源已经整理打包,有需要的小伙伴可以微信扫描下方二维码免费领取

👉CSDN大礼包🎁:全网最全《LLM大模型学习资源包》免费分享(安全资料,放心领取)👈

Logo

更多推荐