lab1
Objetivos¶
- Apresentar e utilizar o classificador k-nearest neighbours (kNN)
- Apresentar a técnica de separação de dados (treino e teste)
- Avaliar Aprendizagem do modelo
Começando¶
Vamos dar continuidade ao nosso estudo de aprendizagem de máquina, já vimos:
- Tudo começa, conhecendo os dados disponíveis.
- Como carregar um data frame
- Como visualizar os dados em gráficos (histograma, box plot, violin plot, matriz de confusão)
- Fizemos uma breve introdução sobre análise exploratória buscando correlacionar os dados para gerar informações.
Hoje, vamos seguir nossa jornada e finalizar nosso estudo aplicando a técnica de KNN.
k-Nearest Neighbors¶
O KNN(K vizinhos mais próximos) é considerado um dos algoritmos mais simples dentro da categoria de aprendizagem supervisionada sendo muito utilizado para problemas de classificação, porém também pode ser utilizado em problemas de regressão.
Problemas de classificação = Vale lembrar que em problemas de classificação não estamos interessados em valores exatos, queremos apenas saber se um dado pertence ou não a uma dada classe.
Uma intuição sobre o método¶
Para realizar a classificação o KNN calcula a distância objeto desconhecido (target) para todos os outros elementos, encontra os mais K vizinhos mais próximos faz uma contagem dos rótulos e considera que o objeto desconhecido pertence ao rótulo de maior contagem.
A imagem abaixo exemplifica o funcionamento, mas se ficou um pouco complicado de entender, rode o script python iknn.py e faça algumas simulações para compreender.
Bora lá!!¶
Vamos juntos realizar nosso primeiro projeto, do começo ao fim, de aprendizagem de máquina.
Definição do problema¶
A primeira coisa que precisamos fazer é a definição do problema. Neste primeiro caso vamos trabalhar com o mesmo dataset da última aula, dataset iris. Vamos desenvolver um sistema de machine learning capaz de classificar sua espécie com base nos dimensionais da pétala.
São 150 exemplares de flor de íris, pertencentes a três espécies diferentes: setosa, versicolor e virginica, sendo 50 amostras de cada espécie. Os atributos de largura e comprimento de sépala e largura e comprimento de pétala de cada flor fooram medidos manualmente.
Desafio 1¶
Do ponto de vista de machine learning, que problema é esse:
Aprendizado supervisionado ou não-supervisionado?
R:
Classificação ou regressão?
R:
### Sua resposta.....
# Inicializção das bibliotecas
%matplotlib inline
import pandas as pd
import matplotlib.pyplot as plt
# Caminho do arquivo
url = "https://archive.ics.uci.edu/ml/machine-learning-databases/iris/iris.data"
# Define o nome das colunas
header = ['sepal_length', 'sepal_width', 'petal_length', 'petal_width', 'species']
# Lê e carrega o arquivo para a memória
df = pd.read_csv(url, header=None, names=header)
# Retorna um trecho com as 5 primeiras linhas do dataframe
df.head()
sepal_length | sepal_width | petal_length | petal_width | species | |
---|---|---|---|---|---|
0 | 5.1 | 3.5 | 1.4 | 0.2 | Iris-setosa |
1 | 4.9 | 3.0 | 1.4 | 0.2 | Iris-setosa |
2 | 4.7 | 3.2 | 1.3 | 0.2 | Iris-setosa |
3 | 4.6 | 3.1 | 1.5 | 0.2 | Iris-setosa |
4 | 5.0 | 3.6 | 1.4 | 0.2 | Iris-setosa |
df.tail()
sepal_length | sepal_width | petal_length | petal_width | species | |
---|---|---|---|---|---|
145 | 6.7 | 3.0 | 5.2 | 2.3 | Iris-virginica |
146 | 6.3 | 2.5 | 5.0 | 1.9 | Iris-virginica |
147 | 6.5 | 3.0 | 5.2 | 2.0 | Iris-virginica |
148 | 6.2 | 3.4 | 5.4 | 2.3 | Iris-virginica |
149 | 5.9 | 3.0 | 5.1 | 1.8 | Iris-virginica |
# Mostra informações sobre o dataframe em si
df.info()
<class 'pandas.core.frame.DataFrame'> RangeIndex: 150 entries, 0 to 149 Data columns (total 5 columns): # Column Non-Null Count Dtype --- ------ -------------- ----- 0 sepal_length 150 non-null float64 1 sepal_width 150 non-null float64 2 petal_length 150 non-null float64 3 petal_width 150 non-null float64 4 species 150 non-null object dtypes: float64(4), object(1) memory usage: 6.0+ KB
# class distribution
print(df.groupby('species').size())
species Iris-setosa 50 Iris-versicolor 50 Iris-virginica 50 dtype: int64
Desafio 2¶
Aplique os métodos que achar conveniente (vimos algumas opções na última aula) para visualizar os dados de forma gráfica.
## Sua resposta e seus gráficos para análisar..
PARE!!!¶
A análise feita no desafio 2 é uma das etapas mais importantes. Caso você tenha pulado essa etapa, volte e faça suas análises.
Com essa etapa concluída, vamos criar um sub-dataset com os atributos que serão utilizados.
# Selecionando um sub-dataframe com os campos petal_length e petal_width,
# e outro com a variável de classes
entradas = df[['petal_length', 'petal_width']]
classes = df['species']
print(f"Formato das tabelas de dados {entradas.shape} e classes {classes.shape}")
Formato das tabelas de dados (150, 2) e classes (150,)
Dividindo os dados em conjunto de treinamento e de testes¶
Dividir nosso dataset em dois conjuntos de dados.
Treinamento - Representa 80% das amostras do conjunto de dados original,
Teste - com 20% das amostras
Vamos escolher aleatoriamente algumas amostras do conjunto original. Isto pode ser feito com Scikit-Learn usando a função train_test_split()
scikit-learn: pip3 install scikit-learn
# Separamos 20% para o teste
from sklearn.model_selection import train_test_split
entradas_treino, entradas_teste, classes_treino, classes_teste = train_test_split(entradas, classes, test_size=0.2)
print(f"Formato das tabelas de dados de treino {entradas_treino.shape} e teste {entradas_teste.shape}")
Formato das tabelas de dados de treino (120, 2) e teste (30, 2)
#Primeiras linhas do dataframe
entradas_treino.head()
petal_length | petal_width | |
---|---|---|
71 | 4.0 | 1.3 |
78 | 4.5 | 1.5 |
129 | 5.8 | 1.6 |
48 | 1.5 | 0.2 |
148 | 5.4 | 2.3 |
classes_treino.head()
species | |
---|---|
71 | Iris-versicolor |
78 | Iris-versicolor |
129 | Iris-virginica |
48 | Iris-setosa |
148 | Iris-virginica |
Chegou a hora de aplicar o modelo preditivo¶
Treinar um modelo no python é simples se usar o Scikit-Learn. Treinar um modelo no Scikit-Learn é simples: basta criar o classificador, e chamar o método fit().
Uma observação sobre a sintaxe dos classificadores do scikit-learn
- O método
fit(X,Y)
recebe uma matriz ou dataframe X onde cada linha é uma amostra de aprendizado, e um array Y contendo as saídas esperadas do classificador, seja na forma de texto ou de inteiros - O método
predict(X)
recebe uma matriz ou dataframe X onde cada linha é uma amostra de teste, retornando um array de classes
# Importa a biblioteca
from sklearn.neighbors import KNeighborsClassifier
# Cria o classificar KNN
k = 9
modelo = KNeighborsClassifier(n_neighbors=k)
# Cria o modelo de machine learning
modelo.fit(entradas_treino, classes_treino)
KNeighborsClassifier(n_neighbors=9)In a Jupyter environment, please rerun this cell to show the HTML representation or trust the notebook.
On GitHub, the HTML representation is unable to render, please try loading this page with nbviewer.org.
KNeighborsClassifier(n_neighbors=9)
Pronto!! bora testar se esta funcionando....
# Para obter as previsões, basta chamar o método predict()
classes_encontradas = modelo.predict(entradas_teste)
print("Predição: {}".format(classes_encontradas))
Predição: ['Iris-setosa' 'Iris-versicolor' 'Iris-setosa' 'Iris-setosa' 'Iris-versicolor' 'Iris-versicolor' 'Iris-versicolor' 'Iris-versicolor' 'Iris-setosa' 'Iris-setosa' 'Iris-versicolor' 'Iris-versicolor' 'Iris-virginica' 'Iris-setosa' 'Iris-versicolor' 'Iris-setosa' 'Iris-versicolor' 'Iris-versicolor' 'Iris-virginica' 'Iris-setosa' 'Iris-versicolor' 'Iris-virginica' 'Iris-versicolor' 'Iris-versicolor' 'Iris-setosa' 'Iris-virginica' 'Iris-setosa' 'Iris-versicolor' 'Iris-virginica' 'Iris-setosa']
# Para determinar a quantidade de acertos (acuracia)
from sklearn.metrics import accuracy_score
acertos = accuracy_score(classes_teste, classes_encontradas)
print("Acerto médio de classificação: ", acertos)
Acerto médio de classificação: 0.9333333333333333
Utilizando o modelo treinado com amostras fora do dataset¶
Vamos colocar alguns valores e ver a predição do classificador.
# Criamos um modelo utilizando duas entradas e uma saida, logo temos que passar duas entradas para o modelo faça a predição.
modelo.predict([[3.3, 3.2]])
/usr/local/lib/python3.10/dist-packages/sklearn/base.py:465: UserWarning: X does not have valid feature names, but KNeighborsClassifier was fitted with feature names warnings.warn(
array(['Iris-versicolor'], dtype=object)
Visualizando o modelo de forma gráfica¶
# Unificamos os dados de entrada e as classes de treino e teste em um daframe cada
df_treino = pd.concat((entradas_treino, classes_treino), axis=1)
novas_classes = pd.Series(classes_encontradas, name="species", index=entradas_teste.index)
df_teste = pd.concat((entradas_teste, novas_classes), axis=1)
import seaborn as sns
## Unificamos os dataframes de treinamento e teste em um novo DataFrame
# indicando a origem dos dados
novo_df = pd.concat((df_treino, df_teste), keys=['train', 'test'])
novo_df['origin'] = ''
novo_df.loc['train','origin'] = 'Treino'
novo_df.loc['test','origin'] = 'Teste'
# Usamos o scatterplot do seaborn, informando mudando o marcador de acordo com a origem do dado
sns.scatterplot(x='petal_length', y='petal_width', hue='species', style='origin', data=novo_df)
plt.show()
Desafio 3¶
Fizemos o treinamento para k=3, mude o valor de k e análise a acurácia do modelo.
Dica: Faça um loop for que varre um range de k, a saída pode ser armazenada em uma lista. No final do loop exiba em um gráfico.
### Implemente sua sua solução.....
#### Resposta loop for para diferntes k
k_range = list(range(1,26))
acertos = []
for k in k_range:
modelo = KNeighborsClassifier(n_neighbors=k)
modelo.fit(entradas_treino, classes_treino)
classes_encontradas = modelo.predict(entradas_teste)
acertos.append(accuracy_score(classes_teste, classes_encontradas))
plt.plot(k_range, acertos)
plt.xlabel('Valor de k do KNN')
plt.ylabel('Taxa de acertos')
plt.title('Taxa de acertos x valor de k do KNN')
plt.show()
Desafio 4¶
Refaça os notebook substituindo as entradas (variaveis independentes) e analise se o modelo obtido ficou melhor ou pior.
## implemente sua sua solução....
Desafio 5¶
Faça o treinamento utilizando um algoritimo de ML difernte do KNN, ao final compare os resultados e análise qual modelo obteve uma performance melhor.
# implemente sua solução......
### uma solução possivel usando o Arvore de decisão.
from sklearn.tree import DecisionTreeClassifier, plot_tree
# Criando o modelo de árvore de decisão
tree_classifier = DecisionTreeClassifier(max_depth=3)
tree_classifier.fit(entradas_treino, classes_treino)
# Avaliando o modelo
y_pred = tree_classifier.predict(entradas_teste)
accuracy = accuracy_score(classes_teste, y_pred)
print("Acurácia do modelo:", accuracy)
feature_names = ['sepal length (cm)', 'sepal width (cm)', 'petal length (cm)', 'petal width (cm)']
class_names = ['setosa', 'versicolor', 'virginica']
# Visualizando a árvore de decisão
plt.figure(figsize=(20,10))
plot_tree(tree_classifier, filled=True, feature_names=feature_names, class_names=class_names)
plt.title('Visualização da Árvore de Decisão do Dataset Iris')
plt.show()
Acurácia do modelo: 0.9