Lecture 4: \(k\)-Nearest Neighbours and SVM RBFs#

UBC 2024-25

If two things are similar, the thought of one will tend to trigger the thought of the other
– Aristotle

Imports, announcements, and LOs#

Imports#

import os
import sys

import IPython
import matplotlib.pyplot as plt
import numpy as np
import pandas as pd
from IPython.display import HTML

sys.path.append(os.path.join(os.path.abspath(".."), "code"))
import ipywidgets as widgets
import mglearn
from IPython.display import display
from ipywidgets import interact, interactive
from plotting_functions import *
from sklearn.dummy import DummyClassifier
from sklearn.model_selection import cross_validate, train_test_split
from utils import *

%matplotlib inline
pd.set_option("display.max_colwidth", 200)
import warnings

warnings.filterwarnings("ignore")
DATA_DIR = "../data/"



Learning outcomes#

From this lecture, you will be able to

  • explain the notion of similarity-based algorithms;

  • broadly describe how \(k\)-NNs use distances;

  • discuss the effect of using a small/large value of the hyperparameter \(k\) when using the \(k\)-NN algorithm;

  • describe the problem of curse of dimensionality;

  • explain the general idea of SVMs with RBF kernel;

  • broadly describe the relation of gamma and C hyperparameters of SVMs with the fundamental tradeoff.



Quick recap#

  • Why do we split the data?

  • What are the 4 types of data splits we discussed in the last lecture?

  • What are the benefits of cross-validation?

  • What is overfitting?

  • What’s the fundamental trade-off in supervised machine learning?

  • What is the golden rule of machine learning?



Important

If you want to run this notebook you will have to install ipywidgets. Follow the installation instructions here.

Motivation and distances [video]#

Analogy-based models#

  • Suppose you are given the following training examples with corresponding labels and are asked to label a given test example.

../../_images/knn-motivation.png

source

  • An intuitive way to classify the test example is by finding the most “similar” example(s) from the training set and using that label for the test example.

Analogy-based algorithms in practice#

General idea of \(k\)-nearest neighbours algorithm#

  • Consider the following toy dataset with two classes.

    • blue circles \(\rightarrow\) class 0

    • red triangles \(\rightarrow\) class 1

    • green stars \(\rightarrow\) test examples

X, y = mglearn.datasets.make_forge()
X_test = np.array([[8.2, 3.66214339], [9.9, 3.2], [11.2, 0.5]])
plot_train_test_points(X, y, X_test)
../../_images/378038ef318980c395a920c8ce4c64ec3d8119a836ab16f0b4679f88da81ad16.png
  • Given a new data point, predict the class of the data point by finding the “closest” data point in the training set, i.e., by finding its “nearest neighbour” or majority vote of nearest neighbours.

import matplotlib
import panel as pn
from panel import widgets
from panel.interact import interact

pn.extension()
def f(n_neighbors):
    plt.clf()
    fig = plt.figure(figsize=(6, 4))
    plot_knn_clf(X, y, X_test, n_neighbors=n_neighbors)
    plt.close()
    return pn.pane.Matplotlib(fig, tight=True)


n_neighbors_selector = pn.widgets.IntSlider(
    name="n_neighbors", start=1, end=10, value=1
)
# interact(f, n_neighbors=n_neighbors_selector)
interactive_plot = interact(f, n_neighbors=n_neighbors_selector).embed(max_opts=10)
interactive_plot
n_neighbors 1
n_neighbors 10
n_neighbors 9
n_neighbors 8
n_neighbors 7
n_neighbors 6
n_neighbors 5
n_neighbors 4
n_neighbors 3
n_neighbors 2
n_neighbors 1
<Figure size 640x480 with 0 Axes>

Geometric view of tabular data and dimensions#

  • To understand analogy-based algorithms it’s useful to think of data as points in a high dimensional space.

  • Our X represents the problem in terms of relevant features (\(d\)) with one dimension for each feature (column).

  • Examples are points in a \(d\)-dimensional space.

How many dimensions (features) are there in the cities data?

cities_df = pd.read_csv(DATA_DIR + "canada_usa_cities.csv")
X_cities = cities_df[["longitude", "latitude"]]
y_cities = cities_df["country"]
mglearn.discrete_scatter(X_cities.iloc[:, 0], X_cities.iloc[:, 1], y_cities)
plt.xlabel("longitude")
plt.ylabel("latitude");
../../_images/794b59a254668f7cb10439648922aa1ddca815562238420f748dd358ddaa96ab.png
  • Recall the Spotify Song Attributes dataset from homework 2.

  • How many dimensions (features) we used in the homework?

