From c32bf3fe2b4eeb7b29b40b059f2fe0a99e40eeae Mon Sep 17 00:00:00 2001 From: Jonathan Baldie Date: Thu, 1 Feb 2018 22:15:36 +0000 Subject: [PATCH] Configure an Activation Function per hidden layer (#208) * ability to specify per-layer activation function * some tests for new addition to layer * appease style CI whitespace issue * more flexible addition of layers, and developer can pass Layer object in manually * new test for layer object in mlp constructor * documentation for added MLP functionality --- .../multilayer-perceptron-classifier.md | 18 +++++++ .../Network/MultilayerPerceptron.php | 13 +++-- .../Network/LayeredNetworkTest.php | 18 +++++++ .../Network/MultilayerPerceptronTest.php | 52 +++++++++++++++++++ 4 files changed, 98 insertions(+), 3 deletions(-) diff --git a/docs/machine-learning/neural-network/multilayer-perceptron-classifier.md b/docs/machine-learning/neural-network/multilayer-perceptron-classifier.md index 5acf0933..7365a715 100644 --- a/docs/machine-learning/neural-network/multilayer-perceptron-classifier.md +++ b/docs/machine-learning/neural-network/multilayer-perceptron-classifier.md @@ -19,6 +19,24 @@ $mlp = new MLPClassifier(4, [2], ['a', 'b', 'c']); ``` +An Activation Function may also be passed in with each individual hidden layer. Example: + +``` +use Phpml\NeuralNetwork\ActivationFunction\PReLU; +use Phpml\NeuralNetwork\ActivationFunction\Sigmoid; +$mlp = new MLPClassifier(4, [[2, new PReLU], [2, new Sigmoid]], ['a', 'b', 'c']); +``` + +Instead of configuring each hidden layer as an array, they may also be configured with Layer objects. Example: + +``` +use Phpml\NeuralNetwork\Layer; +use Phpml\NeuralNetwork\Node\Neuron; +$layer1 = new Layer(2, Neuron::class, new PReLU); +$layer2 = new Layer(2, Neuron::class, new Sigmoid); +$mlp = new MLPClassifier(4, [$layer1, $layer2], ['a', 'b', 'c']); +``` + ## Train To train a MLP simply provide train samples and labels (as array). Example: diff --git a/src/Phpml/NeuralNetwork/Network/MultilayerPerceptron.php b/src/Phpml/NeuralNetwork/Network/MultilayerPerceptron.php index bfec9297..a6d3be0d 100644 --- a/src/Phpml/NeuralNetwork/Network/MultilayerPerceptron.php +++ b/src/Phpml/NeuralNetwork/Network/MultilayerPerceptron.php @@ -142,10 +142,17 @@ private function addInputLayer(int $nodes): void $this->addLayer(new Layer($nodes, Input::class)); } - private function addNeuronLayers(array $layers, ?ActivationFunction $activationFunction = null): void + private function addNeuronLayers(array $layers, ?ActivationFunction $defaultActivationFunction = null): void { - foreach ($layers as $neurons) { - $this->addLayer(new Layer($neurons, Neuron::class, $activationFunction)); + foreach ($layers as $layer) { + if (is_array($layer)) { + $function = $layer[1] instanceof ActivationFunction ? $layer[1] : $defaultActivationFunction; + $this->addLayer(new Layer($layer[0], Neuron::class, $function)); + } elseif ($layer instanceof Layer) { + $this->addLayer($layer); + } else { + $this->addLayer(new Layer($layer, Neuron::class, $defaultActivationFunction)); + } } } diff --git a/tests/Phpml/NeuralNetwork/Network/LayeredNetworkTest.php b/tests/Phpml/NeuralNetwork/Network/LayeredNetworkTest.php index 21f5e9cf..2a7e6966 100644 --- a/tests/Phpml/NeuralNetwork/Network/LayeredNetworkTest.php +++ b/tests/Phpml/NeuralNetwork/Network/LayeredNetworkTest.php @@ -4,6 +4,7 @@ namespace Phpml\Tests\NeuralNetwork\Network; +use Phpml\NeuralNetwork\ActivationFunction; use Phpml\NeuralNetwork\Layer; use Phpml\NeuralNetwork\Network\LayeredNetwork; use Phpml\NeuralNetwork\Node\Input; @@ -45,6 +46,15 @@ public function testSetInputAndGetOutput(): void $this->assertEquals([0.5], $network->getOutput()); } + public function testSetInputAndGetOutputWithCustomActivationFunctions(): void + { + $network = $this->getLayeredNetworkMock(); + $network->addLayer(new Layer(2, Input::class, $this->getActivationFunctionMock())); + + $network->setInput($input = [34, 43]); + $this->assertEquals($input, $network->getOutput()); + } + /** * @return LayeredNetwork|PHPUnit_Framework_MockObject_MockObject */ @@ -52,4 +62,12 @@ private function getLayeredNetworkMock() { return $this->getMockForAbstractClass(LayeredNetwork::class); } + + /** + * @return ActivationFunction|PHPUnit_Framework_MockObject_MockObject + */ + private function getActivationFunctionMock() + { + return $this->getMockForAbstractClass(ActivationFunction::class); + } } diff --git a/tests/Phpml/NeuralNetwork/Network/MultilayerPerceptronTest.php b/tests/Phpml/NeuralNetwork/Network/MultilayerPerceptronTest.php index 885c1e14..006733f5 100644 --- a/tests/Phpml/NeuralNetwork/Network/MultilayerPerceptronTest.php +++ b/tests/Phpml/NeuralNetwork/Network/MultilayerPerceptronTest.php @@ -4,8 +4,12 @@ namespace Phpml\Tests\NeuralNetwork\Network; +use Phpml\NeuralNetwork\ActivationFunction; +use Phpml\NeuralNetwork\Layer; use Phpml\NeuralNetwork\Network\MultilayerPerceptron; +use Phpml\NeuralNetwork\Node\Neuron; use PHPUnit\Framework\TestCase; +use PHPUnit_Framework_MockObject_MockObject; class MultilayerPerceptronTest extends TestCase { @@ -26,4 +30,52 @@ public function testLearningRateSetter(): void $backprop = $this->readAttribute($mlp, 'backpropagation'); $this->assertEquals(0.24, $this->readAttribute($backprop, 'learningRate')); } + + public function testLearningRateSetterWithCustomActivationFunctions(): void + { + $activation_function = $this->getActivationFunctionMock(); + + /** @var MultilayerPerceptron $mlp */ + $mlp = $this->getMockForAbstractClass( + MultilayerPerceptron::class, + [5, [[3, $activation_function], [5, $activation_function]], [0, 1], 1000, null, 0.42] + ); + + $this->assertEquals(0.42, $this->readAttribute($mlp, 'learningRate')); + $backprop = $this->readAttribute($mlp, 'backpropagation'); + $this->assertEquals(0.42, $this->readAttribute($backprop, 'learningRate')); + + $mlp->setLearningRate(0.24); + $this->assertEquals(0.24, $this->readAttribute($mlp, 'learningRate')); + $backprop = $this->readAttribute($mlp, 'backpropagation'); + $this->assertEquals(0.24, $this->readAttribute($backprop, 'learningRate')); + } + + public function testLearningRateSetterWithLayerObject(): void + { + $activation_function = $this->getActivationFunctionMock(); + + /** @var MultilayerPerceptron $mlp */ + $mlp = $this->getMockForAbstractClass( + MultilayerPerceptron::class, + [5, [new Layer(3, Neuron::class, $activation_function), new Layer(5, Neuron::class, $activation_function)], [0, 1], 1000, null, 0.42] + ); + + $this->assertEquals(0.42, $this->readAttribute($mlp, 'learningRate')); + $backprop = $this->readAttribute($mlp, 'backpropagation'); + $this->assertEquals(0.42, $this->readAttribute($backprop, 'learningRate')); + + $mlp->setLearningRate(0.24); + $this->assertEquals(0.24, $this->readAttribute($mlp, 'learningRate')); + $backprop = $this->readAttribute($mlp, 'backpropagation'); + $this->assertEquals(0.24, $this->readAttribute($backprop, 'learningRate')); + } + + /** + * @return ActivationFunction|PHPUnit_Framework_MockObject_MockObject + */ + private function getActivationFunctionMock() + { + return $this->getMockForAbstractClass(ActivationFunction::class); + } }