昇思25天学习打卡营第14天|GPT2文本摘要

一、简介:

GPT-2(Generative Pre-trained Transformer 2)是由OpenAI开发的一种基于Transformer架构(decoder-only)的大型自然语言处理模型。它通过在大规模文本数据上进行预训练,能够理解和生成自然语言文本。GPT-2的特点在于其庞大的模型规模和强大的生成能力,能够生成连贯且上下文相关的文本,适用于多种自然语言处理任务,如文本生成、翻译、问答系统等。

二、环境准备:

在开始训练模型之前,需要先安装mindspore和mindnlp两个工具库,mindspore的下载可以参考我的昇思25天学习打卡营第1天|快速入门-CSDN博客 ,mindnlp可以直接使用pip方式下载:

pip install tokenizers==0.15.0 -i https://pypi.tuna.tsinghua.edu.cn/simple

pip install mindnlp

下载完成之后就可以,开始下面的训练了:

三、数据集准备:

1、数据集加载:

本次实验使用的是nlpcc2017摘要数据,内容为新闻正文及其摘要,总计50000个样本

from mindnlp.utils import http_get
import time

# 下载数据
url = 'https://download.mindspore.cn/toolkits/mindnlp/dataset/text_generation/nlpcc2017/train_with_summ.txt'
path = http_get(url, './')

from mindspore.dataset import TextFileDataset

# 加载数据
dataset = TextFileDataset(str(path), shuffle=False)
dataset.get_dataset_size()

# 划分训练集和测试集
train_dataset, test_dataset = dataset.split([0.9, 0.1], randomize=False)

2、数据预处理:

import json
import numpy as np
from mindnlp.transformers import BertTokenizer

# 构建处理的类
def process_dataset(dataset, tokenizer, batch_size=6, max_seq_len=1024, shuffle=False):
    def read_map(text):
        data = json.loads(text.tobytes())
        return np.array(data['article']), np.array(data['summarization'])

    def merge_and_pad(article, summary):
        # tokenization
        # pad to max_seq_length, only truncate the article
        tokenized = tokenizer(text=article, text_pair=summary,
                              padding='max_length', truncation='only_first', max_length=max_seq_len)
        return tokenized['input_ids'], tokenized['input_ids']
    
    dataset = dataset.map(read_map, 'text', ['article', 'summary'])
    # change column names to input_ids and labels for the following training
    dataset = dataset.map(merge_and_pad, ['article', 'summary'], ['input_ids', 'labels'])

    dataset = dataset.batch(batch_size)
    if shuffle:
        dataset = dataset.shuffle(batch_size)

    return dataset


# 使用BertTokenizer分词器
tokenizer = BertTokenizer.from_pretrained('bert-base-chinese')
len(tokenizer)

train_dataset = process_dataset(train_dataset, tokenizer, batch_size=4)
print(next(train_dataset.create_tuple_iterator()))

 四、模型搭建:

1、构建GPT2ForSummarization模型:

from mindspore import ops
from mindnlp.transformers import GPT2LMHeadModel

class GPT2ForSummarization(GPT2LMHeadModel):
    def construct(
        self,
        input_ids = None,
        attention_mask = None,
        labels = None,
    ):
        outputs = super().construct(input_ids=input_ids, attention_mask=attention_mask)
        shift_logits = outputs.logits[..., :-1, :]
        shift_labels = labels[..., 1:]
        # Flatten the tokens
        loss = ops.cross_entropy(shift_logits.view(-1, shift_logits.shape[-1]), shift_labels.view(-1), ignore_index=tokenizer.pad_token_id)
        return loss

在这段代码中,`shift_logits`和`shift_labels`的操作体现了“shift right”的概念。"Shift right"(右移)是一种在自然语言处理中常用的操作,通过将模型的输出(logits)和标签(labels)向右移动一个位置,使得每个时间步的预测与下一个时间步的实际标签对齐,从而在训练过程中有效地计算损失函数,帮助模型学习到正确的上下文关系,生成连贯的文本。具体来说:

1. `shift_logits = outputs.logits[..., :-1, :]`:这一行代码将`logits`向右移动了一个位置。原本`logits`的形状是`[batch_size, sequence_length, vocab_size]`,通过`[..., :-1, :]`的操作,去掉了最后一个时间步的`logits`,相当于将`logits`向右移动了一个位置。

