sessionInfo()
## R version 3.6.2 (2019-12-12)
## Platform: x86_64-apple-darwin15.6.0 (64-bit)
## Running under: macOS Catalina 10.15.3
##
## Matrix products: default
## BLAS: /Library/Frameworks/R.framework/Versions/3.6/Resources/lib/libRblas.0.dylib
## LAPACK: /Library/Frameworks/R.framework/Versions/3.6/Resources/lib/libRlapack.dylib
##
## locale:
## [1] en_US.UTF-8/en_US.UTF-8/en_US.UTF-8/C/en_US.UTF-8/en_US.UTF-8
##
## attached base packages:
## [1] stats graphics grDevices utils datasets methods base
##
## loaded via a namespace (and not attached):
## [1] compiler_3.6.2 magrittr_1.5 tools_3.6.2 htmltools_0.4.0
## [5] yaml_2.2.1 Rcpp_1.0.3 stringi_1.4.6 rmarkdown_2.1
## [9] knitr_1.28 stringr_1.4.0 xfun_0.12 digest_0.6.24
## [13] rlang_0.4.4 evaluate_0.14
Source: https://tensorflow.rstudio.com/keras/articles/examples/mnist_cnn.html
In this example, we train a convolutional neural networks (CNN) on the MNIST data set. Achieve testing accuracy 99.16% after 12 epochs.
The MNIST database (Modified National Institute of Standards and Technology database) is a large database of handwritten digits (\(28 \times 28\)) that is commonly used for training and testing machine learning algorithms.
60,000 training images, 10,000 testing images.
For CNN, instead of vectorizing the images, we keep the 2D structure.
library(keras)
# Data Preparation -----------------------------------------------------
batch_size <- 128
num_classes <- 10
epochs <- 12
# Input image dimensions
img_rows <- 28
img_cols <- 28
# The data, shuffled and split between train and test sets
mnist <- dataset_mnist()
x_train <- mnist$train$x
y_train <- mnist$train$y
x_test <- mnist$test$x
y_test <- mnist$test$y
# Redefine dimension of train/test inputs
x_train <- array_reshape(x_train, c(nrow(x_train), img_rows, img_cols, 1))
x_test <- array_reshape(x_test, c(nrow(x_test), img_rows, img_cols, 1))
input_shape <- c(img_rows, img_cols, 1)
# Transform RGB values into [0,1] range
x_train <- x_train / 255
x_test <- x_test / 255
cat('x_train_shape:', dim(x_train), '\n')
## x_train_shape: 60000 28 28 1
cat(nrow(x_train), 'train samples\n')
## 60000 train samples
cat(nrow(x_test), 'test samples\n')
## 10000 test samples
# Convert class vectors to binary class matrices
y_train <- to_categorical(y_train, num_classes)
y_test <- to_categorical(y_test, num_classes)
image(t(x_train[1, 28:1, ,]), useRaster=TRUE, axes=FALSE, col=grey(seq(0, 1, length = 256)))
y_train[1, ]
## [1] 0 0 0 0 0 1 0 0 0 0
Define model:
model <- keras_model_sequential()
model %>%
layer_conv_2d(filters = 32, kernel_size = c(3,3), activation = 'relu',
input_shape = input_shape) %>%
layer_conv_2d(filters = 64, kernel_size = c(3,3), activation = 'relu') %>%
layer_max_pooling_2d(pool_size = c(2, 2)) %>%
layer_dropout(rate = 0.25) %>%
layer_flatten() %>%
layer_dense(units = 128, activation = 'relu') %>%
layer_dropout(rate = 0.5) %>%
layer_dense(units = num_classes, activation = 'softmax')
Compile model:
# Compile model
model %>% compile(
loss = loss_categorical_crossentropy,
optimizer = optimizer_adadelta(),
metrics = c('accuracy')
)
summary(model)
## Model: "sequential"
## ________________________________________________________________________________
## Layer (type) Output Shape Param #
## ================================================================================
## conv2d (Conv2D) (None, 26, 26, 32) 320
## ________________________________________________________________________________
## conv2d_1 (Conv2D) (None, 24, 24, 64) 18496
## ________________________________________________________________________________
## max_pooling2d (MaxPooling2D) (None, 12, 12, 64) 0
## ________________________________________________________________________________
## dropout (Dropout) (None, 12, 12, 64) 0
## ________________________________________________________________________________
## flatten (Flatten) (None, 9216) 0
## ________________________________________________________________________________
## dense (Dense) (None, 128) 1179776
## ________________________________________________________________________________
## dropout_1 (Dropout) (None, 128) 0
## ________________________________________________________________________________
## dense_1 (Dense) (None, 10) 1290
## ================================================================================
## Total params: 1,199,882
## Trainable params: 1,199,882
## Non-trainable params: 0
## ________________________________________________________________________________
system.time({
history <- model %>% fit(
x_train, y_train,
batch_size = batch_size,
epochs = epochs,
verbose = 1,
validation_data = list(x_test, y_test)
)
})
## user system elapsed
## 3037.099 1009.337 700.612
plot(history)
Testing:
scores <- model %>% evaluate(
x_test, y_test, verbose = 0
)
# Output metrics
cat('Test loss:', scores[[1]], '\n')
## Test loss: 0.02748634
cat('Test accuracy:', scores[[2]], '\n')
## Test accuracy: 0.9915
Step 1: Install cloudml
package, which provides an R interface to GCP Machine Learning Engine, and gcloud tools:
library(cloudml)
gcloud_install()
#gcloud_init() # to change GCP account setting
Step 2: Turn on ML Engine API in GCP and give CE instance permision to access ML Engines.
Step 3: Submit job, wrapped in R script mnist_cnn.R
, to a GCP ML Engine instance without GPU (training took 2032 sec):
library(cloudml)
cloudml_train("mnist_cnn.R", master_type = "standard")
To submit to a GCP ML Engine instance with a single NVIDIA Tesla K80 GPU (training took 125 secs)
cloudml_train("mnist_cnn.R", master_type = "standard_gpu")
To submit to a GCP ML Engine instance with a single NVIDIA P100 GPU (training took 77 secs)
cloudml_train("mnist_cnn.R", master_type = "standard_p100")
To submit to a GCP ML Engine instance with a cloud TPU (training took ??? secs)
cloudml_train("mnist_cnn.R", master_type = "cloud_tpu")
Results are saved to the runs
folder.
See available ML Engine machine types at https://cloud.google.com/ml-engine/docs/tensorflow/machine-types#machine_type_table
Check job status:
job_status("cloudml_2019_03_12_004910758")
Collect job output:
job_collect("cloudml_2019_03_12_004910758")
List previous runs:
ls_runs()
View run reports:
view_run("runs/cloudml_2019_03_12_004910758")