spotify_df = pd.read_csv(DATA_DIR + "spotify.csv", index_col=0)
X_spotify = spotify_df.drop(columns=["target", "song_title", "artist"])
print("The number of features in the Spotify dataset: %d" % X_spotify.shape[1])
X_spotify.head()
The number of features in the Spotify dataset: 13
acousticness danceability duration_ms energy instrumentalness key liveness loudness mode speechiness tempo time_signature valence
0 0.0102 0.833 204600 0.434 0.021900 2 0.1650 -8.795 1 0.4310 150.062 4.0 0.286
1 0.1990 0.743 326933 0.359 0.006110 1 0.1370 -10.401 1 0.0794 160.083 4.0 0.588
2 0.0344 0.838 185707 0.412 0.000234 2 0.1590 -7.148 1 0.2890 75.044 4.0 0.173
3 0.6040 0.494 199413 0.338 0.510000 5 0.0922 -15.236 1 0.0261 86.468 4.0 0.230
4 0.1800 0.678 392893 0.561 0.512000 5 0.4390 -11.648 0 0.0694 174.004 4.0 0.904

Dimensions in ML problems#

In ML, usually we deal with high dimensional problems where examples are hard to visualize.

  • \(d \approx 20\) is considered low dimensional

  • \(d \approx 1000\) is considered medium dimensional

  • \(d \approx 100,000\) is considered high dimensional

Feature vectors#

Feature vector

is composed of feature values associated with an example.

Some example feature vectors are shown below.

print(
    "An example feature vector from the cities dataset: %s"
    % (X_cities.iloc[0].to_numpy())
)
print(
    "An example feature vector from the Spotify dataset: \n%s"
    % (X_spotify.iloc[0].to_numpy())
)
An example feature vector from the cities dataset: [-130.0437   55.9773]
An example feature vector from the Spotify dataset: 
[ 1.02000e-02  8.33000e-01  2.04600e+05  4.34000e-01  2.19000e-02
  2.00000e+00  1.65000e-01 -8.79500e+00  1.00000e+00  4.31000e-01
  1.50062e+02  4.00000e+00  2.86000e-01]

Similarity between examples#

Let’s take 2 points (two feature vectors) from the cities dataset.

two_cities = X_cities.sample(2, random_state=120)
two_cities
longitude latitude
69 -104.8253 38.8340
35 -112.0741 33.4484

The two sampled points are shown as big black circles.

mglearn.discrete_scatter(
    X_cities.iloc[:, 0], X_cities.iloc[:, 1], y_cities, s=8, alpha=0.3
)
mglearn.discrete_scatter(
    two_cities.iloc[:, 0], two_cities.iloc[:, 1], markers="o", c="k", s=18
);
../../_images/bceb64932319e4ac053f7bb571cae9d7c232d03a1f06de2a4340dbd9c2b7367a.png

Distance between feature vectors#

  • For the cities at the two big circles, what is the distance between them?

  • A common way to calculate the distance between vectors is calculating the Euclidean distance.

  • The euclidean distance between vectors \(u = <u_1, u_2, \dots, u_n>\) and \(v = <v_1, v_2, \dots, v_n>\) is defined as:

\[distance(u, v) = \sqrt{\sum_{i =1}^{n} (u_i - v_i)^2}\]

Euclidean distance#

two_cities
longitude latitude
69 -104.8253 38.8340
35 -112.0741 33.4484
  • Subtract the two cities

  • Square the difference

  • Sum them up

  • Take the square root

# Subtract the two cities
print("Subtract the cities: \n%s\n" % (two_cities.iloc[1] - two_cities.iloc[0]))

# Squared sum of the difference
print(
    "Sum of squares: %0.4f" % (np.sum((two_cities.iloc[1] - two_cities.iloc[0]) ** 2))
)

# Take the square root
print(
    "Euclidean distance between cities: %0.4f"
    % (np.sqrt(np.sum((two_cities.iloc[1] - two_cities.iloc[0]) ** 2)))
)
Subtract the cities: 
longitude   -7.2488
latitude    -5.3856
dtype: float64

Sum of squares: 81.5498
Euclidean distance between cities: 9.0305
two_cities
longitude latitude
69 -104.8253 38.8340
35 -112.0741 33.4484
# Euclidean distance using sklearn
from sklearn.metrics.pairwise import euclidean_distances

euclidean_distances(two_cities)
array([[0.        , 9.03049217],
       [9.03049217, 0.        ]])

Note: scikit-learn supports a number of other distance metrics.

Finding the nearest neighbour#

  • Let’s look at distances from all cities to all other cities

