gpt2中文训练教程-gpt2文本生成

Posted 147SEO

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了gpt2中文训练教程-gpt2文本生成相关的知识,希望对你有一定的参考价值。

ChatGPT是基于GPT模型的智能对话系统,可用于自然语言生成、文本对话、机器翻译等自然语言处理任务。ChatGPT使用了强化学习技术,能够自动调节模型的参数和优化模型的输出,从而实现更加自然和准确的对话效果。

ChatGPT的核心是GPT模型,它是一种基于Transformer结构的深度学习模型,由OpenAI实现。GPT模型可以对输入的自然语言文本进行多步推导,通过对上下文的理解来生成自然流畅的文本输出。

在ChatGPT中,用户可以通过输入文本消息与机器进行交互。机器会基于对话历史和当前输入的文本消息,自动生成一个适当的回复。ChatGPT的训练数据集通常基于人类自然语言的对话,以确保生成的文本与人类对话类似。在训练中,ChatGPT结合了深度学习和强化学习的技术,而通过强化学习技术的引入,ChatGPT能够自动调整模型参数,优化对话效果并提高输出质量。

ChatGPT已经得到广泛应用。它被广泛应用于实时对话,例如客户服务、智能助手等。此外,ChatGPT也被用于一些文本生成任务,例如新闻报道、市场分析和天气预测等。

总的来说,ChatGPT是一种非常有前途的自然语言处理技术,它能够帮助我们更好地理解人类语言的含义和结构,并在实际应用中提供一种高效的自然语言处理解决方案,并有着广泛的运用前景。

GPT-4和ChatGPT是基于GPT架构的两种不同类型的自然语言处理模型。以下是它们之间的详细区别:

  1. 研究方向和目标

GPT-4和ChatGPT的研究方向和目标不同。GPT-4的主要目标是进一步提高自然语言生成的质量和多样性,从而更接近人类水平的语言理解和生成。而ChatGPT的主要目标是实现智能对话,即让机器能够与人类自然对话,并理解复杂的语言结构和语义含义。

  1. 模型结构和规模

GPT-4和ChatGPT的模型结构和规模也有所不同。GPT-4预计将会包含更多的层和参数,甚至可能会达到百万亿级别的参数数量。而ChatGPT则使用了比较小的模型,通常包含几百万到数千万个参数。

  1. 学习数据和场景

GPT-4和ChatGPT的学习数据和场景也不同。为了实现更高质量的自然语言生成,GPT-4预计将使用更丰富、更复杂的数据集,包括各种类型的语言文本,例如诗歌、小说、科学论文等等。而ChatGPT的学习数据集则更加专注于对话数据,以实现智能对话。

  1. 应用场景和用途

应用场景和用途也是GPT-4和ChatGPT的主要区别之一。GPT-4主要应用于自然语言生成的领域,例如写作、翻译、语音识别等等。而ChatGPT则主要应用于智能对话的领域,例如客户服务、智能助手等等。

总之,虽然GPT-4和ChatGPT都基于GPT架构,但是它们的研究方向、目标、模型结构和使用场景都有所不同。在未来的发展中,将会有更多的基于这些模型,包括更成熟、更精细、更专业的变种,管理着不同的自然语言处理任务和应用场景。

Pytorch——GPT-2 预训练模型及文本生成

介绍

在本次将学习另一个有着优秀表现的预训练模型:GPT-2 模型,以及使用它进行文本生成任务实践。

知识点

  • GPT-2 的核心思想
  • GPT-2 模型结构详解
  • GPT-2 进行文本生成

OpenAI 在论文 Improving Language Understanding by Generative Pre-Training 中提出了 GPT 模型。GPT 模型是由单向 Transformer 的解码器构建的模型,OpenAI 团队在一个非常大的书籍数据集 the Toronto Book Corpus 上对其进行了无监督预训练。

而后,OpenAI 团队又提出了 GPT-2 模型,GPT-2 模型是 GPT 模型的后继,使用了更大的训练集训练,有更多的参数,是 GPT 模型的扩大版。GPT-2 的训练集为数量有 8 百万的网页,由研究人员从网络上爬取得到,大小共有 40 GB 的文本数据,训练任务为给出上文,使模型预测下一个单词。

由于 GPT 与 GPT-2 模型的差别就在于 GPT-2 使用了更多的训练数据,增加了模型参数,在具体结构上并无较大差异。所以,下面我们主要介绍实际表现更优异的 GPT-2 模型。

GPT-2 核心思想

