pytutorial/tensorflow/intro/README.md
David Rotermund 17f804dfde
Update README.md
Signed-off-by: David Rotermund <54365609+davrot@users.noreply.github.com>
2024-01-03 20:56:57 +01:00

28 KiB

Tensorflow / Keras -- A fast non-introduction

{:.no_toc}

* TOC {:toc}

Top

This is a fast overview how to get an MNIST example running under TF Keras

If you just want to start with Tensorflow / Keras (especially if it is a scientific project), then you want to reconsider using Keras. In this case please check (& use) PyTorch.

Questions to David Rotermund

Data loader / Data generator

keras.utils.Sequence "Base object for fitting to a sequence of data, such as a dataset."
tf.keras.utils.to_categorical "Converts a class vector (integers) to binary class matrix."

Basic

from tensorflow import keras
import numpy as np


class DataGenerator(keras.utils.Sequence):
    def __init__(
        self,
        train: bool = True,
        size_of_batch: int = 32,
        number_of_classes: int = 10,
        do_shuffle: bool = True,
    ) -> None:
        super(DataGenerator, self).__init__()

        if train is True:
            self.pattern_storage: np.ndarray = np.load("./train_pattern_storage.npy")
            self.label_storage: np.ndarray = np.load("./train_label_storage.npy")
        else:
            self.pattern_storage = np.load("./test_pattern_storage.npy")
            self.label_storage = np.load("./test_label_storage.npy")

        self.pattern_storage = self.pattern_storage.astype(np.float32)
        self.pattern_storage /= np.max(self.pattern_storage)

        self.dimensions: tuple[int, int] = (
            self.pattern_storage.shape[1],
            self.pattern_storage.shape[2],
        )

        # How many pattern are there?
        self.number_of_pattern: int = self.label_storage.shape[0]

        self.size_of_batch: int = size_of_batch

        self.number_of_classes: int = number_of_classes
        self.do_shuffle: bool = do_shuffle

        if self.pattern_storage.ndim == 3:
            self.number_of_channel: int = 1
        else:
            self.number_of_channel = self.pattern_storage.shape[3]

        self.available_indices: np.ndarray = np.arange(self.number_of_pattern)

        self.on_epoch_end()

    def on_epoch_end(self) -> None:
        self.available_indices = np.arange(self.number_of_pattern)

        if self.do_shuffle is True:
            np.random.shuffle(self.available_indices)

    def __getitem__(self, index: int) -> tuple[np.ndarray, np.ndarray]:
        selected_indices: np.ndarray = self.available_indices[
            index * self.size_of_batch : (index + 1) * self.size_of_batch
        ]
        image, target = self.__data_generation(selected_indices)
        return image, target

    def __data_generation(
        self, list_of_indice: np.ndarray
    ) -> tuple[np.ndarray, np.ndarray]:
        image = np.empty(
            (self.size_of_batch, *self.dimensions, self.number_of_channel),
            dtype=np.float32,
        )
        target = np.empty((self.size_of_batch), dtype=int)

        for i in range(0, len(list_of_indice)):

            if self.pattern_storage.ndim == 3:
                image[i, :, :, 0] = self.pattern_storage[
                    self.available_indices[list_of_indice[i]], :, :
                ]
            else:
                image[i, :, :, :] = self.pattern_storage[
                    self.available_indices[list_of_indice[i]], :, :, :
                ]

            target[i] = self.label_storage[self.available_indices[list_of_indice[i]]]

        return image, keras.utils.to_categorical(
            target, num_classes=self.number_of_classes
        )

    def __len__(self):
        return int(np.floor(self.number_of_pattern / self.size_of_batch))


if __name__ == "__main__":
    pass

With data augmentation

To the pre-processing chain self.data_augmentation you can add other preprocessing layers which are then applied to the input before given to the network.

from tensorflow import keras
import numpy as np


