Wednesday, January 8, 2025

How to Tackle Nonlinear Data in Machine Learning

Programming LanguageHow to Tackle Nonlinear Data in Machine Learning


Have you ever considered how your phone can recognize handwritten text and convert it into regular computer text? Or how your email can separate messages automatically into spam and non-spam categories?

Both of these examples work based on classification tasks, as does the facial recognition feature on your phone.

When building a classification algorithm, real-world data often has a non-linear relationship. And many machine learning classification algorithms struggle with non-linear algorithms. But in this article, we’ll be looking at how Support Vector Machine (SVM) kernel functions can help to solve this problem. We’ll go in-depth into a Python implementation of non-linear classification and SVM kernel functions.

Prerequisites

  1. Basic Understanding of Machine Learning

  2. Linear Algebra Basics

  3. Basic Python Programming Skills

  4. Understanding of Data Visualization

  5. A Google Colab or Jupyter Notebook Account

Table of Contents

  1. Overview of the Support Vector Machine (SVM) Technique

  2. Fundamentals of SVM

  3. SVM Objective Function

  4. Understanding Kernel Functions

  5. Popular Kernel Functions

  6. How to Choose the Right Kernel

  7. SVM Kernel Implementation

  8. Conclusion

Overview of the Support Vector Machine (SVM) Technique

Support Vector Machine (SVM) is a supervised learning algorithm. It uses a hyperplane that divides features inside a feature space into distinct categories. It’s effective for both classification and regression applications.

By identifying the optimal dividing line or plane that will serve as the decision boundary, SVM seeks to maximize the margin between the various target variables. It’s primarily utilized in classification tasks and is very helpful in ignoring outliers. It categorizes the data points of the features in the dataset into distinct outputs or classes.

SVM seeks to achieve the optimal maximum margin and an ideal or near-perfect separation. There are various applications for SVM, such as image classification, face detection, text classification, image classification, and bioinformatics. SVM is also efficient in linear and non-linear classification problems.

Importance of Kernel methods in SVM

Nonlinear classification is a sort of classification that involves categorizing features that have non-linear, curved, or complex decision boundaries. Decision boundaries are regions of space that separate two different classes.

In linear classification tasks, the region of space between the different classes such as if the email is spam or not can be easily separated with a straight line. But in non-linear relationships, it could have a circular, parabola, or a complex-shape decision boundary.

Non-linear classification tasks have patterns that cannot be discovered by linear models. This is because the features have a non-linear relationship with each other.

Two diagrams illustrating decision boundaries between two classes: (a) shows a complex, wavy decision boundary, while (b) shows a simpler, smoother boundary. Blue dots represent Class A, and red triangles represent Class B.Machine Learning Techniques for THz Imaging and Time-Domain Spectroscopy by Hochong Park and Joo-Huik Son

SVM as a linear classification algorithm isn’t efficient for a non-linear data. To handle this sort of data, it will require a kernel method, which is the core topic of this article.

A kernel method is a technique used in SVM to transform non-linear data into higher dimensions. For example, if the data has a complex decision boundary in a 2-Dimensional space (as I’ll explain further in the later part of this article), it can be transformed into a 3-Dimensional space. This allows efficient classification just with a linear plane.

The goal of the article is to teach you about SVM kernels and their application to non-linear classification tasks.

Fundamentals of SVM

Linear Classifiers and Margin Maximization

Linear classifiers are classification algorithms that make predictions by using a straight line of best fit as a decision boundary between two or more categories.

Marginal planes are used to determine the support vector in the classification task. Support vectors are the data points in the dataset that are used to separate the different target variable categories – they are data points very close to the decision boundary.

In the image below, the marginal planes are the yellow lines, while the hyperplane is the red line. The hyperplane serves as the line of best fit or decision boundary. The data points that are closest to the marginal plane are the support vectors – the data points encircled in green in the image below.

Hard Margin: Decision Boundary for classification of two labels Image by the Author

The marginal plane aims to achieve a maximum margin between its plane and the hyperplane – both having equal distance from hyperplane to achieve the best classification. The hyperplane in the image above shows a perfect linear relationship between feature x1 and feature x2. The support vectors also help to establish the location of the marginal plane.

