在编写线性分类器之前,我们先来了解一下什么是线性函数。

线性函数


当我们想把输入x转化为标签y的时候,比如,把图像分类成数字,我们会推导出一个函数 y=Wx+b。

x将是我们的像素值列表,y将是对数,对应每一个数字。让我们来看看y = Wx,其中权重W确定x在预测每个y时的影响。


python练习线性分类器 线性分类器算法_权重


y = Wx允许我们绘出一条直线将数据对应到各自的标签。 然而,这条线必须通过原点,因为当x等于0,y也等于0。

我们希望能够将线从原点移开以适应更复杂的数据。最简单的解决方案是给函数添加一个数字,我们称之为“bias”(偏差)。


python练习线性分类器 线性分类器算法_TensorFlow_02


我们的新函数变为Wx + b,允许我们对线性可分离数据创建预测。让我们使用一个具体的例子并计算结果。


矩阵乘法


python练习线性分类器 线性分类器算法_python练习线性分类器_03


a和b分别等于多少呢?

转置

我们一直使用y = Wx + b函数作为线性函数。 但是有另一个函数做同样的事情,y = xW + b。 这些函数做同样的事情,可以互换,除了涉及的矩阵的维度。

要从一个函数转换到另一个函数,您只需要交换每个矩阵的行和列维度。这称为转置。 TensorFlow使用的是xW + b。



python练习线性分类器 线性分类器算法_初始化_04


这个例子与之前的是相同的,除了矩阵被转置。 x现在具有尺寸1x3,现在W具有尺寸3x2,并且b现在具有尺寸1x2。 计算结果将产生一个尺寸为1x2的矩阵。 此1x2矩阵中的元素与测验的2x1矩阵中的元素相同。 同样,这些矩阵被简单地转置。


python练习线性分类器 线性分类器算法_正态分布_05


现在,你明白线性函数了吗?下面让我们通过tensorflow来训练这个函数。

TensorFlow中的Weights(权重)和Bias(偏差)

训练神经网络的目的是修改权重和偏差以最好地预测标签。 为了使用权重和偏差,你需要一个可以修改的Tensor。也就是说不能使用tf.placeholder()和tf.constant(),因为那些Tensors不能被修改。所以这里应该使用tf.Variable。

tf.Variable()


x = tf.Variable(5)


tf.Variable类创建了一个具有可以修改的初始值的张量,很像一个普通的Python变量。 该张量在会话中存储其状态,因此您必须手动初始化张量的状态。

使用tf.global_variables_initializer()函数初始化所有可变张量的状态。


init = tf.global_variables_initializer()
with tf.Session() as sess:
    sess.run(init)


使用tf.Variable类允许我们改变权重和偏差,但是需要选择一个初始值。如上所示,使用session(会话)调用tf.global_variables_initializer()操作。tf.global_variables_initializer()会返回一个操作,从计算图中初始化所有TensorFlow变量。



用正态分布的随机数初始化权重是一个好的做法。而使权重随机化则有助于避免模型在每次训练时都卡在相同的位置。


类似地,从正态分布选择权重可以防止任何一个权重压倒其他权重。

TensorFlow提供了一个tf.truncated_normal()函数从正态分布生成随机数。

tf.truncated_normal()


n_features = 120
n_labels = 5
weights = tf.Variable(tf.truncated_normal((n_features, n_labels)))


tf.truncated_normal()函数返回具有来自正态分布的随机值的张量,该正态分布的幅度与平均值相差不超过2个标准偏差。

由于权重已经帮助防止模型卡住,你不需要再随机化偏差。 在这里最简单的解决方案,将偏差设置为0。

tf.zeros()


_labels = 5
bias = tf.Variable(tf.zeros(n_labels))


tf.zeros()函数返回具有全零的张量。

线性分类器

下面让我们使用tenosrflow构建一个线性分类器。


python练习线性分类器 线性分类器算法_权重_06


我们将使用TensorFlow的MNIST数据集对手写数字0,1和2进行分类。 以上是您将接受培训的数据的一个小样本。 注意一些1是如何用顶部和不同角度的衬线写的。 相似性和差异将在塑造模型的权重中起到作用。

上面的图像是数字0,1和2的训练权重。权重显示他们找到的每个数字的唯一属性。


# quiz.py
# Note: You can't run code in this tab
import tensorflow as tf

def weights(n_features, n_labels):
    """
    Return TensorFlow weights
    :param n_features: Number of features
    :param n_labels: Number of labels
    :return: TensorFlow weights
    """
    # TODO: Return weights
    return tf.Variable(tf.truncated_normal((n_features, n_labels)))


def biases(n_labels):
    """
    Return TensorFlow bias
    :param n_labels: Number of labels
    :return: TensorFlow bias
    """
    # TODO: Return biases
    return tf.Variable(tf.zeros(n_labels))


def linear(input, w, b):
    """
    Return linear function in TensorFlow
    :param input: TensorFlow input
    :param w: TensorFlow weights
    :param b: TensorFlow biases
    :return: TensorFlow linear function
    """
    # TODO: Linear Function (xW + b)
    return tf.add(tf.matmul(input, w), b)


import tensorflow as tf
# Sandbox
# Note: You can't run code in this tab
from tensorflow.examples.tutorials.mnist import input_data
from quiz import weights, biases, linear


def mnist_features_labels(n_labels):
    """
    Gets the first <n> labels from the MNIST dataset
    :param n_labels: Number of labels to use
    :return: Tuple of feature list and label list
    """
    mnist_features = []
    mnist_labels = []

    mnist = input_data.read_data_sets('/datasets/ud730/mnist', one_hot=True)

    # In order to make quizzes run faster, we're only looking at 10000 images
    for mnist_feature, mnist_label in zip(*mnist.train.next_batch(10000)):

        # Add features and labels if it's for the first <n>th labels
        if mnist_label[:n_labels].any():
            mnist_features.append(mnist_feature)
            mnist_labels.append(mnist_label[:n_labels])

    return mnist_features, mnist_labels


# Number of features (28*28 image is 784 features)
n_features = 784
# Number of labels
n_labels = 3

# Features and Labels
features = tf.placeholder(tf.float32)
labels = tf.placeholder(tf.float32)

# Weights and Biases
w = weights(n_features, n_labels)
b = biases(n_labels)

# Linear Function xW + b
logits = linear(features, w, b)

# Training data
train_features, train_labels = mnist_features_labels(n_labels)

with tf.Session() as session:
    session.run(tf.global_variables_initializer())

    # Softmax
    prediction = tf.nn.softmax(logits)

    # Cross entropy
    # This quantifies how far off the predictions were.
    # You'll learn more about this in future lessons.
    cross_entropy = -tf.reduce_sum(labels * tf.log(prediction), reduction_indices=1)

    # Training loss
    # You'll learn more about this in future lessons.
    loss = tf.reduce_mean(cross_entropy)

    # Rate at which the weights are changed
    # You'll learn more about this in future lessons.
    learning_rate = 0.08

    # Gradient Descent
    # This is the method used to train the model
    # You'll learn more about this in future lessons.
    optimizer = tf.train.GradientDescentOptimizer(learning_rate).minimize(loss)

    # Run optimizer and get loss
    _, l = session.run(
        [optimizer, loss],
        feed_dict={features: train_features, labels: train_labels})

# Print loss
print('Loss: {}'.format(l))