class DataGenerator(keras.utils.Sequence):
    def __init__(
        self,
        train: bool = True,
        size_of_batch: int = 32,
        number_of_classes: int = 10,
        do_shuffle: bool = True,
    ) -> None:
        super(DataGenerator, self).__init__()

        if train is True:
            self.pattern_storage: np.ndarray = np.load("./train_pattern_storage.npy")
            self.label_storage: np.ndarray = np.load("./train_label_storage.npy")
        else:
            self.pattern_storage = np.load("./test_pattern_storage.npy")
            self.label_storage = np.load("./test_label_storage.npy")

        self.pattern_storage = self.pattern_storage.astype(np.float32)
        self.pattern_storage /= np.max(self.pattern_storage)

        self.dimensions: tuple[int, int] = (
            self.pattern_storage.shape[1],
            self.pattern_storage.shape[2],
        )
        reduction: tuple[int, int] = (4, 4)

        if train is True:
            self.data_augmentation = keras.Sequential(
                [
                    keras.layers.RandomCrop(
                        height=self.dimensions[0] - reduction[0],
                        width=self.dimensions[1] - reduction[1],
                    ),
                ]
            )
        else:
            self.data_augmentation = keras.Sequential(
                [
                    keras.layers.CenterCrop(
                        height=self.dimensions[0] - reduction[0],
                        width=self.dimensions[1] - reduction[1],
                    ),
                ]
            )

        # How many pattern are there?
        self.number_of_pattern: int = self.label_storage.shape[0]

        self.size_of_batch: int = size_of_batch

        self.number_of_classes: int = number_of_classes
        self.do_shuffle: bool = do_shuffle

        if self.pattern_storage.ndim == 3:
            self.number_of_channel: int = 1
        else:
            self.number_of_channel = self.pattern_storage.shape[3]

        self.available_indices: np.ndarray = np.arange(self.number_of_pattern)

        self.on_epoch_end()

    def on_epoch_end(self) -> None:
        self.available_indices = np.arange(self.number_of_pattern)

        if self.do_shuffle is True:
            np.random.shuffle(self.available_indices)

    def __getitem__(self, index: int) -> tuple[np.ndarray, np.ndarray]:
        selected_indices: np.ndarray = self.available_indices[
            index * self.size_of_batch : (index + 1) * self.size_of_batch
        ]
        image, target = self.__data_generation(selected_indices)
        return image, target

    def __data_generation(
        self, list_of_indice: np.ndarray
    ) -> tuple[np.ndarray, np.ndarray]:
        image = np.empty(
            (self.size_of_batch, *self.dimensions, self.number_of_channel),
            dtype=np.float32,
        )
        target = np.empty((self.size_of_batch), dtype=int)

        for i in range(0, len(list_of_indice)):

            if self.pattern_storage.ndim == 3:
                image[i, :, :, 0] = self.pattern_storage[
                    self.available_indices[list_of_indice[i]], :, :
                ]
            else:
                image[i, :, :, :] = self.pattern_storage[
                    self.available_indices[list_of_indice[i]], :, :, :
                ]

            target[i] = self.label_storage[self.available_indices[list_of_indice[i]]]

        image = self.data_augmentation(image)
        return image, keras.utils.to_categorical(
            target, num_classes=self.number_of_classes
        )

    def __len__(self):
        return int(np.floor(self.number_of_pattern / self.size_of_batch))


if __name__ == "__main__":
    pass

Train an example MNIST network

tf.keras.backend.clear_session "Resets all state generated by Keras."
tf.keras.Sequential "Sequential groups a linear stack of layers into a tf.keras.Model."
network.add() "Adds a layer instance on top of the layer stack."
tf.keras.layers.Conv2D "2D convolution layer (e.g. spatial convolution over images)."
tf.keras.layers.MaxPool2D "Max pooling operation for 2D spatial data."
tf.keras.layers.Flatten "Flattens the input. Does not affect the batch size."
tf.keras.layers.Dense "Just your regular densely-connected NN layer."
network.compile() "Configures the model for training."
tf.keras.metrics.categorical_crossentropy "Computes the categorical crossentropy loss."
tf.keras.optimizers.Adam "Optimizer that implements the Adam algorithm."
network.fit() Trains the model for a fixed number of epochs (iterations on a dataset).
network.summary() "Prints a string summary of the network."
network.save() "Saves the model to Tensorflow SavedModel or a single HDF5 file."

Parameters for the layers:

padding "One of "valid", "same" or "causal" (case-insensitive). "valid" means no padding. "same" results in padding with zeros evenly to the left/right or up/down of the input such that output has the same height/width dimension as the input. "causal" results in causal (dilated) convolutions, e.g. output[t] does not depend on input[t+1:]. "
use_bias "Boolean, whether the layer uses a bias vector."
activation "Activation function to use. If you don't specify anything, no activation is applied (see keras.activations)."
data_format " A string, one of channels_last (default) or channels_first."
from tensorflow import keras
from DataGenerator import DataGenerator

epoch_max: int = 50
number_of_classes: int = 10
size_of_batch_train: int = 100

train_data = DataGenerator(
    train=True,
    size_of_batch=size_of_batch_train,
    number_of_classes=number_of_classes,
    do_shuffle=True,
)

number_of_channels: int = train_data.number_of_channel
input_dimensions = train_data.dimensions
number_of_pattern_train = train_data.number_of_pattern

number_of_output_channels_conv1: int = 32
number_of_output_channels_conv2: int = 64
number_of_neurons_flatten1: int = 1024

kernel_size_conv1: tuple[int, int] = (5, 5)
kernel_size_pool1: tuple[int, int] = (2, 2)
kernel_size_conv2: tuple[int, int] = (5, 5)
kernel_size_pool2: tuple[int, int] = (2, 2)

stride_conv1: tuple[int, int] = (1, 1)
stride_pool1: tuple[int, int] = (2, 2)
stride_conv2: tuple[int, int] = (1, 1)
stride_pool2: tuple[int, int] = (2, 2)


keras.backend.clear_session()

network = keras.Sequential()

# Conv 1
network.add(
    keras.layers.Conv2D(
        number_of_output_channels_conv1,
        kernel_size=kernel_size_conv1,
        activation="relu",
        input_shape=(input_dimensions[0], input_dimensions[1], number_of_channels),
        padding="valid",
        strides=stride_conv1,
        data_format="channels_last",
        use_bias=True,
    )
)

# Pool 1
network.add(
    keras.layers.MaxPooling2D(
        pool_size=kernel_size_pool1,
        padding="valid",
        strides=stride_pool1,
        data_format="channels_last",
    )
)

# Conv 2
network.add(
    keras.layers.Conv2D(
        number_of_output_channels_conv2,
        kernel_size=kernel_size_conv2,
        activation="relu",
        padding="valid",
        strides=stride_conv2,
        data_format="channels_last",
        use_bias=True,
    )
)

# Pool 2
network.add(
    keras.layers.MaxPooling2D(
        pool_size=kernel_size_pool2,
        padding="valid",
        strides=stride_pool2,
        data_format="channels_last",
    )
)

# Flatten
network.add(keras.layers.Flatten(data_format="channels_last"))

# Full layer
network.add(
    keras.layers.Dense(number_of_neurons_flatten1, activation="relu", use_bias=True)
)

# Output layer
network.add(keras.layers.Dense(number_of_classes, activation="softmax"))

network.compile(
    loss=keras.losses.categorical_crossentropy,
    optimizer=keras.optimizers.Adam(),
    metrics=["accuracy"],
)

for epoch_id in range(0, epoch_max):
    print(f"Epoch: {epoch_id} of {epoch_max - 1}")
    network.fit(x=train_data)

    network.summary()
    network.save("Model_" + str(epoch_id) + ".h5")

Test the example network performance

tf.keras.models.load_model "Loads a model saved via model.save()."
network.evaluate() "Returns the loss value & metrics values for the model in test mode."
from tensorflow import keras
from DataGenerator import DataGenerator

number_of_classes: int = 10
size_of_batch_test: int = 100
model_id: int = 49

test_data = DataGenerator(
    train=False,
    size_of_batch=size_of_batch_test,
    number_of_classes=number_of_classes,
    do_shuffle=False,
)

keras.backend.clear_session()

network = keras.models.load_model("./Model_" + str(model_id) + ".h5")

test_loss, test_acc = network.evaluate(x=test_data)

print(f"Correct: {test_acc * 100.0:.2f}%")

How to extract the activities from the network

from tensorflow import keras
from DataGenerator import DataGenerator
import numpy as np