根据研究发现,语言有灵活的表达能力,即能够将任务、输入、输出表示成一个字符串,并且模型可以用这种形式的字符串进行训练,学习相应任务。例如,在翻译任务中,一个训练样本可以被写成

(translate to french, english text, french text)

同样地,在阅读理解任务中,一个训练样本可以被写成

(answer the question, document, question, answer)

并且,人们可以用以上格式的多种任务的训练样本同时训练一个模型,使得该模型获得同时执行多种任务的能力。

于是 OpenAI 研究人员推测,一个具有足够能力的语言模型将学习到推理和执行训练样本中所展示出的任务,以便更好地预测它们。如果一个语言模型能够做到这一点,那么它实际上就是在进行无监督的多任务学习。于是研究人员决定通过分析语言模型在各种各样的任务上的性能来测试这种情况是否属实,这样便有了 GPT-2 的产生。

而且正如他们所推测的,GPT-2 在多种任务上的性能表现很好,具体如下图:

技术图片

来源

所以 GPT-2 对于其他预训练模型的一个突破就在于,它能够在未针对特定下游任务进行训练的条件下,就在下游任务如:阅读理解、机器翻译、问答和文本概括上有很好的表现。这也表明了,在模型足够大,训练数据足够充足时,无监督训练技术也能训练出在多种下游任务上有很好表现的模型。

因为监督学习需要大量的数据,并且需要被仔细清理过的数据,想要得到这样的数据需要昂贵的人力成本。无监督学习可以克服这个缺点,因为它不需要人工标注,有大量现成的数据可以利用。这也表明了 GPT-2 模型研究的意义。

在了解了构建 GPT-2 模型的思想后,接下来我们将详细了解一下 GPT-2 模型的结构。

GPT-2 模型结构

GPT-2 的整体结构如下图,GPT-2 是以 Transformer 为基础构建的, 使用字节对编码的方法进行数据预处理,通过预测下一个词任务进行预训练的语言模型,下面我们从 GPT-2 的预处理方法出发,来一步步详细解析一下 GPT-2。

技术图片

来源

字节对编码

GPT-2 模型在数据预处理时使用了字节对编码(Byte Pair Encoding,简称 BPE)方法,BPE 是一种能够解决未登录词问题,并减小词典大小的方法。它综合利用了单词层面编码和字符层面编码的优势,举例来说,我们要对下面的字符串编码,

aaabdaaabac

字节对 aa 出现的次数最多,所以我们将它替换成一个没在字符串中被用过的字符 Z

ZabdZabac
Z=aa

然后我们重复这个过程,用 Y 替换 ab

ZYdZYac
Y=ab
Z=aa

继续,用 X 替换 ZY

XdXac
X=ZY
Y=ab
Z=aa

这个过程重复进行,直到没有字节对出现超过一次。当需要解码时,就将上述替换过程反向进行。

下面是一段 BPE 算法原文中对 BPE 算法的实现:

import re
import collections


def get_stats(vocab):
    pairs = collections.defaultdict(int)
    for word, freq in vocab.items():
        symbols = word.split()
        for i in range(len(symbols)-1):
            pairs[symbols[i], symbols[i+1]] += freq  # 计算字节对出现频率
    return pairs


def merge_vocab(pair, v_in):
    v_out = {}
    bigram = re.escape(' '.join(pair))  # 将字节对中可解释为正则运算符的字符转义
    p = re.compile(r'(?<!S)' + bigram + r'(?!S)')  # 将要合并的字节对前后只能为空白字符
    for word in v_in:
        w_out = p.sub(''.join(pair), word)  # 合并符合条件的字节对
        v_out[w_out] = v_in[word]
    return v_out

vocab = {'l o w </w>': 5, 'l o w e r </w>': 2,
         'n e w e s t </w>': 6, 'w i d e s t </w>': 3}
num_merges = 10
for i in range(num_merges):
    pairs = get_stats(vocab)
    best = max(pairs, key=pairs.get)  # 选择频率最大的字节对
    vocab = merge_vocab(best, vocab)
    print(best)

单向 Transformer 解码器结构

GPT-2 模型由多层单向 Transformer 的解码器部分构成,本质上是自回归模型,自回归的意思是指,每次产生新单词后,将新单词加到原输入句后面,作为新的输入句。其中 Transformer 解码器结构如下图:

技术图片

来源

GPT-2 模型中只使用了多个 Masked Self-Attention 和 Feed Forward Neural Network 两个模块。如下图所示:

技术图片

来源