dists = euclidean_distances(X_cities)
np.fill_diagonal(dists, np.inf)
dists.shape
(209, 209)
pd.DataFrame(dists)
0 1 2 3 4 5 6 7 8 9 ... 199 200 201 202 203 204 205 206 207 208
0 inf 4.955113 9.869531 10.106452 10.449666 19.381676 28.366626 33.283857 33.572105 36.180388 ... 9.834455 58.807684 16.925593 56.951696 59.384127 58.289799 64.183423 52.426410 58.033459 51.498562
1 4.955113 inf 14.677579 14.935802 15.305346 24.308448 33.200978 38.082949 38.359992 40.957919 ... 14.668787 63.533498 21.656349 61.691640 64.045304 63.032656 68.887343 57.253724 62.771969 56.252160
2 9.869531 14.677579 inf 0.334411 0.808958 11.115406 20.528403 25.525757 25.873103 28.479109 ... 0.277381 51.076798 10.783789 49.169693 51.934205 50.483751 56.512897 44.235152 50.249720 43.699224
3 10.106452 14.935802 0.334411 inf 0.474552 10.781004 20.194002 25.191396 25.538702 28.144750 ... 0.275352 50.743133 10.480249 48.836189 51.599860 50.150395 56.179123 43.904226 49.916254 43.365623
4 10.449666 15.305346 0.808958 0.474552 inf 10.306500 19.719500 24.716985 25.064200 27.670344 ... 0.675814 50.269880 10.051472 48.363192 51.125476 49.677629 55.705696 43.435186 49.443317 42.892477
... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ... ...
204 58.289799 63.032656 50.483751 50.150395 49.677629 39.405415 30.043890 25.057003 24.746328 22.127878 ... 50.333340 0.873356 41.380643 1.345136 3.373031 inf 6.102435 6.957987 0.316363 6.800190
205 64.183423 68.887343 56.512897 56.179123 55.705696 45.418031 36.031385 31.032874 30.709185 28.088948 ... 56.358333 5.442806 47.259286 7.369875 5.108681 6.102435 inf 12.950733 6.303916 12.819584
206 52.426410 57.253724 44.235152 43.904226 43.435186 33.258427 24.059863 19.187663 18.932124 16.380495 ... 44.100248 7.767852 35.637982 5.930561 9.731583 6.957987 12.950733 inf 6.837848 3.322755
207 58.033459 62.771969 50.249720 49.916254 49.443317 39.167214 29.799983 24.810368 24.497386 21.878183 ... 50.098326 0.930123 41.121628 1.082749 3.286821 0.316363 6.303916 6.837848 inf 6.555740
208 51.498562 56.252160 43.699224 43.365623 42.892477 32.612755 23.244592 18.256813 17.946783 15.328953 ... 43.546610 7.378764 34.596810 5.473691 8.568009 6.800190 12.819584 3.322755 6.555740 inf

209 rows × 209 columns

Let’s look at the distances between City 0 and some other cities.

print("Feature vector for city 0: \n%s\n" % (X_cities.iloc[0]))
print("Distances from city 0 to the first 5 cities: %s" % (dists[0][:5]))
# We can find the closest city with `np.argmin`:
print(
    "The closest city from city 0 is: %d \n\nwith feature vector: \n%s"
    % (np.argmin(dists[0]), X_cities.iloc[np.argmin(dists[0])])
)
Feature vector for city 0: 
longitude   -130.0437
latitude      55.9773
Name: 0, dtype: float64

Distances from city 0 to the first 5 cities: [        inf  4.95511263  9.869531   10.10645223 10.44966612]
The closest city from city 0 is: 81 

with feature vector: 
longitude   -129.9912
latitude      55.9383
Name: 81, dtype: float64

Ok, so the closest city to City 0 is City 81.

Question#

  • Why did we set the diagonal entries to infinity before finding the closest city?

Finding the distances to a query point#

We can also find the distances to a new “test” or “query” city:

# Let's find a city that's closest to the a query city
query_point = [[-80, 25]]

dists = euclidean_distances(X_cities, query_point)
dists[0:10]
array([[58.85545875],
       [63.80062924],
       [49.30530902],
       [49.01473536],
       [48.60495488],
       [39.96834506],
       [32.92852376],
       [29.53520104],
       [29.52881619],
       [27.84679073]])
# The query point is closest to
print(
    "The query point %s is closest to the city with index %d and the distance between them is: %0.4f"
    % (query_point, np.argmin(dists), dists[np.argmin(dists)])
)
The query point [[-80, 25]] is closest to the city with index 72 and the distance between them is: 0.7982



\(k\)-Nearest Neighbours (\(k\)-NNs) [video]#

