#python-3.x #machine-learning #nlp #huggingface-transformers #transformer
Вопрос:
Этот вопрос аналогичен вопросу о том, как я могу проверить confusion_matrix после точной настройки с помощью пользовательских наборов данных?, на Обмене стеками науки о данных.
Фон
Я хотел бы проверить confusion_matrix, включая точность, отзыв и оценку f1, как показано ниже, после точной настройки с помощью пользовательских наборов данных.
Процесс тонкой настройки и задача-классификация последовательностей с обзорами IMDb в руководстве по тонкой настройке с помощью пользовательских наборов данных по Обнимающемуся лицу.
После завершения точной настройки с тренером, как я могу проверить confusion_matrix в этом случае?
Изображение confusion_matrix, включая точность, отзыв и исходный сайт с оценкой f1: только для примера выходного изображения
predictions = np.argmax(trainer.test(test_x), axis=1)
# Confusion matrix and classification report.
print(classification_report(test_y, predictions))
precision recall f1-score support
0 0.75 0.79 0.77 1000
1 0.81 0.87 0.84 1000
2 0.63 0.61 0.62 1000
3 0.55 0.47 0.50 1000
4 0.66 0.66 0.66 1000
5 0.62 0.64 0.63 1000
6 0.74 0.83 0.78 1000
7 0.80 0.74 0.77 1000
8 0.85 0.81 0.83 1000
9 0.79 0.80 0.80 1000
avg / total 0.72 0.72 0.72 10000
Код
from transformers import DistilBertForSequenceClassification, Trainer, TrainingArguments
training_args = TrainingArguments(
output_dir='./results', # output directory
num_train_epochs=3, # total number of training epochs
per_device_train_batch_size=16, # batch size per device during training
per_device_eval_batch_size=64, # batch size for evaluation
warmup_steps=500, # number of warmup steps for learning rate scheduler
weight_decay=0.01, # strength of weight decay
logging_dir='./logs', # directory for storing logs
logging_steps=10,
)
model = DistilBertForSequenceClassification.from_pretrained("distilbert-base-uncased")
trainer = Trainer(
model=model, # the instantiated 🤗 Transformers model to be trained
args=training_args, # training arguments, defined above
train_dataset=train_dataset, # training dataset
eval_dataset=val_dataset # evaluation dataset
)
trainer.train()
Что я делал до сих пор
Подготовка набора данных для классификации последовательностей с помощью обзоров IMDb, и я уточняю настройки с тренером.
from pathlib import Path
def read_imdb_split(split_dir):
split_dir = Path(split_dir)
texts = []
labels = []
for label_dir in ["pos", "neg"]:
for text_file in (split_dir/label_dir).iterdir():
texts.append(text_file.read_text())
labels.append(0 if label_dir is "neg" else 1)
return texts, labels
train_texts, train_labels = read_imdb_split('aclImdb/train')
test_texts, test_labels = read_imdb_split('aclImdb/test')
from sklearn.model_selection import train_test_split
train_texts, val_texts, train_labels, val_labels = train_test_split(train_texts, train_labels, test_size=.2)
from transformers import DistilBertTokenizerFast
tokenizer = DistilBertTokenizerFast.from_pretrained('distilbert-base-uncased')
train_encodings = tokenizer(train_texts, truncation=True, padding=True)
val_encodings = tokenizer(val_texts, truncation=True, padding=True)
test_encodings = tokenizer(test_texts, truncation=True, padding=True)
import torch
class IMDbDataset(torch.utils.data.Dataset):
def __init__(self, encodings, labels):
self.encodings = encodings
self.labels = labels
def __getitem__(self, idx):
item = {key: torch.tensor(val[idx]) for key, val in self.encodings.items()}
item['labels'] = torch.tensor(self.labels[idx])
return item
def __len__(self):
return len(self.labels)
train_dataset = IMDbDataset(train_encodings, train_labels)
val_dataset = IMDbDataset(val_encodings, val_labels)
test_dataset = IMDbDataset(test_encodings, test_labels)
Комментарии:
1. Проверьте комментарий, я написал его с нуля сейчас, если у вас есть проблемы, мы можем их решить.
Ответ №1:
Что вы могли бы сделать в этой ситуации, так это повторить набор проверки(или набор тестов, если на то пошло) и вручную создать список y_true
и. y_pred
import torch
import torch.nn.functional as F
from sklearn import metrics
y_preds = []
y_trues = []
for index,val_text in enumerate(val_texts):
tokenized_val_text = tokenizer([val_text],
truncation=True,
padding=True,
return_tensor='pt')
logits = model(tokenized_val_text)
prediction = F.softmax(logits, dim=1)
y_pred = torch.argmax(prediction).numpy()
y_true = val_labels[index]
y_preds.append(y_pred)
y_trues.append(y_true)
Окончательно,
confusion_matrix = metrics.confusion_matrix(y_trues, y_preds, labels=["neg", "pos"]))
print(confusion_matrix)
Наблюдения:
- Выходными данными модели являются
logits
нормализованные , а не нормализованные вероятности. - Таким образом, мы применяем
softmax
первое измерение для преобразования в фактические вероятности (например0.2% class 0
,0.8% class 1
). - Мы применяем
.argmax()
операцию для получения индекса класса.