可以看到,GPT-2 模型会将语句输入上图所示的结构中,预测下一个词,然后再将新单词加入,作为新的输入,继续预测。损失函数会计算预测值与实际值之间的偏差。

从上一节我们了解到 BERT 是基于双向 Transformer 结构构建,而 GPT-2 是基于单向 Transformer,这里的双向与单向,是指在进行注意力计算时,BERT会同时考虑被遮蔽词左右的词对其的影响,而 GPT-2 只会考虑在待预测词位置左侧的词对待预测词的影响。

通过上述数据预处理方法和模型结构,以及大量的数据训练出了 GPT-2 模型。OpenAI 团队由于安全考虑,没有开源全部训练参数,而是提供了小型的预训练模型,接下来我们将在 GPT-2 预训练模型的基础上进行。

GPT-2 文本生成

GPT-2 就是一个语言模型,能够根据上文预测下一个单词,所以它就可以利用预训练已经学到的知识来生成文本,如生成新闻。也可以使用另一些数据进行微调,生成有特定格式或者主题的文本,如诗歌、戏剧。所以接下来,我们会用 GPT-2 模型进行一个文本生成。

预训练模型生成新闻

想要直接运行一个预训练好的 GPT-2 模型,最简单的方法是让它自由工作,即随机生成文本。换句话说,在开始时,我们给它一点提示,即一个预定好的起始单词,然后让它自行地随机生成后续的文本。

但这样有时可能会出现问题,例如模型陷入一个循环,不断生成同一个单词。为了避免这种情况, GPT-2 设置了一个 top-k 参数,这样模型就会从概率前 k 大的单词中随机选取一个单词,作为下一个单词。下面是选择 top-k 的函数的实现,

import random

def select_top_k(predictions, k=10):
    predicted_index = random.choice(
        predictions[0, -1, :].sort(descending=True)[1][:10]).item()
    return predicted_index

下面引入 GPT-2 模型,我们将使用在 PyTorch-Transformers 模型库中封装好的 GPT2Tokenizer()GPT2LMHeadModel() 类来实际看一下 GPT-2 在预训练后的对下一个词预测的能力。首先,需要安装 PyTorch-Transformers。

!pip install pytorch_transformers==1.0  # 安装 PyTorch-Transformers

使用 PyTorch-Transformers 模型库,先设置好准备输入模型的例子,使用 GPT2Tokenizer() 建立分词器对象对原句编码。

import torch
from pytorch_transformers import GPT2Tokenizer

import logging
logging.basicConfig(level=logging.INFO)

# 载入预训练模型的分词器
tokenizer = GPT2Tokenizer.from_pretrained('gpt2')

# 使用 GPT2Tokenizer 对输入进行编码
text = "Yesterday, a man named Jack said he saw an alien,"
indexed_tokens = tokenizer.encode(text)
tokens_tensor = torch.tensor([indexed_tokens])
tokens_tensor.shape

接下来使用 GPT2LMHeadModel() 建立模型,并将模型模式设为验证模式。由于预训练模型参数体积很大,且托管在外网,所以本次先从网盘下载预训练模型,本地无需此步骤。

from pytorch_transformers import GPT2LMHeadModel

# 读取 GPT-2 预训练模型
model = GPT2LMHeadModel.from_pretrained("./")
model.eval()

total_predicted_text = text
n = 100  # 预测过程的循环次数
for _ in range(n):
    with torch.no_grad():
        outputs = model(tokens_tensor)
        predictions = outputs[0]

    predicted_index = select_top_k(predictions, k=10)
    predicted_text = tokenizer.decode(indexed_tokens + [predicted_index])
    total_predicted_text += tokenizer.decode(predicted_index)

    if '<|endoftext|>' in total_predicted_text:
        # 如果出现文本结束标志,就结束文本生成
        break

    indexed_tokens += [predicted_index]
    tokens_tensor = torch.tensor([indexed_tokens])

print(total_predicted_text)

运行结束后,我们观察一下模型生成的文本,可以看到,大致感觉上这好像是一段正常的文本,不过,仔细看就会发现语句中的逻辑问题,这也是之后研究人员会继续攻克的问题。

除了直接利用预训练模型生成文本,我们还可以使用微调的方法使 GPT-2 模型生成有特定风格和格式的文本。

微调生成戏剧文本

接下来,我们将使用一些戏剧剧本对 GPT-2 进行微调。由于 OpenAI 团队开源的 GPT-2 模型预训练参数为使用英文数据集预训练后得到的,虽然可以在微调时使用中文数据集,但需要大量数据和时间才会有好的效果,所以这里我们使用了英文数据集进行微调,从而更好地展现 GPT-2 模型的能力。