small_cities = cities_df.sample(30, random_state=90)
one_city = small_cities.sample(1, random_state=44)
small_train_df = pd.concat([small_cities, one_city]).drop_duplicates(keep=False)
X_small_cities = small_train_df.drop(columns=["country"]).to_numpy()
y_small_cities = small_train_df["country"].to_numpy()
test_point = one_city[["longitude", "latitude"]].to_numpy()
plot_train_test_points(
    X_small_cities,
    y_small_cities,
    test_point,
    class_names=["Canada", "USA"],
    test_format="circle",
)
../../_images/f7074e556553cdb9b141c772336db4ad8b5298ad0f3dbbe86847a0288a9f9c2f.png
  • Given a new data point, predict the class of the data point by finding the “closest” data point in the training set, i.e., by finding its “nearest neighbour” or majority vote of nearest neighbours.

Suppose we want to predict the class of the black point.

  • An intuitive way to do this is predict the same label as the “closest” point (\(k = 1\)) (1-nearest neighbour)

  • We would predict a target of USA in this case.

plot_knn_clf(
    X_small_cities,
    y_small_cities,
    test_point,
    n_neighbors=1,
    class_names=["Canada", "USA"],
    test_format="circle",
)
n_neighbors 1
../../_images/eb872a80275a515c711e6a4716e2ac6b0f1ecec9b7528332155a96d7863869da.png

How about using \(k > 1\) to get a more robust estimate?

  • For example, we could also use the 3 closest points (k = 3) and let them vote on the correct class.

  • The Canada class would win in this case.

plot_knn_clf(
    X_small_cities,
    y_small_cities,
    test_point,
    n_neighbors=3,
    class_names=["Canada", "USA"],
    test_format="circle",
)
n_neighbors 3
../../_images/9dc0d7c55999b35759abe15c6cb225e3f5e452279d866ab37cafefba1751bde8.png
from sklearn.neighbors import KNeighborsClassifier

k_values = [1, 3]

for k in k_values:
    neigh = KNeighborsClassifier(n_neighbors=k)
    neigh.fit(X_small_cities, y_small_cities)
    print(
        "Prediction of the black dot with %d neighbours: %s"
        % (k, neigh.predict(test_point))
    )
Prediction of the black dot with 1 neighbours: ['USA']
Prediction of the black dot with 3 neighbours: ['Canada']

Choosing n_neighbors#

  • The primary hyperparameter of the model is n_neighbors (\(k\)) which decides how many neighbours should vote during prediction?

  • What happens when we play around with n_neighbors?

  • Are we more likely to overfit with a low n_neighbors or a high n_neighbors?

  • Let’s examine the effect of the hyperparameter on our cities data.

X = cities_df.drop(columns=["country"])
y = cities_df["country"]

# split into train and test sets
X_train, X_test, y_train, y_test = train_test_split(
    X, y, test_size=0.1, random_state=123
)
k = 1
knn1 = KNeighborsClassifier(n_neighbors=k)
scores = cross_validate(knn1, X_train, y_train, return_train_score=True)
pd.DataFrame(scores)
fit_time score_time test_score train_score
0 0.001741 0.004357 0.710526 1.0
1 0.001377 0.003219 0.684211 1.0
2 0.001856 0.003624 0.842105 1.0
3 0.001549 0.003651 0.702703 1.0
4 0.001433 0.003208 0.837838 1.0
k = 100
knn100 = KNeighborsClassifier(n_neighbors=k)
scores = cross_validate(knn100, X_train, y_train, return_train_score=True)
pd.DataFrame(scores)
fit_time score_time test_score train_score
0 0.001403 0.043724 0.605263 0.600000
1 0.001580 0.004092 0.605263 0.600000
2 0.001272 0.003900 0.605263 0.600000
3 0.001450 0.004129 0.594595 0.602649
4 0.001361 0.004387 0.594595 0.602649
plot_knn_decision_boundaries(X_train, y_train, k_values=[1, 11, 100])
../../_images/67c1f373d162419b445bbdcc235311e7e1a8ac03aa45a0bc920c03e9d04c268b.png

How to choose n_neighbors?#

  • n_neighbors is a hyperparameter

  • We can use hyperparameter optimization to choose n_neighbors.

results_dict = {
    "n_neighbors": [],
    "mean_train_score": [],
    "mean_cv_score": [],
    "std_cv_score": [],
    "std_train_score": [],
}
param_grid = {"n_neighbors": np.arange(1, 50, 5)}

for k in param_grid["n_neighbors"]:
    knn = KNeighborsClassifier(n_neighbors=k)
    scores = cross_validate(knn, X_train, y_train, return_train_score=True)
    results_dict["n_neighbors"].append(k)

    results_dict["mean_cv_score"].append(np.mean(scores["test_score"]))
    results_dict["mean_train_score"].append(np.mean(scores["train_score"]))
    results_dict["std_cv_score"].append(scores["test_score"].std())
    results_dict["std_train_score"].append(scores["train_score"].std())