number_of_classes: int = 10
size_of_batch_test: int = 100
model_id: int = 49
pattern_batch_id: int = 0
pattern_id: int = 42

test_data = DataGenerator(
    train=False,
    size_of_batch=size_of_batch_test,
    number_of_classes=number_of_classes,
    do_shuffle=False,
)

keras.backend.clear_session()

network = keras.models.load_model("./Model_" + str(model_id) + ".h5")

image, target = test_data.__getitem__(pattern_batch_id)
the_target = target[pattern_id]

print("Layer 1 (Conv1)")
input_0 = image[pattern_id : pattern_id + 1, :, :, :]
output_0 = network.layers[0](input_0)

print("Input Shape:")
print(input_0.shape)
print("Output Shape:")
print(output_0.numpy().shape)
print("")

print("Layer 2 (Pool1)")
input_1 = output_0
output_1 = network.layers[1](input_1)

print("Input Shape:")
print(input_1.numpy().shape)
print("Output Shape:")
print(output_1.numpy().shape)
print("")

print("Layer 3 (Conv2)")
input_2 = output_1
output_2 = network.layers[2](input_2)

print("Input Shape:")
print(input_2.numpy().shape)
print("Output Shape:")
print(output_2.numpy().shape)
print("")

print("Layer 4 (Pool2)")
input_3 = output_2
output_3 = network.layers[3](input_3)

print("Input Shape:")
print(input_3.numpy().shape)
print("Output Shape:")
print(output_3.numpy().shape)
print("")

print("Layer 5 (Flatten)")
input_4 = output_3
output_4 = network.layers[4](input_4)

print("Input Shape:")
print(input_4.numpy().shape)
print("Output Shape:")
print(output_4.numpy().shape)
print("")

print("Layer 6 (Full)")
input_5 = output_4
output_5 = network.layers[5](input_5)

print("Input Shape:")
print(input_5.numpy().shape)
print("Output Shape:")
print(output_5.numpy().shape)
print("")

print("Layer 7 (Output)")
input_6 = output_5
output_6 = network.layers[6](input_6)

print("Input Shape:")
print(input_6.numpy().shape)
print("Output Shape:")
print(output_6.numpy().shape)
print("")

print("\nEstimation")
print(np.round(output_6.numpy(), 4))
print("Strongest reponse is at " + str(np.argmax(output_6.numpy())))
print("Correct output is " + str(np.argmax(the_target)))

Extracting weight and bias

Here is one way to extract the weights and bias of the whole network. Alternatively you can use get_weights from tf.keras.layers.Layer in combination with get_layer of tf.keras.Sequential.

from tensorflow import keras
from DataGenerator import DataGenerator

number_of_classes: int = 10
size_of_batch_test: int = 100
model_id: int = 49
pattern_batch_id: int = 0
pattern_id: int = 42

test_data = DataGenerator(
    train=False,
    size_of_batch=size_of_batch_test,
    number_of_classes=number_of_classes,
    do_shuffle=False,
)

keras.backend.clear_session()

network = keras.models.load_model("./Model_" + str(model_id) + ".h5")

weights_bias = network.get_weights()

counter_layer: int = 0
for i in range(0, len(weights_bias), 2):
    print("Layer " + str(counter_layer) + " weights_bias position: " + str(i) + " =>")
    print(weights_bias[i].shape)
    counter_layer += 1

print("")

counter_layer = 0
for i in range(1, len(weights_bias), 2):
    print("Bias " + str(counter_layer) + " weights_bias position: " + str(i) + " =>")
    print(weights_bias[i].shape)
    counter_layer += 1

Type of layers

Reduced list with the most relevant network layers

