QuaPy/examples/10.one_vs_all.py

55 lines
2.4 KiB
Python
Raw Normal View History

2023-02-09 19:39:16 +01:00
import quapy as qp
2023-02-13 12:01:52 +01:00
from quapy.method.aggregative import MS2
from quapy.method.base import newOneVsAll
2023-02-09 19:39:16 +01:00
from quapy.model_selection import GridSearchQ
2023-02-14 11:14:38 +01:00
from quapy.protocol import UPP
2023-02-09 19:39:16 +01:00
from sklearn.linear_model import LogisticRegression
import numpy as np
"""
In this example, we will create a quantifier for tweet sentiment analysis considering three classes: negative, neutral,
and positive. We will use a one-vs-all approach using a binary quantifier for demonstration purposes.
"""
qp.environ['SAMPLE_SIZE'] = 100
2023-02-10 19:02:17 +01:00
qp.environ['N_JOBS'] = -1
2023-02-09 19:39:16 +01:00
"""
Any binary quantifier can be turned into a single-label quantifier by means of getOneVsAll function.
This function returns an instance of OneVsAll quantifier. Actually, it either returns the subclass OneVsAllGeneric
when the quantifier is an instance of BaseQuantifier, and it returns OneVsAllAggregative when the quantifier is
an instance of AggregativeQuantifier. Although OneVsAllGeneric works in all cases, using OneVsAllAggregative has
some additional advantages (namely, all the advantages that AggregativeQuantifiers enjoy, i.e., faster predictions
during evaluation).
"""
2023-02-13 12:01:52 +01:00
quantifier = newOneVsAll(MS2(LogisticRegression()))
2023-02-09 19:39:16 +01:00
print(f'the quantifier is an instance of {quantifier.__class__.__name__}')
# load a ternary dataset
train_modsel, val = qp.datasets.fetch_twitter('hcr', for_model_selection=True, pickle=True).train_test
"""
2023-02-14 11:14:38 +01:00
model selection: for this example, we are relying on the UPP protocol, i.e., a variant of the
2023-02-09 19:39:16 +01:00
artificial-prevalence protocol that generates random samples (100 in this case) for randomly picked priors
from the unit simplex. The priors are sampled using the Kraemer algorithm. Note this is in contrast to the
standard APP protocol, that instead explores a prefixed grid of prevalence values.
"""
param_grid = {
'binary_quantifier__classifier__C': np.logspace(-2,2,5), # classifier-dependent hyperparameter
'binary_quantifier__classifier__class_weight': ['balanced', None] # classifier-dependent hyperparameter
}
print('starting model selection')
2023-02-14 11:14:38 +01:00
model_selection = GridSearchQ(quantifier, param_grid, protocol=UPP(val), verbose=True, refit=False)
2023-02-10 19:02:17 +01:00
quantifier = model_selection.fit(train_modsel).best_model()
2023-02-09 19:39:16 +01:00
print('training on the whole training set')
train, test = qp.datasets.fetch_twitter('hcr', for_model_selection=False, pickle=True).train_test
quantifier.fit(train)
# evaluation
2023-02-14 11:14:38 +01:00
mae = qp.evaluation.evaluate(quantifier, protocol=UPP(test), error_metric='mae')
2023-02-09 19:39:16 +01:00
print(f'MAE = {mae:.4f}')