results_df = pd.DataFrame(results_dict)
results_df = results_df.set_index("n_neighbors")
results_df
mean_train_score mean_cv_score std_cv_score std_train_score
n_neighbors
1 1.000000 0.755477 0.069530 0.000000
6 0.831135 0.792603 0.046020 0.013433
11 0.819152 0.802987 0.041129 0.011336
16 0.801863 0.782219 0.074141 0.008735
21 0.777934 0.766430 0.062792 0.016944
26 0.755364 0.723613 0.061937 0.025910
31 0.743391 0.707681 0.057646 0.030408
36 0.728777 0.707681 0.064452 0.021305
41 0.706128 0.681223 0.061241 0.018310
46 0.694155 0.660171 0.093390 0.018178
results_df[["mean_train_score", "mean_cv_score"]].plot();
../../_images/51fcff1359342b3f6d48602c5ed75b00681de323ced8b614bb8d437d98f73a06.png
best_n_neighbours = results_df.idxmax()["mean_cv_score"]
best_n_neighbours
11

Let’s try our best model on test data.

knn = KNeighborsClassifier(n_neighbors=best_n_neighbours)
knn.fit(X_train, y_train)
print("Test accuracy: %0.3f" % (knn.score(X_test, y_test)))
Test accuracy: 0.905

Seems like we got lucky with the test set here.



❓❓ Questions for you#

(iClicker) Exercise 4.1#

iClicker cloud join link: https://join.iclicker.com/VYFJ

Select all of the following statements which are TRUE.

  • (A) Analogy-based models find examples from the test set that are most similar to the query example we are predicting.

  • (B) Euclidean distance will always have a non-negative value.

  • (C) With \(k\)-NN, setting the hyperparameter \(k\) to larger values typically reduces training error.

  • (D) Similar to decision trees, \(k\)-NNs finds a small set of good features.

  • (E) In \(k\)-NN, with \(k > 1\), the classification of the closest neighbour to the test example always contributes the most to the prediction.

Break (5 min)#



More on \(k\)-NNs [video]#

Other useful arguments of KNeighborsClassifier#

  • weights \(\rightarrow\) When predicting label, you can assign higher weight to the examples which are closer to the query example.

  • Exercise for you: Play around with this argument. Do you get a better validation score?

Regression with \(k\)-nearest neighbours (\(k\)-NNs)#

  • Can we solve regression problems with \(k\)-nearest neighbours algorithm?

  • In \(k\)-NN regression we take the average of the \(k\)-nearest neighbours.

  • We can also have weighted regression.

See an example of regression in the lecture notes.

mglearn.plots.plot_knn_regression(n_neighbors=1)
../../_images/c7b109d936a1c9013c56b7e4f33dd169990f7c129473dcf5744b94104d7e7faa.png
mglearn.plots.plot_knn_regression(n_neighbors=3)
../../_images/d42fddaf7012e6557b8792cef1aba2d591b0c7ca6cabd345c50105176361c8ad.png

Pros of \(k\)-NNs for supervised learning#

  • Easy to understand, interpret.

  • Simple hyperparameter \(k\) (n_neighbors) controlling the fundamental tradeoff.

  • Can learn very complex functions given enough data.

  • Lazy learning: Takes no time to fit

Cons of \(k\)-NNs for supervised learning#

  • Can be potentially be VERY slow during prediction time, especially when the training set is very large.

  • Often not that great test accuracy compared to the modern approaches.

  • It does not work well on datasets with many features or where most feature values are 0 most of the time (sparse datasets).

Attention

For regular \(k\)-NN for supervised learning (not with sparse matrices), you should scale your features. We’ll be looking into it soon.

(Optional) Parametric vs non parametric#

  • You might see a lot of definitions of these terms.

  • A simple way to think about this is:

    • do you need to store at least \(O(n)\) worth of stuff to make predictions? If so, it’s non-parametric.

  • Non-parametric example: \(k\)-NN is a classic example of non-parametric models.

  • Parametric example: decision stump

  • If you want to know more about this, find some reading material here, here, and here.

  • By the way, the terms “parametric” and “non-paramteric” are often used differently by statisticians, see here for more…

Note

\(\mathcal{O}(n)\) is referred to as big \(\mathcal{O}\) notation. It tells you how fast an algorithm is or how much storage space it requires. For example, in simple terms, if you have \(n\) examples and you need to store them all you can say that the algorithm requires \(\mathcal{O}(n)\) worth of stuff.

Curse of dimensionality#

  • Affects all learners but especially bad for nearest-neighbour.

  • \(k\)-NN usually works well when the number of dimensions \(d\) is small but things fall apart quickly as \(d\) goes up.

  • If there are many irrelevant attributes, \(k\)-NN is hopelessly confused because all of them contribute to finding similarity between examples.

  • With enough irrelevant attributes the accidental similarity swamps out meaningful similarity and \(k\)-NN is no better than random guessing.