Activation Applies an activation function to an output.
AveragePooling1D Average pooling for temporal data.
AveragePooling2D Average pooling operation for spatial data.
AveragePooling3D Average pooling operation for 3D data (spatial or spatio-temporal).
BatchNormalization Layer that normalizes its inputs.
Conv1D 1D convolution layer (e.g. temporal convolution).
Conv2D 2D convolution layer (e.g. spatial convolution over images).
Conv3D 3D convolution layer (e.g. spatial convolution over volumes).
Dense Just your regular densely-connected NN layer.
Dropout Applies Dropout to the input.
Flatten Flattens the input. Does not affect the batch size.
MaxPooling1D Max pooling operation for 1D temporal data.
MaxPooling2D Max pooling operation for 2D spatial data.
MaxPooling3D Max pooling operation for 3D data (spatial or spatio-temporal).
SpatialDropout1D Spatial 1D version of Dropout.
SpatialDropout2D Spatial 2D version of Dropout.
SpatialDropout3D Spatial 3D version of Dropout.
ZeroPadding1D Zero-padding layer for 1D input (e.g. temporal sequence).
ZeroPadding2D Zero-padding layer for 2D input (e.g. picture).
ZeroPadding3D Zero-padding layer for 3D data (spatial or spatio-temporal).

Preprocessing layers

Reduced list with the most relevant preprocessing layers

CenterCrop A preprocessing layer which crops images.
RandomContrast A preprocessing layer which randomly adjusts contrast during training.
RandomCrop A preprocessing layer which randomly crops images during training.
RandomFlip A preprocessing layer which randomly flips images during training.
RandomHeight A preprocessing layer which randomly varies image height during training.
RandomRotation A preprocessing layer which randomly rotates images during training.
RandomTranslation A preprocessing layer which randomly translates images during training.
RandomWidth A preprocessing layer which randomly varies image width during training.
RandomZoom A preprocessing layer which randomly zooms images during training.
Rescaling A preprocessing layer which rescales input values to a new range.
Resizing A preprocessing layer which resizes images.

Activation functions

Reduced list with the most relevant activation functions

hard_sigmoid(...) Hard sigmoid activation function.
relu(...) Applies the rectified linear unit activation function.
sigmoid(...) Sigmoid activation function, sigmoid(x) = 1 / (1 + exp(-x)).
softmax(...) Softmax converts a vector of values to a probability distribution.
softplus(...) Softplus activation function, softplus(x) = log(exp(x) + 1).
softsign(...) Softsign activation function, softsign(x) = x / (abs(x) + 1).
tanh(...) Hyperbolic tangent activation function.

Loss-functions

Reduced list with the most relevant loss functions

BinaryCrossentropy Computes the cross-entropy loss between true labels and predicted labels.
CategoricalCrossentropy Computes the crossentropy loss between the labels and predictions.
KLDivergence Computes Kullback-Leibler divergence loss between y_true and y_pred.
MeanAbsoluteError Computes the mean of absolute difference between labels and predictions.
MeanSquaredError Computes the mean of squares of errors between labels and predictions.
Poisson Computes the Poisson loss between y_true and y_pred.
SparseCategoricalCrossentropy Computes the crossentropy loss between the labels and predictions.

Optimizer

Reduced list with the most relevant optimizer

Adagrad Optimizer that implements the Adagrad algorithm.
Adam Optimizer that implements the Adam algorithm.
RMSprop Optimizer that implements the RMSprop algorithm.
SGD Gradient descent (with momentum) optimizer.

Metrics

A very reduced list with the most relevant metrics

Accuracy Calculates how often predictions equal labels.
BinaryAccuracy Calculates how often predictions match binary labels.
BinaryCrossentropy Computes the crossentropy metric between the labels and predictions.
CategoricalAccuracy Calculates how often predictions match one-hot labels.
CategoricalCrossentropy Computes the crossentropy metric between the labels and predictions.
KLDivergence Computes Kullback-Leibler divergence metric between y_true and y_pred.
Mean Computes the (weighted) mean of the given values.
MeanAbsoluteError Computes the mean absolute error between the labels and predictions.
MeanSquaredError Computes the mean squared error between y_true and y_pred.
Poisson Computes the Poisson metric between y_true and y_pred.
Precision Computes the precision of the predictions with respect to the labels.
RootMeanSquaredError Computes root mean squared error metric between y_true and y_pred.
SparseCategoricalAccuracy Calculates how often predictions match integer labels.
SparseCategoricalCrossentropy Computes the crossentropy metric between the labels and predictions.
SparseTopKCategoricalAccuracy Computes how often integer targets are in the top K predictions.
Sum Computes the (weighted) sum of the given values.
TopKCategoricalAccuracy Computes how often targets are in the top K predictions.