In my previous post about Dependency Injection Container (DiC) configuration I made mention of some more complicated scenarios that you can use to configure your objects. Not that you need help to make it more complicated. One of the things I have observed is that the more familiar I am with the DiC the more complicated the configuration gets. This is both good and bad. It’s good because you start understanding how and why all of those frigging long example DI configuration arrays are like that. The bad is that long DiC configuration arrays greatly reduce the ability of a new developer to start being productive. That’s why I think this series of blog posts are good. They show you how to get started small which is really the only way to get started if you aren’t intimately familiar with a DiC.
Let’s take our Test class and change its behavior by adding a method that allows you to set some test data (use your imagination here before complaining in the comments that it’s not a real world example).
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 | class Test { protected $test; protected $data; public function setTestData($test) { $this->data = $test; } public function getTestData() { return $this->data; } public function setTest($test) { $this->test = $test; } public function getTest() { return $this->test; } } |
We dutifully set up our DiC
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | $config = array( 'instance' => array( 'Test' => array( // Class name 'parameters' => array( 'test' => 'some data' ) ) ) ); use Zend\Di\Configuration; $diConfig = new Configuration($config); $di = new Di(); $diConfig->configure($di); |
And we run our test code getting our ‘some data’ parameter, right?
1 | string(9) "some data" |
Yep. Hmm, but what about getTestData()?
1 | var_dump($test->getTestData()); |
echos
1 | string(9) "some data" |
Hmm, that shouldn’t be happening. We did’t tell the DiC that we wanted to inject data there, did we? It looks like it’s injecting data into both setter methods. But why?
Look at the names of the parameters for both setters. They are both named “test”. The DiC will find the first method with a matching parameter name and inject the value there (if it couldn’t find it in the constructor) . You have a couple of options to fix this. If it’s your own code and you don’t mind refactoring, refactor it. However, if it’s third party code that you shouldn’t touch or code that other parts of the system are dependent on, you have a problem.
But to get around the problem you can specify the setter parameter as a fully qualified (FQ) parameter (I show another , probably better, method here). The format is “class::method:paramPos”. In this example it would be “Test::setTest:0”. So let’s change our DI configuration.
1 2 3 4 5 6 7 8 9 | $config = array( 'instance' => array( 'Test' => array( // Class name 'parameters' => array( 'Test::setTest:0' => 'some data' ) ) ) ); |
And now check the return values of both getters.
1 2 3 | $test = $di->get('Test'); var_dump($test->getTest()); var_dump($test->getTestData()); |
Now we get
1 2 | string(9) "some data" NULL |
Works.
Comments
Eric Sadeh MD
Very nice article. You share a bundle of information in this article . You stuff is really very helpful and informative. Keep writing. Thanks a lot for sharing.
Programowanie w PHP » Blog Archive » Kevin Schroeder’s Blog: ZF2 Dependency Injection: Managing Configuration – Part 2
[…] Schroeder has posted the second part of his ZF2 dependency injection series (part one here) talking about their […]