We have the hard margin and the soft margin, serving as model optimization methodologies for the SVM. The hard margin shows that you cannot find a data point of feature x1 in the same area where there are feature x2 data points and vice versa. It used to describe a perfect classification by the algorithm. The image above gives a representation of a hard margin.

A soft margin shows that the classification is imperfect, because you can find some data points of feature x1 in the same area where we have data points of feature two, which could be caused by outliers. The image below gives a representation of soft margin.

Soft Margin: Decision Boundary for classification of two labels Image by the Author

SVM Objective Function

For a binary classification, such as a dog or a cat, the dog can be represented as class 1 and cat as -1. This shows that the decision boundary or hyperplane is the determining factor. Any value above the plane is given as 1, and the class below the plane is given as -1.

The mathematical function for the hyperplane is given as:

$$f(x) = \mathbfw^T\mathbfx + b$$

$$\beginarrayl \text The variables used are: \\ \mathbfw: \textWeight vector (defining the orientation of the hyperplane) \\ b: \textBias term (defining the position of the hyperplane) \\ \mathbfx: \textInput feature vector \\ \\ \textThe classification decision is based on the sign of f(x)\text: \\ f(x) > 0: \textClass 1 \\ f(x) < 0: \textClass -1 \endarray$$

Hard Margin SVM

The Hard Margin SVM ensures all the data points are all properly classified without error, ensuring that the data points don’t find themselves in the other part of the hyperplane, and also maximizing the margin. It’s an effective method for a “noise-free” dataset. This is achieved by minimizing an objective function given below:

$$\beginarrayl \textHard Margin SVM Objective Function: \ \min_\mathbfw,b \frac12\|\mathbfw\|^2 \\ \\ \textSubject to: \\ y_i(\mathbfw^T\mathbfx_i + b) \geq 1, \,\, \forall i \\ \\ \textWhere: \\ y_i: \text Class label of the i\text-th sample (+1 \text or -1) \\ \mathbfx_i: \text Feature vector of the i\text-th sample \endarray$$

This constraint given above in the objective function ensures that all the data points are not misclassified and the stay outside the margin.

Soft Margin SVM

The Soft Margin SVM is lenient, as it allows some misclassifications. It’s suitable for real-world datasets, which are noisy, and it handles non-linearly separable data. It introduces a slack variable that penalizes incorrect predictions.

$$\beginarrayl \textObjective Function: \ \min_\mathbfw,b,\xi \frac12\|\mathbfw\|^2 + C\sum_i=1^n \xi_i \\ \\ \textSubject to: \\ y_i(\mathbfw^T\mathbfx_i + b) \geq 1 – \xi_i, \,\, \forall i \\ \xi_i \geq 0, \,\, \forall i \\ \\ \textWhere: \\ \xi_i: \text Slack variables representing the degree of misclassification or \\ \textmargin violation. \\ C: \text Regularization parameter controlling the trade-off between \\ \textmargin maximization and error minimization. \endarray$$

The hyperparameter C helps to control the penalty for a balance between margin maximization and error minimization. A large C value minimizes the classification errors, but causes a smaller margin. A small C value allows some misclassifications but causes a larger margin.

Nonlinear Classification Problems

Non-linear classification problems include datasets with non-linear patterns that are difficult for linear SVM models to capture. This is a drawback, but SVM kernels can help.

58fcb341-735f-4a57-943b-c748b7a3f85c

Non-linear classification contains datasets with complicated relationships and linear models like linear regression will not be able to accurately generate predictions or identify trends.

Understanding Kernel Functions

In kernel functions, we transform the dataset used in the classification task into a higher dimensional feature space. This line of action enables the hyperplane (a linear decision boundary) to split the data as linearly separable data.

For example, if a dataset contains three features in a 2D plane, the kernel function converts the data to a 3D plane, making it much simpler to partition the dataset using a basic hyperplane. This technique can be used to capture non-linear relationships in data.

To provide a clearer mental image, consider three distinct feature sets in the 2D plane (x and y). This can be taken to a 3D plane by the kernel machine, where features x1 and feature x2 may be in the x-y plane, which is readily divided by a simple hyperplane, and feature x3 may be in the y-z plane, which is already separated.