from sklearn.datasets import make_classification

nfeats_accuracy = {"nfeats": [], "dummy_valid_accuracy": [], "KNN_valid_accuracy": []}
for n_feats in range(4, 2000, 100):
    X, y = make_classification(n_samples=2000, n_features=n_feats, n_classes=2)
    X_train, X_test, y_train, y_test = train_test_split(
        X, y, test_size=0.2, random_state=123
    )
    dummy = DummyClassifier(strategy="most_frequent")
    dummy_scores = cross_validate(dummy, X_train, y_train, return_train_score=True)

    knn = KNeighborsClassifier()
    scores = cross_validate(knn, X_train, y_train, return_train_score=True)
    nfeats_accuracy["nfeats"].append(n_feats)
    nfeats_accuracy["KNN_valid_accuracy"].append(np.mean(scores["test_score"]))
    nfeats_accuracy["dummy_valid_accuracy"].append(np.mean(dummy_scores["test_score"]))
pd.DataFrame(nfeats_accuracy)
nfeats dummy_valid_accuracy KNN_valid_accuracy
0 4 0.508750 0.943125
1 104 0.500625 0.792500
2 204 0.501250 0.700000
3 304 0.500000 0.703125
4 404 0.501875 0.605000
5 504 0.501875 0.599375
6 604 0.506250 0.663750
7 704 0.500625 0.616250
8 804 0.500625 0.638125
9 904 0.501250 0.610000
10 1004 0.505000 0.580625
11 1104 0.511875 0.613125
12 1204 0.505625 0.623125
13 1304 0.500625 0.591250
14 1404 0.503125 0.583750
15 1504 0.501875 0.603125
16 1604 0.508750 0.600625
17 1704 0.507500 0.575000
18 1804 0.504375 0.588750
19 1904 0.511875 0.601875



Support Vector Machines (SVMs) with RBF kernel [video]#

  • Very high-level overview

  • Our goals here are

    • Use scikit-learn’s SVM model.

    • Broadly explain the notion of support vectors.

    • Broadly explain the similarities and differences between \(k\)-NNs and SVM RBFs.

    • Explain how C and gamma hyperparameters control the fundamental tradeoff.

(Optional) RBF stands for radial basis functions. We won’t go into what it means in this video. Refer to this video if you want to know more.

Overview#

  • Another popular similarity-based algorithm is Support Vector Machines with RBF Kernel (SVM RBFs)

  • Superficially, SVM RBFs are more like weighted \(k\)-NNs.

    • The decision boundary is defined by a set of positive and negative examples and their weights together with their similarity measure.

    • A test example is labeled positive if on average it looks more like positive examples than the negative examples.

  • The primary difference between \(k\)-NNs and SVM RBFs is that

    • Unlike \(k\)-NNs, SVM RBFs only remember the key examples (support vectors).

    • SVMs use a different similarity metric which is called a “kernel”. A popular kernel is Radial Basis Functions (RBFs)

    • They usually perform better than \(k\)-NNs!

Let’s explore SVM RBFs#

Let’s try SVMs on the cities dataset.

mglearn.discrete_scatter(X_cities.iloc[:, 0], X_cities.iloc[:, 1], y_cities)
plt.xlabel("longitude")
plt.ylabel("latitude")
plt.legend(loc=1);
../../_images/8eb8c1157782d7b864a6c0fc32de02572c82de68d87d20853a428324a4e0480d.png
X_train, X_test, y_train, y_test = train_test_split(
    X_cities, y_cities, test_size=0.2, random_state=123
)
knn = KNeighborsClassifier(n_neighbors=best_n_neighbours)
scores = cross_validate(knn, X_train, y_train, return_train_score=True)
print("Mean validation score %0.3f" % (np.mean(scores["test_score"])))
pd.DataFrame(scores)
Mean validation score 0.803
fit_time score_time test_score train_score
0 0.002012 0.005469 0.794118 0.819549
1 0.001509 0.003509 0.764706 0.819549
2 0.001575 0.003381 0.727273 0.850746
3 0.001345 0.003356 0.787879 0.828358
4 0.001529 0.003178 0.939394 0.783582
from sklearn.svm import SVC

svm = SVC(gamma=0.01)  # Ignore gamma for now
scores = cross_validate(svm, X_train, y_train, return_train_score=True)
print("Mean validation score %0.3f" % (np.mean(scores["test_score"])))
pd.DataFrame(scores)
Mean validation score 0.820
fit_time score_time test_score train_score
0 0.002415 0.001406 0.823529 0.842105
1 0.001740 0.001411 0.823529 0.842105
2 0.001665 0.001228 0.727273 0.858209
3 0.001649 0.001233 0.787879 0.843284
4 0.001775 0.001244 0.939394 0.805970