2. `shift_labels = labels[..., 1:]`:这一行代码将`labels`向右移动了一个位置。原本`labels`的形状是`[batch_size, sequence_length]`,通过`[..., 1:]`的操作,去掉了第一个时间步的`labels`,相当于将`labels`向右移动了一个位置。

通过这两步操作,`shift_logits`和`shift_labels`在时间步上对齐了,即`shift_logits`的每个时间步对应的是`shift_labels`的下一个时间步。这种对齐方式是为了计算损失函数,使得模型预测的下一个词与实际的下一个词进行比较,从而训练模型生成连贯的文本。

2、动态学习率:

通过定义一个动态学习率调度器 `LinearWithWarmUp`,通过在训练初期线性增加学习率(预热阶段),然后在训练后期线性降低学习率(衰减阶段),以帮助模型在训练过程中更快地收敛并稳定性能。这种策略结合了学习率的预热和衰减,使得模型在不同训练阶段都能获得合适的学习率,从而提高训练效果。

from mindspore import ops
from mindspore.nn.learning_rate_schedule import LearningRateSchedule

class LinearWithWarmUp(LearningRateSchedule):
    """
    设置动态学习率
    """
    def __init__(self, learning_rate, num_warmup_steps, num_training_steps):
        super().__init__()
        self.learning_rate = learning_rate
        self.num_warmup_steps = num_warmup_steps
        self.num_training_steps = num_training_steps

    def construct(self, global_step):
        if global_step < self.num_warmup_steps:
            return global_step / float(max(1, self.num_warmup_steps)) * self.learning_rate
        return ops.maximum(
            0.0, (self.num_training_steps - global_step) / (max(1, self.num_training_steps - self.num_warmup_steps))
        ) * self.learning_rate

五、模型训练:

下面的训练耗时巨大,最好使用A100完成

num_epochs = 1
warmup_steps = 2000
learning_rate = 1.5e-4

num_training_steps = num_epochs * train_dataset.get_dataset_size()


from mindspore import nn
from mindnlp.transformers import GPT2Config, GPT2LMHeadModel

config = GPT2Config(vocab_size=len(tokenizer))
model = GPT2ForSummarization(config)

lr_scheduler = LinearWithWarmUp(learning_rate=learning_rate, num_warmup_steps=warmup_steps, num_training_steps=num_training_steps)
optimizer = nn.AdamWeightDecay(model.trainable_params(), learning_rate=lr_scheduler)

# 记录模型参数数量
print('number of model parameters: {}'.format(model.num_parameters()))

from mindnlp._legacy.engine import Trainer
from mindnlp._legacy.engine.callbacks import CheckpointCallback

ckpoint_cb = CheckpointCallback(save_path='checkpoint', ckpt_name='gpt2_summarization',
                                epochs=1, keep_checkpoint_max=2)

trainer = Trainer(network=model, train_dataset=train_dataset,
                  epochs=1, optimizer=optimizer, callbacks=ckpoint_cb)
trainer.set_amp(level='O1')  # 开启混合精度

trainer.run(tgt_columns="labels")

 六、模型推理:

首先将测试数据集,转换成中文数据:

def process_test_dataset(dataset, tokenizer, batch_size=1, max_seq_len=1024, max_summary_len=100):
    def read_map(text):
        data = json.loads(text.tobytes())
        return np.array(data['article']), np.array(data['summarization'])

    def pad(article):
        tokenized = tokenizer(text=article, truncation=True, max_length=max_seq_len-max_summary_len)
        return tokenized['input_ids']

    dataset = dataset.map(read_map, 'text', ['article', 'summary'])
    dataset = dataset.map(pad, 'article', ['input_ids'])
    
    dataset = dataset.batch(batch_size)

    return dataset

test_dataset = process_test_dataset(test_dataset, tokenizer, batch_size=1)

print(next(test_dataset.create_tuple_iterator(output_numpy=True)))

导入模型数据,并进行预测:

model = GPT2LMHeadModel.from_pretrained('./checkpoint/gpt2_summarization_epoch_0.ckpt', config=config)

model.set_train(False)
model.config.eos_token_id = model.config.sep_token_id
i = 0
for (input_ids, raw_summary) in test_dataset.create_tuple_iterator():
    output_ids = model.generate(input_ids, max_new_tokens=50, num_beams=5, no_repeat_ngram_size=2)
    output_text = tokenizer.decode(output_ids[0].tolist())
    print(output_text)
    i += 1
    if i == 1:
        break

 由于,我的模型还没训练完成,所以结果就不展示了(doge)

  • 30
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值