首先,下载训练数据集,这里使用了莎士比亚的戏剧作品《罗密欧与朱丽叶》作为训练样本。数据集已经提前下载好并放在云盘中,链接:https://pan.baidu.com/s/1LiTgiake1KC8qptjRncJ5w 提取码:km06

with open('./romeo_and_juliet.txt', 'r') as f:
    dataset = f.read()

len(dataset)

预处理训练集,将训练集编码、分段。

indexed_text = tokenizer.encode(dataset)
del(dataset)

dataset_cut = []
for i in range(len(indexed_text)//512):
    # 将字符串分段成长度为 512
    dataset_cut.append(indexed_text[i*512:i*512+512])
del(indexed_text)

dataset_tensor = torch.tensor(dataset_cut)
dataset_tensor.shape

这里使用 PyTorch 提供的 DataLoader() 构建训练集数据集表示,使用 TensorDataset() 构建训练集数据迭代器。

from torch.utils.data import DataLoader, TensorDataset

# 构建数据集和数据迭代器,设定 batch_size 大小为 2
train_set = TensorDataset(dataset_tensor,
                          dataset_tensor)  # 标签与样本数据相同
train_loader = DataLoader(dataset=train_set,
                          batch_size=2,
                          shuffle=False)
train_loader

检查是否机器有 GPU,如果有就在 GPU 运行,否则就在 CPU 运行。

device = torch.device("cuda" if torch.cuda.is_available() else "cpu")
device

开始训练。

from torch import nn
from torch.autograd import Variable
import time

pre = time.time()

epoch = 30  # 循环学习 30 次

model.to(device)
model.train()
optimizer = torch.optim.Adam(model.parameters(), lr=1e-5)  # 定义优化器

for i in range(epoch):
    total_loss = 0
    for batch_idx, (data, target) in enumerate(train_loader):
        data, target = Variable(data).to(device), Variable(
            target).to(device)

        optimizer.zero_grad()

        loss, logits, _ = model(data, labels=target)

        total_loss += loss

        loss.backward()
        optimizer.step()

        if batch_idx == len(train_loader)-1:
            # 在每个 Epoch 的最后输出一下结果
            print('average loss:', total_loss/len(train_loader))

print('训练时间:', time.time()-pre)

训练结束后,可以使模型生成文本,观察输出。

text = "From fairest creatures we desire"  # 这里也可以输入不同的英文文本
indexed_tokens = tokenizer.encode(text)
tokens_tensor = torch.tensor([indexed_tokens])

model.eval()
total_predicted_text = text

# 使训练后的模型进行 500 次预测
for _ in range(500):
    tokens_tensor = tokens_tensor.to('cuda')

    with torch.no_grad():
        outputs = model(tokens_tensor)
        predictions = outputs[0]

    predicted_index = select_top_k(predictions, k=10)

    predicted_text = tokenizer.decode(indexed_tokens + [predicted_index])
    total_predicted_text += tokenizer.decode(predicted_index)
    if '<|endoftext|>' in total_predicted_text:
        # 如果出现文本结束标志,就结束文本生成
        break

    indexed_tokens += [predicted_index]

    if len(indexed_tokens) > 1023:
        # 模型最长输入长度为1024,如果长度过长则截断
        indexed_tokens = indexed_tokens[-1023:]

    tokens_tensor = torch.tensor([indexed_tokens])

print(total_predicted_text)

从生成结果可以看到,模型已经学习到了戏剧剧本的文本结构。但是仔细读起来会发现缺少逻辑和关联,这是因为由于时间和设备的限制,对模型的训练比较有限。如果有条件可以用更多的数据,训练更长的时间,这样模型也会有更好的表现。

总结

本次中,我们首先学习了 GPT-2 的构建思想和它的结构,并观察了预训练结果,最后使用了 Fine-Tuning 方法应用 GPT-2 预训练模型进行了戏剧文本生成任务。

相关链接

以上是关于gpt2中文训练教程-gpt2文本生成的主要内容,如果未能解决你的问题,请参考以下文章

在深度学习模型训练中修改学习率

训练自己的GPT2-Chinese模型

基于Mindspore2.0的GPT2预训练模型迁移教程

基于Mindspore2.0的GPT2预训练模型迁移教程

带有 wiki 文章的 GPT2 输入大小

使用/指定 attention_mask 使用 Trainer 和 TrainingArguments 训练 GPT2