This is the developer tutorial for implementing a new neural network classifier.
In this tutorial, you will learn how to create the generator file "*.gen.m" for a new neural network classifier, which can then be compiled by braph2genesis. All kinds of neural network models are (direct or indirect) extensions of the base element NNBase. Here, you will use as example the neural network classifier NNClassifierMLP (multi-layer perceptron classifier).
Implementation of a neural network classifier (
NNClassifierMLP)
Implementation of a neural network classifier (NNClassifierMLP) ⬆
You will start by implementing in detail NNClassifierMLP, which is a direct extension of NNBase.
A multi-layer perceptron classifier NNClassifierMLP comprises a multi-layer perceptron classifier model and a given dataset.
Code 1. NNClassifierMLP element header. The
headersection of the generator code for "_NNClassifierMLP.gen.m" provides the general information about theNNClassifierMLPelement.%% ¡header! NNClassifierMLP < NNBase (nn, multi-layer perceptron classifier) comprises a multi-layer perceptron classifier model and a given dataset. ① %%% ¡description! A neural network multi-layer perceptron classifier (NNClassifierMLP) comprises a multi-layer perceptron classifier model and a given dataset. NNClassifierMLP trains the multi-layer perceptron classifier with a formatted inputs ("CB", channel and batch) derived from the given dataset. %%% ¡build! 1① defines
NNClassifierMLPas a subclass ofNNBase. The moniker will benn.
Code 2. NNClassifierMLP element prop update. The
props_updatesection of the generator code for "_NNClassifierMLP.gen.m" updates the properties of theNNClassifierMLPelement. This defines the core properties of the data point.%% ¡props_update! %%% ¡prop! NAME (constant, string) is the name of the neural network multi-layer perceptron classifier. %%%% ¡default! 'NNClassifierMLP' %%% ¡prop! DESCRIPTION (constant, string) is the description of the neural network multi-layer perceptron classifier. %%%% ¡default! 'A neural network multi-layer perceptron classifier (NNClassifierMLP) comprises a multi-layer perceptron classifier model and a given dataset. NNClassifierMLP trains the multi-layer perceptron classifier with a formatted inputs ("CB", channel and batch) derived from the given dataset.' %%% ¡prop! TEMPLATE (parameter, item) is the template of the neural network multi-layer perceptron classifier. %%%% ¡settings! 'NNClassifierMLP' %%% ¡prop! ID (data, string) is a few-letter code for the neural network multi-layer perceptron classifier. %%%% ¡default! 'NNClassifierMLP ID' %%% ¡prop! LABEL (metadata, string) is an extended label of the neural network multi-layer perceptron classifier. %%%% ¡default! 'NNClassifierMLP label' %%% ¡prop! NOTES (metadata, string) are some specific notes about the neural network multi-layer perceptron classifier. %%%% ¡default! 'NNClassifierMLP notes' %%% ¡prop! ① D (data, item) is the dataset to train the neural network model, and its data point class DP_CLASS defaults to one of the compatible classes within the set of DP_CLASSES. %%%% ¡settings! 'NNDataset' %%%% ¡default! NNDataset('DP_CLASS', 'NNDataPoint_CON_CLA') %%% ¡prop! DP_CLASSES (parameter, classlist) is the list of compatible data points. %%%% ¡default! ② {'NNDataPoint_CON_CLA' 'NNDataPoint_CON_FUN_MP_CLA' 'NNDataPoint_Graph_CLA' 'NNDataPoint_Measure_CLA'} %%% ¡prop! INPUTS (query, cell) constructs the data in the CB (channel-batch) format. %%%% ¡calculate! ③ % inputs = nn.get('inputs', D) returns a cell array with the % inputs for all data points in dataset D. if isempty(varargin) value = {}; return end d = varargin{1}; inputs_group = d.get('INPUTS'); if isempty(inputs_group) value = {}; else flattened_inputs_group = []; for i = 1:1:length(inputs_group) inputs_individual = inputs_group{i}; flattened_inputs_individual = []; while ~isempty(inputs_individual) currentData = inputs_individual{end}; % Get the last element from the stack inputs_individual = inputs_individual(1:end-1); % Remove the last element if iscell(currentData) % If it's a cell array, add its contents to the stack inputs_individual = [inputs_individual currentData{:}]; else % If it's numeric or other data, append it to the vector flattened_inputs_individual = [currentData(:); flattened_inputs_individual]; end end flattened_inputs_group = [flattened_inputs_group; flattened_inputs_individual']; end value = {flattened_inputs_group}; end %%% ¡prop! TARGETS (query, cell) constructs the targets in the CB (channel-batch) format with one-hot vectors. %%%% ¡calculate! ④ % targets = nn.get('TARGETS', D) returns a cell array with the % targets for all data points in dataset D with one-hot vectors. if isempty(varargin) value = {}; return end d = varargin{1}; targets = cellfun(@(target) cell2mat(target), d.get('TARGETS'), 'UniformOutput', false); targets = categorical(cell2mat(targets))'; value = onehotencode(targets, 2, "ClassNames", flip(string(unique(targets)))); %%% ¡prop! MODEL (result, net) is a trained neural network model. %%%% ¡calculate! ⑤ inputs = cell2mat(nn.get('INPUTS', nn.get('D'))); ⑥ targets = nn.get('TARGET_CLASSES', nn.get('D')); ⑦ if isempty(inputs) || isempty(targets) value = network(); else number_features = size(inputs, 2); number_targets = size(targets, 2); targets = categorical(targets); number_classes = numel(categories(targets)); layers = nn.get('LAYERS'); ⑧ nn_architecture = [featureInputLayer(number_features, 'Name', 'Input')]; for i = 1:1:length(layers) nn_architecture = [nn_architecture fullyConnectedLayer(layers(i), 'Name', ['Dense_' num2str(i)]) batchNormalizationLayer('Name', ['BatchNormalization_' num2str(i)]) dropoutLayer('Name', ['Dropout_' num2str(i)]) ]; end nn_architecture = [nn_architecture reluLayer('Name', 'Relu_output') fullyConnectedLayer(number_classes, 'Name', 'Dense_output') softmaxLayer classificationLayer('Name', 'Output') ]; % specify trianing options ⑨ options = trainingOptions(nn.get('SOLVER'), ... 'MiniBatchSize', nn.get('BATCH'), ... 'MaxEpochs', nn.get('EPOCHS'), ... 'Shuffle', nn.get('SHUFFLE'), ... 'Plots', nn.get('PLOT_TRAINING'), ... 'Verbose', nn.get('VERBOSE')); % train the neural network ⑩ value = trainNetwork(inputs, targets, nn_architecture, options); end① defines
NNDatasetwhich contains theNNDataPointto train this classifier.② defines the compatible
NNDataPointclasses with thisNNClassifierMLP.③ is a query that transforms the input data of
NNDataPointto the CB (channel-batch) format by flattening its included cells.④ is a query that constructs the one-hot vectors for the target classes.
⑤ trains the classifier with the defined dataset by the code under
¡calculate!.⑥ and ⑦ extract the inputs and targets.
⑧ defines the neural network architecture with user specified number of neurons and number of layers.
⑨ defines the neural network training options.
⑩ trains the model with those parameters and the neural network architecture.
Code 3. NNClassifierMLP element props. The
propssection of generator code for "_NNClassifierMLP.gen.m" defines the properties to be used in "NNClassifierMLP".%% ¡props! %%% ¡prop! TARGET_CLASSES (query, stringlist) constructs the target classes which represent the class of each data point. %%%% ¡calculate! ① % target_classes = nn.get('TARGET_CLASSES', D) returns a cell array with the % target classes for all data points in dataset D. if isempty(varargin) value = {''}; return end d = varargin{1}; dp_dict = d.get('DP_DICT'); if dp_dict.get('LENGTH') == 0 value = {''}; else nn_targets = []; for i = 1:1:dp_dict.get('LENGTH') target = dp_dict.get('IT', i).get('TARGET_CLASS'); nn_targets = [nn_targets; target]; end value = nn_targets; end %%% ¡prop! ② LAYERS (data, rvector) defines the number of layers and their neurons. %%%% ¡default! [32 32] %%%% ¡gui! pr = PanelPropRVectorSmart('EL', nn, 'PROP', NNClassifierMLP.LAYERS, ... 'MIN', 0, 'MAX', 2000, ... 'DEFAULT', NNClassifierMLP.getPropDefault('LAYERS'), ... varargin{:}); %%% ¡prop! WAITBAR (gui, logical) detemines whether to show the waitbar. %%%% ¡default! true① is a query that collects all the target class for all data points.
② defines the number of neuron per layer. For example,
[32 32]represents two layers, each containing 32 neurons.
Code 4. NNClassifierMLP element tests. The
testssection from the element generator "_NNClassifierMLP.gen.m". A test for creating example files should be prepared to test the properties of the data point. Furthermore, additional test should be prepared for validating the value of input and target for the data point.%% ¡tests! %%% ¡test! %%%% ¡name! train the classifier with example data %%%% ¡code! % ensure the example data is generated if ~isfile([fileparts(which('NNDataPoint_CON_CLA')) filesep 'Example data NN CLA CON XLS' filesep 'atlas.xlsx']) create_data_NN_CLA_CON_XLS() % create example files end % Load BrainAtlas im_ba = ImporterBrainAtlasXLS( ... 'FILE', [fileparts(which('NNDataPoint_CON_CLA')) filesep 'Example data NN CLA CON XLS' filesep 'atlas.xlsx'], ... 'WAITBAR', true ... ); ba = im_ba.get('BA'); % Load Groups of SubjectCON im_gr1 = ImporterGroupSubjectCON_XLS( ... 'DIRECTORY', [fileparts(which('NNDataPoint_CON_CLA')) filesep 'Example data NN CLA CON XLS' filesep 'CON_Group_1_XLS'], ... 'BA', ba, ... 'WAITBAR', true ... ); gr1 = im_gr1.get('GR'); im_gr2 = ImporterGroupSubjectCON_XLS( ... 'DIRECTORY', [fileparts(which('NNDataPoint_CON_CLA')) filesep 'Example data NN CLA CON XLS' filesep 'CON_Group_2_XLS'], ... 'BA', ba, ... 'WAITBAR', true ... ); gr2 = im_gr2.get('GR'); % create item lists of NNDataPoint_CON_CLA [~, group_folder_name] = fileparts(im_gr1.get('DIRECTORY')); it_list1 = cellfun(@(x) NNDataPoint_CON_CLA( ... 'ID', x.get('ID'), ... 'SUB', x, ... 'TARGET_CLASS', {group_folder_name}), ... gr1.get('SUB_DICT').get('IT_LIST'), ... 'UniformOutput', false); [~, group_folder_name] = fileparts(im_gr2.get('DIRECTORY')); it_list2 = cellfun(@(x) NNDataPoint_CON_CLA( ... 'ID', x.get('ID'), ... 'SUB', x, ... 'TARGET_CLASS', {group_folder_name}), ... gr2.get('SUB_DICT').get('IT_LIST'), ... 'UniformOutput', false); % create NNDataPoint_CON_CLA DICT items dp_list1 = IndexedDictionary(... 'IT_CLASS', 'NNDataPoint_CON_CLA', ... 'IT_LIST', it_list1 ... ); dp_list2 = IndexedDictionary(... 'IT_CLASS', 'NNDataPoint_CON_CLA', ... 'IT_LIST', it_list2 ... ); % create a NNDataset containing the NNDataPoint_CON_CLA DICT d1 = NNDataset( ... 'DP_CLASS', 'NNDataPoint_CON_CLA', ... 'DP_DICT', dp_list1 ... ); d2 = NNDataset( ... 'DP_CLASS', 'NNDataPoint_CON_CLA', ... 'DP_DICT', dp_list2 ... ); % combine the two datasets d = NNDatasetCombine('D_LIST', {d1, d2}).get('D'); nn = NNClassifierMLP('D', d, 'LAYERS', [10 10 10]); trained_model = nn.get('MODEL'); % Check whether the number of fully-connected layer matches (excluding Dense_output layer) ① assert(length(nn.get('LAYERS')) == sum(contains({trained_model.Layers.Name}, 'Dense')) - 1, ... [BRAPH2.STR ':NNClassifierMLP:' BRAPH2.FAIL_TEST], ... 'NNClassifierMLP does not construct the layers correctly. The number of the inputs should be the same as the length of dense layers the property.' ... )① checks whether the number of layers from the trained model is correctly set.