Decision boundary of SVMs#

  • We can think of SVM with RBF kernel as “smooth KNN”.

fig, axes = plt.subplots(1, 2, figsize=(16, 5))

for clf, ax in zip([knn, svm], axes):
    clf.fit(X_train, y_train)
    mglearn.plots.plot_2d_separator(
        clf, X_train.to_numpy(), fill=True, eps=0.5, ax=ax, alpha=0.4
    )
    mglearn.discrete_scatter(X_train.iloc[:, 0], X_train.iloc[:, 1], y_train, ax=ax)
    ax.set_title(clf)
    ax.set_xlabel("longitude")
    ax.set_ylabel("latitude")
../../_images/bb11ab9599ff3789325f1eae89dbd5bc5952baa0d9e5dfbc7d67bc24d140cb34.png

Support vectors#

  • Each training example either is or isn’t a “support vector”.

    • This gets decided during fit.

  • Main insight: the decision boundary only depends on the support vectors.

  • Let’s look at the support vectors.

from sklearn.datasets import make_blobs

n = 20
n_classes = 2
X_toy, y_toy = make_blobs(
    n_samples=n, centers=n_classes, random_state=300
)  # Let's generate some fake data
mglearn.discrete_scatter(X_toy[:, 0], X_toy[:, 1], y_toy)
plt.xlabel("Feature 0")
plt.ylabel("Feature 1")
svm = SVC(kernel="rbf", C=10, gamma=0.1).fit(X_toy, y_toy)
mglearn.plots.plot_2d_separator(svm, X_toy, fill=True, eps=0.5, alpha=0.4)
../../_images/199f40be46190b9e1ab36b6206bc13d51b551a77c600c1a5d9d2cb3797b34902.png
svm.support_
array([ 3,  8,  9, 14, 19,  1,  4,  6, 17], dtype=int32)
plot_support_vectors(svm, X_toy, y_toy)
../../_images/49dc6120b80e581c6e16e629922c43b43b8f2f739b1a13fde03820ebbbb7e3e9.png

The support vectors are the bigger points in the plot above.

Hyperparameters of SVM#

  • Key hyperparameters of rbf SVM are

    • gamma

    • C

  • We are not equipped to understand the meaning of these parameters at this point but you are expected to describe their relation to the fundamental tradeoff.

See scikit-learn’s explanation of RBF SVM parameters.

Relation of gamma and the fundamental trade-off#

  • gamma controls the complexity (fundamental trade-off), just like other hyperparameters we’ve seen.

    • larger gamma \(\rightarrow\) more complex

    • smaller gamma \(\rightarrow\) less complex

gamma = [0.001, 0.01, 0.1, 1.0, 10.0]
plot_svc_gamma(
    gamma,
    X_train.to_numpy(),
    y_train.to_numpy(),
    x_label="longitude",
    y_label="latitude",
)
../../_images/6207e78a60bdde737dd8324420f5d4037b95ed8ce52472f259645dc5bc914b06.png

Relation of C and the fundamental trade-off#

  • C also affects the fundamental tradeoff

    • larger C \(\rightarrow\) more complex

    • smaller C \(\rightarrow\) less complex

C = [0.1, 1.0, 100.0, 1000.0, 100000.0]
plot_svc_C(
    C, X_train.to_numpy(), y_train.to_numpy(), x_label="longitude", y_label="latitude"
)
../../_images/480a82ff0278c3518b82157792222c812ca48c1829ee0c857f01f8de8722c18b.png

Search over multiple hyperparameters#

  • So far you have seen how to carry out search over a hyperparameter

  • In the above case the best training error is achieved by the most complex model (large gamma, large C).

  • Best validation error requires a hyperparameter search to balance the fundamental tradeoff.

SVM Regressor#

  • Similar to KNNs, you can use SVMs for regression problems as well.

  • See sklearn.svm.SVR for more details.

❓❓ Questions for you#

(iClicker) Exercise 4.2#

iClicker cloud join link: https://join.iclicker.com/VYFJ

Select all of the following statements which are TRUE.

  • (A) \(k\)-NN may perform poorly in high-dimensional space (say, d > 1000).

  • (B) In sklearn’s SVC classifier, large values of gamma tend to result in higher training score but probably lower validation score.

  • (C) If we increase both gamma and C, we can’t be certain if the model becomes more complex or less complex.



Playground#