The Kernel Trick Explained

Transformation into a higher dimensional space is computationally intensive and is not the best option. But we know the importance of kernel functions in classifying non-linear data. So, what’s the way forward to still achieve the same feat while bypassing the cost of computation? It’s called the kernel trick. The kernel trick explains the “magic power” of the kernel functions.

The kernel trick is the computation of the inner or dot product between the data points in the original dimensional space instead of transforming the data into a higher-dimensional space before doing the computation.

The right side of the equation below shows the dot product of ϕ(x), representing the transformed vector into a higher dimensional space (which is not efficient). It’s the same as a kernel function at the left hand side:

$$K(x_i, x_j) = \phi(x_i) \cdot \phi(x_j)$$

The purpose of the kernel trick is to perform computation based on the data point in its original dimensional space, instead of performing calculations on complex data that might require an infinite number of dimensions.

Mathematical Implementation of the Kernel Trick

Suppose we have two classes of data that are non-linear in the 2D space representing the original feature space. No straight line can separate these points because they lie diagonally across the origin.

$$\beginarrayl \textbfMapping Without Kernel Trick: \\ \\ \beginalign* \textbfThe 2D data is given as: \\ & \mathbfx_1 = (1,1), & y_1 = +1 \\ & \mathbfx_2 = (-1,-1), & y_2 = -1 \endalign* \\ \\ \textbfLet’s use a mapping function: \\ \\ \phi(x, y) = (x^2, \sqrt2xy, y^2)\ \\ \\ Mapping\ \mathbfx_1 \ and \ \ \mathbfx_2: \\ \\ \beginarrayl – \ \phi(\mathbfx_1) = (1^2, \sqrt2(1)(1), 1^2) = (1, \sqrt2, 1) \\ \\ -\ \phi(\mathbfx_2) = ((-1)^2, \sqrt2(-1)(-1), (-1)^2) = (1, \sqrt2, 1) \endarray \\ \\ \\ \textbfDot Product in Higher-Dimensional Space: \\ \\ \phi(\mathbfx_1) \cdot \phi(\mathbfx_2) = (1)(1) + (\sqrt2)(\sqrt2) + (1)(1) = 1 + 2 + 1 = 4 \\ \\ \\ \beginarrayl \textThis is the dot product of \mathbfx_1\text and \mathbfx_2\text after explicitly \\ \textmapping them to the higher-dimensional space. \endarray \endarray$$

$$\beginarrayl \textbfUsing the Kernel Trick: \\ \\ \textbfPolynomial Kernel Definition: \\ \\ K(\mathbfx_i, \mathbfx_j) = (\mathbfx_i^\top \mathbfx_j + c)^d \\ \\ \textbfFor this example: \\ \\ d = 2 \ (\textdegree of the polynomial), \quad c = 0 \ (\textno bias term) \\ \\ \textbfGiven: \\ \\ \mathbfx_1 = (1, -1), \quad \mathbfx_2 = (-1, -1) \\ \\ \textbfCompute K(\mathbfx_1, \mathbfx_2): \\ \\ \beginalign* K(\mathbfx_1, \mathbfx_2) &= ((1)(-1) + (1)(-1))^2 \\ &= (-1 – 1)^2 \\ &= (-2)^2 \\ &= 4 \endalign* \\ \\ \beginarrayl \textUsing the kernel trick, we directly compute the dot product in the higher \\ \textdimensional space without explicitly mapping the points. \endarray \endarray$$

Linear kernel

For a dataset that is linearly separable, the linear kernel is ideal. When used for non-linear data sets, which are the main topic of this article, it may result in underfitting and create a linear decision boundary. It’s provided as the input feature vectors’ dot product.

This kernel merely constructs the hyperplane or line of best fit to divide the data points. It does not perform any particular transformation to a higher dimension.

$$Linear Kernel Function: K(x_i, x_j) = x_i \cdot x_j$$

Polynomial kernel

The polynomial kernel transforms the data into a polynomial feature space of order d. It does a dot product on the feature vector with a constant c, all within the degree of d. The higher the degree of the polynomial, the better the kernel captures the relationships in the nonlinear dataset.

