Description
PART I: PROBLEM SET
Your solutions to the problems will be submitted as a single PDF document. Be certain that your problems
are well-numbered and that it is clear what your answers are. Additionally, you will be required to duplicate
your answers to particular problems in the README file that you will submit.
1 Gradient Descent (5 pts)
Let k be a counter for the iterations of gradient descent, and let ↵k be the learning rate for the kth step
of gradient descent. In one sentence, what are the implications of using a constant value for ↵k in gradient
descent? In another sentence, what are the implications for setting ↵k as a function of k?
2 Fitting an SVM by Hand (15 pts)
[Adapted from Murphy & Jaakkola] Consider a dataset with only 2 points in 1D: (x1 = 0, y1 = 1) and
(x2 = p2, y2 = +1). Consider mapping each point to 3D using the feature vector (x) = [1,
p2x, x2]
| (i.e.,
use a 2nd-order polynomial kernel). The maximum margin classifier has the form
min kwk2
2 s.t. (1)
y1(w|(x1) + w0) 1 (2)
y2(w|(x2) + w0) 1 (3)
a.) Write down a vector that is parallel to the optimal vector w. (Hint: recall that w is orthogonal to the
decision boundary between the two points in 3D space.)
b.) What is the value of the margin that is achieved by w? (Hint: think about the geometry of two points
in space, with a line separating one from the other.)
c.) Solve for w, using the fact that the margin is equal to 2
kwk2 .
d.) Solve for w0, using your value of w and the two constraints (2)–(3) for the max margin classifier.
e.) Write down the form of the discriminant h(x) = w0 + w|(x) as an explicit function in terms of x.
3
PART II: PROGRAMMING EXERCISES
1 Polynomial Regression (25 pts)
In the previous assignment, you implemented linear regression. In the first implementation exercise, you will
modify your implementation to fit a polynomial model and explore the bias/variance tradeo↵.
Relevant Files in Homework Skeleton1
• polyreg.py
• linreg closedform.py
• test polyreg univariate.py
• test polyreg learningCurve.py
• data/polydata.dat
1.1 Implementing Polynomial Regression
Recall that polynomial regression learns a function h✓(x) = ✓0 + ✓1x + ✓2×2 + … + ✓dxd. In this case, d
represents the polynomial’s degree. We can equivalently write this in the form of a generalized linear model
h✓(x) = ✓00(x) + ✓11(x) + ✓22(x) + … + ✓dd(x) , (4)
using the basis expansion that j (x) = xj . Notice that, with this basis expansion, we obtain a linear model
where the features are various powers of the single univariate x. We’re still solving a linear regression
problem, but are fitting a polynomial function of the input.
Implement regularized polynomial regression in polyreg.py. You may implement it however you like, using
gradient descent or a closed-form solution. However, I would recommend the closed-form solution since the
data sets are small; for this reason, we’ve included an example closed-form implementation of linear regression
in linreg closedform.py (you are welcome to build upon this implementation, but make CERTAIN you
understand it, since you’ll need to change several lines of it). You are also welcome to build upon your
implementation from the previous assignment, but you must follow the API below. Note that all matrices
are actually 2D numpy arrays in the implementation.
• init(degree=1, regLambda=1E-8) : constructor with arguments of d and
• fit(X,Y): method to train the polynomial regression model
• predict(X): method to use the trained polynomial regression model for prediction
• polyfeatures(X, degree): expands the given n ⇥ 1 matrix X into an n ⇥ d matrix of polynomial
features of degree d. Note that the returned matrix will not include the zero-th power.
Note that the polyfeatures(X, degree) function maps the original univariate data into its higher order
powers. Specifically, X will be an n⇥1 matrix (X 2 Rn⇥1) and this function will return the polynomial expansion of this data, a n⇥d matrix. Note that this function will not add in the zero-th order feature (i.e., x0 = 1).
You should add the x0 feature separately, outside of this function, before training the model. By not including
the x0 column in the matrix returned by polyfeatures(), this allows the polyfeatures function to be more
Figure 1: Fit of polynomial regression with
= 0 and d = 8
general, so it could be applied to multi-variate data as well.
(If it did add the x0 feature, we’d end up with multiple
columns of 1’s for multivariate data.)
Also, notice that the resulting features will be badly scaled
if we use them in raw form. For example, with a polynomial of degree d = 8 and x = 20, the basis expansion yields
x1 = 20 while x8 = 2.56 ⇥ 1010 – an absolutely huge di↵erence in range. Consequently, we will need to standardize the
data before solving linear regression. Standardize the data in
fit() after you perform the polynomial feature expansion.
You’ll need to apply the same standardization transformation in predict() before you apply it to new data.
Run test polyreg univariate.py to test your implementation, which will plot the learned function. In this case, the
1Bold text indicates files or functions that you will need to complete; you should not need to modify any of the other files.
4
script fits a polynomial of degree d = 8 with no regularization = 0. From the plot, we see that the
function fits the data well, but will not generalize well to new data points. Try increasing the amount of
regularization, and examine the resulting e↵ect on the function.
1.2 Examine the Bias-Variance Tradeo↵ through Learning Curves
Learning curves provide a valuable mechanism for evaluating the bias-variance tradeo↵. Implement the
learningCurve() function in polyreg.py to compute the learning curves for a given training/test set.
The learningCurve(Xtrain, ytrain, Xtest, ytest, degree, regLambda) function should take in the
training data (Xtrain, ytrain), the testing data (Xtest, ytest), and values for the polynomial degree d
and regularization parameter .
The function should return two arrays, errorTrain (the array of training errors) and errorTest (the array
of testing errors). The i
th index (start from 0) of each array should return the training error (or testing
error) for learning with i + 1 training instances. Note that the 0th index actually won’t matter, since we
typically start displaying the learning curves with two or more instances.
When computing the learning curves, you should learn on Xtrain[0:i] for i = 1,…, numInstances(Xtrain),
each time computing the testing error over the entire test set. There is no need to shu✏e the training data,
or to average the error over multiple trials – just produce the learning curves for the given training/testing
sets with the instances in their given order. Recall that the error for regression problems is given by
1
n
Xn
i=1
(h✓(xi) yi)
2 . (5)
Once the function is written to compute the learning curves, run the test polyreg learningCurve.py
script to plot the learning curves for various values of and d. You should see plots similar to the following:
Notice the following:
• The y-axis is using a log-scale and the ranges of the y-scale are all di↵erent for the plots. The dashed
black line indicates the y = 1 line as a point of reference between the plots.
5
• The plot of the unregularized model with d = 1 shows poor training error, indicating a high bias (i.e.,
it is a standard univariate linear regression fit).
• The plot of the unregularized model ( = 0) with d = 8 shows that the training error is low, but that
the testing error is high. There is a huge gap between the training and testing errors caused by the
model overfitting the training data, indicating a high variance problem.
• As the regularization parameter increases (e.g., = 1) with d = 8, we see that the gap between the
training and testing error narrows, with both the training and testing errors converging to a low value.
We can see that the model fits the data well and generalizes well, and therefore does not have either a
high bias or a high variance problem. E↵ectively, it has a good tradeo↵ between bias and variance.
• Once the regularization parameter is too high ( = 100), we see that the training and testing errors
are once again high, indicating a poor fit. E↵ectively, there is too much regularization, resulting in
high bias.
Make absolutely certain that you understand these observations, and how they relate to the learning curve
plots. In practice, we can choose the value for via cross-validation to achieve the best bias-variance tradeo↵.
6
2 Logistic Regression (35 points)
Now that we’ve implemented a basic regression model using gradient descent, we will use a similar technique
to implement the logistic regression classifier.
Relevant Files in Homework Skeleton2
• test logreg1.py – python script to test logistic regression
• test logreg2.py – python script to test logistic regression on non-linearly separable data (Extra Credit)
• data/data1.dat – Student admissions prediction data, nearly linearly separable
• data/data2.dat – Microchip data, non-linearly separable
• mapFeature.py – Map instances to a higher dimensional polynomial feature space (Extra Credit).
• logreg.py: implements the LogisticRegression class
2.1 Implementation
Implement regularized logistic regression by completing the LogisticRegression class in logreg.py. Your
class must implement the following API:
• init (alpha, regLambda, epsilon, maxNumIters): the constructor, which takes in ↵, , ✏, and
maxNumIters as arguments
• fit(X,y): train the classifier from labeled data (X, y)
• predict(X): return a vector of n predictions for each of n rows of X
• computeCost(theta, X, y, regLambda): computes the logistic regression objective function for the
given values of ✓, X, y, and (“lambda” is a keyword in python, so we must call the regularization
parameter something di↵erent)
• computeGradient(theta, X, y, reg): computes the d-dimensional gradient of the logistic regression
objective function for the given values of ✓, X, y, and reg =
• sigmoid(z): returns the sigmoid function of z
Note that these methods have already been defined correctly for you in logreg.py; be very careful not to
change the API.
Sigmoid Function You should begin by implementing the sigmoid(z) function. Recall that the logistic
regression hypothesis h() is defined as:
h✓(x) = g(✓T x)
where g() is the sigmoid function defined as:
g(z) = 1
1 + expz
The Sigmoid function has the property that g(+1) ⇡ 1 and g(1) ⇡ 0. Test your function by calling
sigmoid(z) on di↵erent test samples. Be certain that your sigmoid function works with both
vectors and matrices — for either a vector or a matrix, you function should perform the sigmoid function
on every element.
2Bold text indicates files that you will need to complete; you should not need to modify any of the other files.
7
Cost Function and Gradient Implement the functions to compute the cost function and the gradient
of the cost function. Recall the cost function for logistic regression is a scalar value given by
J (✓) = Xn
i=1
[y(i) log(h✓(x(i)
)) (1 y(i)
) log(1 h✓(x(i)
))] +
2
k✓k2
2 .
The gradient of the cost function is a d-dimensional vector, where the jth element (for j = 1…d) is given by
@J (✓)
@✓j
= Xn
i=1
(h✓(x(i)
) y(i)
)x(i)
j + ✓j .
We must be careful not to regularize the ✓0 parameter (corresponding to the 1’s feature we add to each
instance), and so
@J (✓)
@✓0
= Xn
i=1
(h✓(x(i)
) y(i)
) .
Training and Prediction Once you have the cost and gradient functions complete, implement the fit
and predict methods. To make absolutely certain that the un-regularized ✓0 corresponds to the 1’s feature
that we add to the input, we will augment both the training and testing instances within the fit and
predict methods (instead of relying on it being done externally to the classifier). Recall that you can do
this via:
X = np . c [ np . ones ( ( n , 1 ) ) , X]
Your fit method should train the model via gradient descent, relying on the cost and gradient functions.
Instead of simply running gradient descent for a specific number of iterations (as in the linear regression
exercise), we will use a more sophisticated method: we will stop it after the solution has converged. Stop
the gradient descent procedure when ✓ stops changing between consecutive iterations. You can detect this
convergence when
k✓new ✓old k2 ✏ , (6)
for some small ✏ (e.g, ✏ = 10E-4). For readability, we’d recommend implementing this convergence test as
a dedicated function hasConverged. For safety, we will also set the maximum number of gradient descent
iterations, maxNumIters. The values of , ✏, maxNumIters, and ↵ (the gradient descent learning rate) are
arguments to LogisticRegression’s constructor. At the start of gradient descent, ✓ should be initialized
to random values with mean 0, as described in the linear regression exercise.
2.2 Testing Your Implementation
To test your logistic regression implementation, run python test logreg1.py from the command line. This
script trains a logistic regression model using your implementation and then uses it to predict whether or
not a student will be admitted to a school based on their scores on two exams. In the plot, the colors of the
points indicate their true class label and the background color indicates the predicted class label. If your
implementation is correct, the decision boundary should closely match the true class labels of the points,
with only a few errors.
2.3 Analysis
Varying changes the amount of regularization, which acts as a penalty on the objective function for complex
solutions. In test logreg1.py, we provide the code to draw the decision boundary of the model. Vary the
value of in test logreg1.py and plot the decision boundary for each value. Include several plots in your
PDF writeup for this assignment, labeling each plot with the value of in your writeup. Write a brief
paragraph (2-3 sentences) describing what you observe as you increase the value of and explain why.
8
2.4 (Extra Credit) Learning a Nonlinear Decision Surface (10 pts)
We strongly encourage everyone to read through this exercise in detail, even though you are not required to
complete it. This exercise is only for extra credit.
Some classification problems that cannot be solved in a low-dimensional feature space can be separable
in a high-dimension space. We can make our logistic regression classifier much more powerful by adding
additional features to the input. Imagine that we are given a data set with only two features, x1 and x2, but
the decision surface is non-linear in these two features. We can expand the possible feature set into higher
dimensions by adding new features that are functions of the given features. For example, we could add a
new feature that is the product of the original features, or any other mathematical function of those two
features that we want.
In this example, we will map the two features into all polynomial terms of x1 and x2 up to the 6th power:
mapF eature(x1, x2) =
0
BBBBBBBBBBBB@
1
x1
x2
x2
1
x1x2
x2
2
···
x1x5
2
x6
2
1
CCCCCCCCCCCCA
(7)
Given a 2-dimensional instance x, we can project that instance into a 28-dimensional feature space using
the transformation above. Then, instead of training the classifier on instances in the 2-dimensional space,
we train it on the 28-dimensional projection of those instances. The resulting decision boundary will then
be more complex and will appear non-linear in the 2D plot.
Complete the mapFeature function in mapFeature.py to map instances from a 2D feature space to a 28-
dimensional feature space defined by all polynomial terms of x1 and x2 up to the sixth power. Your function
mapFeature(X1, X2) will take in two column matrices, X1 and X2, which correspond to the 1st and 2nd
columns respectively of the data set X (recall that X is n-by-d, and so both X1 and X2 will have n entries).
It will return an n-by-28 dimensional matrix, where each row represents the expanded feature set for one
instance.
Once this function is complete, you can run python test logreg2.py from the command line. This script
will load a data set that is non-linearly separable, use your mapFeature function to project the instances
into a higher dimensional space, and then train a logistic regression model in that higher dimensional feature
space. When we project the resulting non-linear classifier back down into 2D, we get a decision surface that
appears similar to the following:
Figure 2: Nonlinear Logistic Regression Classifier
9
3 Support Vector Machines (20 points)
In this section, we’ll implement various kernels for the support vector machine (SVM). This exercise looks
long, but in practice you’ll be writing only a few lines of code. The scikit learn package already includes
several SVM implementations; in this case, we’ll focus on the SVM for classification, sklearn.svm.SVC.
Before starting this assignment, be certain to read through the documentation for SVC, available at http:
//scikit-learn.org/stable/modules/generated/sklearn.svm.SVC.html.
While we could certainly create our own SVM implementation, most people applying SVMs to real problems
rely on highly optimized SVM toolboxes, such as LIBSVM or SVMlight.3 These toolboxes provide highly
optimized SVM implementations that use a variety of optimization techniques to enable them to scale to
extremely large problems. Therefore, we will focus on implementing custom kernels for the SVMs, which is
often essential for applying these SVM toolboxes to real applications.
Relevant Files in Homework 2 Skeleton4
• example svm.py
• example svmCustomKernel.py
• svmKernels.py
• test svmGaussianKernel.py
• test svmPolyKernel.py
• data/svmData.dat
• data/svmTuningData.dat
3.1 Getting Started
The SVC implementation provided with scikit learn uses the parameter C to control the penalty for misclassifying training instances. We can think of C as being similar to the inverse of the regularization parameter
1
that we used before for linear and logistic regression. C = 0 causes the SVM to incur no penalty for
misclassifications, which will encourage it to fit a larger-margin hyperplane, even if that hyperplane misclassifies more training instances. As C grows large, it causes the SVM to try to classify all training examples
correctly, and so it will choose a smaller margin hyperplane if that hyperplane fits the training data better.
Examine example svm.py, which fits a linear SVM to the data shown below. Note that most of the positive
and negative instances are grouped together, suggesting a clear separation between the classes, but there is
an outlier around (0.5,6.2). In the first part of this exercise, we will see how this outlier a↵ects the SVM fit.
Run example svm.py with C = 0.01, and you can clearly see that the hyperplane follows the natural
separation between most of the data, but misclassifies the outlier. Try increasing the value of C and observe
the e↵ect on the resulting hyperplane. With C = 1, 000, we can see that the decision boundary correctly
classifies all training data, but clearly no longer captures the natural separation between the data.
3The SVC implementation provided with scikit learn is based on LIBSVM, but is not quite as ecient. 4Bold text indicates files that you will need to complete; you should not need to modify any of the other files.
10
3.2 Implementing Custom Kernels
The SVC implementation allows us to define our own kernel functions to learn non-linear decision surfaces
using the SVM. The SVC constructor’s kernel argument can be defined as either a string specifying one of
the built-in kernels (e.g., ’linear’, ’poly’ (polynomial), ’rbf’ (radial basis function), ’sigmoid’, etc.)
or it can take as input a custom kernel function, as we will define in this exercise.
For example, the following code snippet defines a custom kernel and uses it to train the SVM:
def myCustomKernel (X1, X2 ) :
”””
Custom kernel :
k (X1 , X2) = X1 (3 0) X2 .T
(0 2)
Note t h a t X1 and X2 are numpy a r r ay s , so we must use . d o t t o m u l t i p l y them .
”””
M = np . matrix ( [ [ 3 . 0 , 0] , [ 0 , 2 . 0 ] ] )
return np . dot ( np . dot (X1, M) , X2 .T)
# c r e a t e SVM w i t h custom k e r n e l and t r a i n model
c l f = svm .SVC( k e r n el=myCustomKernel )
c l f . f i t (X, Y)
When the SVM calls the custom kernel function during training, X1 and X2 are both initialized to be
the same as X (i.e., ntrain-by-d numpy arrays); in other words, they both contain a complete copy of
the training instances. The custom kernel function returns an ntrain-by-ntrain numpy array during the
training step. Later, when it is used for testing, X1 will be the ntest testing instances and X2 will be the
ntrain training instances, and so it will return an ntest-by-ntrain numpy array. For a complete example, see
example svmCustomKernel.py, which uses the custom kernel above to generate the following figure:
3.3 Implementing the Polynomial Kernel
We will start by writing our own implementation of the polynomial kernel and incorporate it into the SVM.5
Complete the myPolynomialKernel() function in svmKernels.py to implement the polynomial kernel:
K(v, w)=(hv, wi + 1)d , (8)
where d is the degree of polynomial and the “+1” incorporates all lower-order polynomial terms. In this
case, v and w are feature vectors. Vectorize your implementation to make it fast. Once complete, run
test svmPolyKernel.py to produce a plot of the decision surface. For comparison, it also shows the decision
surface learned using the equivalent built-in polynomial kernel; your results should be identical.
5Although scikit learn already defines the polynomial kernel, defining our own version of it provides an easy way to get
started implementing custom kernels.
11
Figure 3: Sample output of test svmPolyKernel.py
For the built-in polynomial kernel, the degree is specified in the SVC constructor. However, in our custom
kernel we must set the degree via a global variable polyDegree. Therefore, be sure to use the value of the
global variable polyDegree as the degree in your polynomial kernel. The test svmPolyKernel.py script
uses polyDegree for the degree of both your custom kernel and the built-in polynomial kernel.
Vary both C and d and study how the SVM reacts.
3.4 Implementing the Gaussian Radial Basis Function Kernel
Next, complete the myGaussianKernel() function in svmKernels.py to implement the Gaussian kernel:
K(v, w) = exp ✓
kv wk2
2
22
◆
. (9)
Be sure to use the gaussSigma for in your implementation. For computing the pairwise squared distances
between the points, you must write the method to compute it yourself; specifically you may not use the helper
methods available in sklearn.metrics.pairwise or that come with scipy. You can test your implementation
and compare it to the equivalent RBF-kernel provided in sklearn by running test svmGaussianKernel.py.
Figure 4: Sample output of test svmGaussianKernel.py
Again, vary both C and and study how the SVM reacts.
Write a brief paragraph describing how the SVM reacts as both C and d vary for the polynomial kernel, and
as C and vary for the Gaussian kernel. Put this paragraph in your writeup, and also in the README file.
3.5 Choosing the Optimal Parameters
This exercise will further help you gain further practical skill in applying SVMs with Gaussian kernels. Choosing the correct values for C and can dramatically a↵ect the quality of the model’s fit to data. Your task
12
is to determine the optimal values of C and for an SVM with your Gaussian kernel as applied to the data
in data/svmTuningData.dat, depicted below. You should use whatever method you like to search over the
space of possible combinations of C and , and determine the optimal fit as measured by accuracy. You may
use any built-in methods from scikit learn you wish (e.g., sklearn.grid search.GridSearchCV). We recommend that you search over multiplicative steps (e.g., …, 0.01, 0.03, 0.06, 0.1, 0.3, 0.6, 1, 3, 6, 10, 30, 60, 100,…).
Once you determine the optimal parameter values, report those optimal values and the corresponding estimated accuracy in the README file. For reference, the SVM with the optimal parameters we found
produced the following decision surface.
The resulting decision surface for your optimal parameters may look slightly di↵erent than ours.
13