Cross Validation in Keras

32,199

Solution 1

If my goal is to fine-tune the network for the entire dataset

It is not clear what you mean by "fine-tune", or even what exactly is your purpose for performing cross-validation (CV); in general, CV serves one of the following purposes:

  • Model selection (choose the values of hyperparameters)
  • Model assessment

Since you don't define any search grid for hyperparameter selection in your code, it would seem that you are using CV in order to get the expected performance of your model (error, accuracy etc).

Anyway, for whatever reason you are using CV, the first snippet is the correct one; your second snippet

model = None
model = create_model()
for train, test in kFold.split(X, Y):
    train_evaluate(model, X[train], Y[train], X[test], Y[test])

will train your model sequentially over the different partitions (i.e. train on partition #1, then continue training on partition #2 etc), which essentially is just training on your whole data set, and it is certainly not cross-validation...

That said, a final step after the CV which is often only implied (and frequently missed by beginners) is that, after you are satisfied with your chosen hyperparameters and/or model performance as given by your CV procedure, you go back and train again your model, this time with the entire available data.

Solution 2

You can use wrappers of the Scikit-Learn API with Keras models.

Given inputs x and y, here's an example of repeated 5-fold cross-validation:

from sklearn.model_selection import RepeatedKFold, cross_val_score
from tensorflow.keras.models import * 
from tensorflow.keras.layers import * 
from tensorflow.keras.wrappers.scikit_learn import KerasRegressor

def buildmodel():
    model= Sequential([
        Dense(10, activation="relu"),
        Dense(5, activation="relu"),
        Dense(1)
    ])
    model.compile(optimizer='adam', loss='mse', metrics=['mse'])
    return(model)

estimator= KerasRegressor(build_fn=buildmodel, epochs=100, batch_size=10, verbose=0)
kfold= RepeatedKFold(n_splits=5, n_repeats=100)
results= cross_val_score(estimator, x, y, cv=kfold, n_jobs=2)  # 2 cpus
results.mean()  # Mean MSE

Solution 3

I think many of your questions will be answered if you read about nested cross-validation. This is a good way to "fine tune" the hyper parameters of your model. There's a thread here:

https://stats.stackexchange.com/questions/65128/nested-cross-validation-for-model-selection

The biggest issue to be aware of is "peeking" or circular logic. Essentially - you want to make sure that none of data used to assess model accuracy is seen during training.

One example where this might be problematic is if you are running something like PCA or ICA for feature extraction. If doing something like this, you must be sure to run PCA on your training set, and then apply the transformation matrix from the training set to the test set.

Solution 4

The commented out functions make this a little less obvious, but the idea is to keep track of your model performance as you iterate through your folds and at the end provide either those lower level performance metrics or an averaged global performance. For example:

The train_evaluate function ideally would output some accuracy score for each split, which could be combined at the end.

def train_evaluate(model, x_train, y_train, x_test, y_test):
    model.fit(x_train, y_train)
    return model.score(x_test, y_test)

X, Y = load_model()
kFold = StratifiedKFold(n_splits=10)
scores = np.zeros(10)
idx = 0
for train, test in kFold.split(X, Y):
    model = create_model()
    scores[idx] = train_evaluate(model, X[train], Y[train], X[test], Y[test])
    idx += 1
print(scores)
print(scores.mean())

So yes you do want to create a new model for each fold as the purpose of this exercise is to determine how your model as it is designed performs on all segments of the data, not just one particular segment that may or may not allow the model to perform well.

This type of approach becomes particularly powerful when applied along with a grid search over hyperparameters. In this approach you train a model with varying hyperparameters using the cross validation splits and keep track of the performance on splits and overall. In the end you will be able to get a much better idea of which hyperparameters allow the model to perform best. For a much more in depth explanation see sklearn Model Selection and pay particular attention to the sections of Cross Validation and Grid Search.

Solution 5

The main idea of testing your model performance is to perform the following steps:

  1. Train a model on a training set.
  2. Evaluate your model on a data not used during training process in order to simulate a new data arrival.

So basically - the data you should finally test your model should mimic the first data portion you'll get from your client/application to apply your model on.

So that's why cross-validation is so powerful - it makes every data point in your whole dataset to be used as a simulation of new data.

And now - to answer your question - every cross-validation should follow the following pattern:

for train, test in kFold.split(X, Y
     model = training_procedure(train, ...)
     score = evaluation_procedure(model, test, ...)

because after all, you'll first train your model and then use it on a new data. In your second approach - you cannot treat it as a mimicry of a training process because e.g. in second fold your model would have information kept from the first fold - which is not equivalent to your training procedure.

Of course - you could apply a training procedure which uses 10 folds of consecutive training in order to finetune network. But this is not cross-validation then - you'll need to evaluate this procedure using some kind of schema above.

Share:
32,199
Alisson Hayasi
Author by

Alisson Hayasi

Updated on July 09, 2022

Comments

  • Alisson Hayasi
    Alisson Hayasi almost 2 years

    I'm implementing a Multilayer Perceptron in Keras and using scikit-learn to perform cross-validation. For this, I was inspired by the code found in the issue Cross Validation in Keras

    from sklearn.cross_validation import StratifiedKFold
    
    def load_data():
        # load your data using this function
    
    def create model():
        # create your model using this function
    
    def train_and_evaluate__model(model, data[train], labels[train], data[test], labels[test)):
        # fit and evaluate here.
    
    if __name__ == "__main__":
        X, Y = load_model()
        kFold = StratifiedKFold(n_splits=10)
        for train, test in kFold.split(X, Y):
            model = None
            model = create_model()
            train_evaluate(model, X[train], Y[train], X[test], Y[test])
    

    In my studies on neural networks, I learned that the knowledge representation of the neural network is in the synaptic weights and during the network tracing process, the weights that are updated to thereby reduce the network error rate and improve its performance. (In my case, I'm using Supervised Learning)

    For better training and assessment of neural network performance, a common method of being used is cross-validation that returns partitions of the data set for training and evaluation of the model.

    My doubt is...

    In this code snippet:

    for train, test in kFold.split(X, Y):
        model = None
        model = create_model()
        train_evaluate(model, X[train], Y[train], X[test], Y[test])
    

    We define, train and evaluate a new neural net for each of the generated partitions?

    If my goal is to fine-tune the network for the entire dataset, why is it not correct to define a single neural network and train it with the generated partitions?

    That is, why is this piece of code like this?

    for train, test in kFold.split(X, Y):
        model = None
        model = create_model()
        train_evaluate(model, X[train], Y[train], X[test], Y[test])
    

    and not so?

    model = None
    model = create_model()
    for train, test in kFold.split(X, Y):
        train_evaluate(model, X[train], Y[train], X[test], Y[test])
    

    Is my understanding of how the code works wrong? Or my theory?