$$Polynomial Kernel Function: K(x_i, x_j) = (x_i \cdot x_j + c)^d$$

Gaussian or Radial Basis Function (RBF) kernel

The Gaussian kernel, also known as the RBF kernel, is often used in SVM to map the input feature vector to an infinite-dimensional feature space using a Gaussian function. This kernel can handle more complex relationships.

$$RBF Kernel Function: K(x_i, x_j) = \exp(-\gamma \|x_i – x_j\|^2)$$

Sigmoid kernel

The sigmoid kernel acts similarly to the activation function in neural networks. It functions similarly to a two-layered perception network and can map data into a higher-dimensional feature space.

$$Sigmoid Kernel Function: K(x_i, x_j) = \tanh(\alpha(x_i \cdot x_j) + c)$$

There are other kernel functions such as Laplacian kernels, hyperbolic kernels, exponential kernels, and custom kernels that you can look into if you’re curious.

How to Choose the Right Kernel

The various kernel functions are applied based on the linear and nonlinear relationships in the feature space. The linear kernel is simple and fast, and it works well with linearly separable data but not with high-dimensional data.

The polynomial kernel is well-suited for data with non-linear or polynomial relationships, as well as low-dimensional data. The RBF kernel is ideal for dense data that you have no prior knowledge of. Finally, the sigmoid kernel works well for binary and categorical data points.

SVM Kernel Implementation

Let’s now go through an example showing how you can use this technique.

Step 1: Import the necessary libraries

import numpy as np
import matplotlib.pyplot as plt
from sklearn.datasets import make_circles
from mpl_toolkits.mplot3d import Axes3D
from sklearn.preprocessing import StandardScaler

Step 2: Generate the non-linear dataset

The non-linear dataset used in this article is a circle dataset from sklearn.datasets. We used 1500 samples with a random_state of 46 to keep the dataset consistent for reproducibility. We added a Gaussian noise to the data of 10%. This function generate_circle_data is implemented to generate the dataset used in the article.

def generate_circle_data(n_samples=1500, noise=0.10, random_state=46):
    """
    Generate two concentric circles dataset.

    Parameters:
    -----------
    n_samples : int
        The total number of points generated
    noise : float
        Standard deviation of Gaussian noise added to the data
    random_state : int
        Random seed for reproducibility

    Returns:
    --------
    X : array of shape [n_samples, 2]
        The generated samples
    y : array of shape [n_samples]
        The integer labels (0 or 1) for class membership of each sample
    """
    return make_circles(n_samples=n_samples, 
                       noise=noise, 
                       random_state=random_state)

Step 3: Plot the 2D Data

The data generated above comes in 2D form. Each color represents the two different data samples. The data points were plotted which allows us to see it as a circular dataset using the Matplotlib library.

def plot_2d_data(X, y, title="2D Circle Dataset"):
    """
    Plot the 2D dataset with different colors for each class.

    Parameters:
    -----------
    X : array-like of shape (n_samples, 2)
        The input samples
    y : array-like of shape (n_samples,)
        The target values (class labels)
    title : str
        The title of the plot
    """
    plt.figure(figsize=(8, 6))
    plt.scatter(X[:, 0], X[:, 1], c=y, marker='.', cmap='viridis')
    plt.title(title)
    plt.xlabel('X₁')
    plt.ylabel('X₂')
    plt.colorbar(label='Class')
    plt.grid(True, alpha=0.3)
    plt.show()

The output image of the dataset is given below:

Output of circular dataset

Step 4: Transform into a Higher-Dimensional Space

The data in 2D is transformed into a 3D space using the polynomial kernel. We achieved this by creating a third feature X3 so it can be mapped into a higher dimensional space for easy separation.

def transform_to_3d(X):
    """Transform 2D data to 3D using radius-based transformation"""
    X1 = X[:, 0].reshape(-1, 1)
    X2 = X[:, 1].reshape(-1, 1)
    
    X3 = X1**2 + X2**2
    return np.hstack((X1, X2, X3))

Step 5: Plot the 3D Transformation