In this interactive playground, you will investigate how various algorithms create decision boundaries to distinguish between Iris flower species using their sepal length and width as features. By adjusting the parameters, you can observe how the decision boundaries change, which can result in either overfitting (where the model fits the training data too closely) or underfitting (where the model is too simplistic).

  • With k-Nearest Neighbours (\(k\)-NN), you’ll determine how many neighboring flowers to consult. Should we rely on a single nearest neighbor? Or should we consider a wider group?

  • With Support Vector Machine (SVM) using the RBF kernel, you’ll tweak the hyperparameters C and gamma to explore the tightrope walk between overly complex boundaries (that might overfit) and overly broad ones (that might underfit).

  • With Decision trees, you’ll observe the effect of max_depth on the decision boundary.

Observe the process of crafting and refining decision boundaries, one parameter at a time! Be sure to take breaks to reflect on the results you are observing.

from matplotlib.figure import Figure

from sklearn.model_selection import train_test_split
from sklearn.datasets import load_iris
from ipywidgets import interact, FloatLogSlider, IntSlider
import mglearn


# Load dataset and preprocessing
iris = load_iris(as_frame=True)
iris_df = iris.data
iris_df['species'] = iris.target
iris_df = iris_df[iris_df['species'] > 0]
X, y = iris_df[['sepal length (cm)', 'sepal width (cm)']], iris_df['species']
X_train, X_test, y_train, y_test = train_test_split(X, y, test_size=0.4, random_state=123)


# Common plot settings
def plot_results(model, X_train, y_train, title, ax):
    mglearn.plots.plot_2d_separator(model, X_train.values, fill=True, alpha=0.4, ax=ax);
    mglearn.discrete_scatter(
        X_train["sepal length (cm)"], X_train["sepal width (cm)"], y_train, s=6, ax=ax
    );
    ax.set_xlabel("sepal length (cm)", fontsize=12);
    ax.set_ylabel("sepal width (cm)", fontsize=12);
    train_score = np.round(model.score(X_train.values, y_train), 2)
    test_score = np.round(model.score(X_test.values, y_test), 2)
    ax.set_title(
        f"{title}\n train score = {train_score}\ntest score = {test_score}", fontsize=8
    );
    pass


# Widgets for SVM, k-NN, and Decision Tree
c_widget = pn.widgets.FloatSlider(
    value=1.0, start=1, end=5, step=0.1, name="C (log scale)"
)
gamma_widget = pn.widgets.FloatSlider(
    value=1.0, start=-3, end=5, step=0.1, name="Gamma (log scale)"
)
n_neighbors_widget = pn.widgets.IntSlider(
    start=1, end=40, step=1, value=5, name="n_neighbors"
)
max_depth_widget = pn.widgets.IntSlider(
    start=1, end=20, step=1, value=3, name="max_depth"
)


# The update function to create the plots
def update_plots(c, gamma=1.0, n_neighbors=5, max_depth=3):
    c_log = round(10**c, 2)  # Transform C to logarithmic scale
    gamma_log = round(10**gamma, 2)   # Transform Gamma to logarithmic scale

    fig = Figure(figsize=(8, 2))
    axes = fig.subplots(1, 3)

    models = [
        SVC(C=c_log, gamma=gamma_log, random_state=42),
        KNeighborsClassifier(n_neighbors=n_neighbors),
        DecisionTreeClassifier(max_depth=max_depth, random_state=42),
    ]
    titles = [
        f"SVM (C={c_log}, gamma={gamma_log})",
        f"k-NN (n_neighbors={n_neighbors})",
        f"Decision Tree (max_depth={max_depth})",
    ]
    for model, title, ax in zip(models, titles, axes):
        model.fit(X_train.values, y_train)
        plot_results(model, X_train, y_train, title, ax);
    # print(c, gamma, n_neighbors, max_depth)
    return pn.pane.Matplotlib(fig, tight=True);


# Bind the function to the panel widgets
interactive_plot = pn.bind(
    update_plots,
    c=c_widget.param.value_throttled,
    gamma=gamma_widget.param.value_throttled,
    n_neighbors=n_neighbors_widget.param.value_throttled,
    max_depth=max_depth_widget.param.value_throttled,
)

# Layout the widgets and the plot
dashboard = pn.Column(
    pn.Row(c_widget, n_neighbors_widget),
    pn.Row(gamma_widget, max_depth_widget),
    interactive_plot,
)

# Display the interactive dashboard
dashboard





Summary#

  • We have KNNs and SVMs as new supervised learning techniques in our toolbox.

  • These are analogy-based learners and the idea is to assign nearby points the same label.

  • Unlike decision trees, all features are equally important.

  • Both can be used for classification or regression (much like the other methods we’ve seen).

Coming up:#

Lingering questions:

  • Are we ready to do machine learning on real-world datasets?

  • What would happen if we use \(k\)-NNs or SVM RBFs on the spotify dataset from hw2?

  • What happens if we have missing values in our data?

  • What do we do if we have features with categories or string values?