The next step is to plot the 3D transformed dataset. It now looks like a U-shaped bowl, and is separated with a hyperplane after fitting a LinearSVC model from the sklearn library as the kernel we’re using. This shows a practical example of the concepts you’ve learned so far:

def plot_3d_transformation_with_separator(X_transformed, y, title="3D Transformed Dataset with Linear Separator"):
    """Plot the 3D transformed dataset with a clear linear separating plane"""

    
    scaler = StandardScaler()
    X_scaled = scaler.fit_transform(X_transformed)

    
    svm = LinearSVC(C=1.0, dual="auto", max_iter=5000)
    svm.fit(X_scaled, y)

    
    fig = plt.figure(figsize=(12, 8))
    ax = fig.add_subplot(111, projection='3d')

    
    class_0 = y == 0
    class_1 = y == 1

    ax.scatter(X_transformed[class_0, 0], 
              X_transformed[class_0, 1], 
              X_transformed[class_0, 2],
              c='blue', 
              marker='o',
              label='Class 0',
              alpha=0.6)

    ax.scatter(X_transformed[class_1, 0], 
              X_transformed[class_1, 1], 
              X_transformed[class_1, 2],
              c='red', 
              marker='^',
              label='Class 1',
              alpha=0.6)

    
    x_min, x_max = X_transformed[:, 0].min() - 0.2, X_transformed[:, 0].max() + 0.2
    y_min, y_max = X_transformed[:, 1].min() - 0.2, X_transformed[:, 1].max() + 0.2

    xx, yy = np.meshgrid(np.linspace(x_min, x_max, 50),
                        np.linspace(y_min, y_max, 50))

    
    w = svm.coef_[0]
    b = svm.intercept_[0]

    
    grid_points = np.c_[xx.ravel(), yy.ravel(), np.zeros(xx.ravel().shape[0])]
    scaled_grid = scaler.transform(grid_points)

    
    z = (-w[0] * scaled_grid[:, 0] - w[1] * scaled_grid[:, 1] - b) / w[2]
    z = z.reshape(xx.shape)
    z = scaler.inverse_transform(np.c_[xx.ravel(), yy.ravel(), z.ravel()])[:, 2].reshape(xx.shape)

    
    surface = ax.plot_surface(xx, yy, z, alpha=0.3, cmap='coolwarm')

    
    ax.set_xlabel('X₁')
    ax.set_ylabel('X₂')
    ax.set_zlabel('X₁² + X₂²')
    ax.set_title(title)

    
    ax.legend()

    
    ax.view_init(elev=20, azim=45)

    
    ax.text2D(0.05, 0.95, 
              "Polynomial Kernel Transformation:\nΦ(x₁,x₂) → (x₁,x₂,x₁²+x₂²)\n\nClasses are linearly separable\nin transformed space", 
              transform=ax.transAxes, 
              bbox=dict(facecolor='white', alpha=0.8))

    plt.show()

def main():
    
    X, y = generate_circle_data()

    
    X_transformed = transform_to_3d(X)
    plot_3d_transformation_with_separator(X_transformed, y)

if __name__ == "__main__":
    main()

The main function is a function of functions that put together all the other functions such as generate_circle_data, transform_to_3d and plot_3d_transformation_with_separator together to establish the model. The image below shows a better separation with the aid of the polynomial kernel.

2e0af04a-93cc-44e8-8ed5-15a484385fd1

Here’s the full code:

Conclusion

In this article, you learned about the efficiency of SVM kernels for non-linear classification applications. The various functions demonstrated computational efficiency by changing input data into higher dimensional data, as shown in the example, without requiring vast amounts of storage or processing.

SVM can be used in a variety of classification tasks, including image and text classification, and it has proven to be extremely efficient.

References

  1. Park, H., & Son, J.-H. (2021). Machine learning techniques for THz imaging and time-domain spectroscopy. Sensors, 21(4), 1186. https://doi.org/10.3390/s21041186

  2. Scikit-learn developers. (2024). Support vector machines. Scikit-learn.https://scikit-learn.org/1.5/modules/svm.html

Check out our other content

Check out other tags:

Most Popular Articles