When I first heard about dependency injection, I thought, "Dependendiwhatsit?" and promptly forgot about it. When I finally took the time to figure out what people were talking about, I laughed. "That's all it is?"
"Dependency Injection" is a 25-dollar term for a 5-cent concept. That's not to say that it's a bad term... and it's a good tool. But the top articles on Google focus on bells and whistles at the expense of the basic concept. I figured I should say something, well, simpler.
The Really Short Version
Dependency injection means giving an object its instance variables. Really. That's it.
The Slightly Longer Version, Part I: Dependency Non-Injection
Classes have these things they call methods on. Let's call those "dependencies." Most people call them "variables." Sometimes, when they're feeling fancy, they call them "instance variables."
public class Example {
private DatabaseThingie myDatabase;
public Example() {
myDatabase = new DatabaseThingie();
}
public void DoStuff() {
...
myDatabase.GetData();
...
}
}
Here, we have a variable... uh, dependency... named "myDatabase." We initialize it in the constructor.
The Slightly Longer Version, Part II: Dependency Injection
If we wanted to, we could pass the variable into the constructor. That would "inject" the "dependency" into the class. Now when we use the variable (dependency), we use the object that we were given rather than the one we created.
public class Example {
private DatabaseThingie myDatabase;
public Example() {
myDatabase = new DatabaseThingie();
}
public Example(DatabaseThingie useThisDatabaseInstead) {
myDatabase = useThisDatabaseInstead;
}
public void DoStuff() {
...
myDatabase.GetData();
...
}
}
That's really all there is to it. The rest is just variations on the theme. You could set the dependency (<cough> variable) in... wait for it... a setter method. You could set the dependency by calling a setter method that's defined in a special interface. You can have the dependency be an interface and then polymorphically pass in some polyjuice. Whatever.
The Slightly Longer Version, Part III: Why Do We Do This?
Among other things, it's handy for isolating classes during testing.
public class ExampleTest {
TestDoStuff() {
MockDatabase mockDatabase = new MockDatabase();
// MockDatabase is a subclass of DatabaseThingie, so we can
// "inject" it here:
Example example = new Example(mockDatabase);
example.DoStuff();
mockDatabase.AssertGetDataWasCalled();
}
}
public class Example {
private DatabaseThingie myDatabase;
public Example() {
myDatabase = new DatabaseThingie();
}
public Example(DatabaseThingie useThisDatabaseInstead) {
myDatabase = useThisDatabaseInstead;
}
public void DoStuff() {
...
myDatabase.GetData();
...
}
}
That's it. Dependency injection is really just passing in an instance variable.
Further Reading
There's a lot of ways to make this simple concept very complicated. (There's lots of ways to make any concept complicated! Simplicity is hard.) Sometimes such complexity is necessary... and it's never my first choice. I've chosen not to discuss those bells and whistles here, but if you really want to know, check out these resources:
Inversion of Control Containers and the Dependency Injection Pattern. Martin Fowler is my favorite author. Usually he's clear and concise. Here he succeeds in making dependency injection sound terribly complicated. Still, this article has a thorough discussion of the various ways dependency injection can be tweaked.
A Beginner's Guide to Dependency Injection. This article is more about "DI containers" than it is about dependency injection itself. It takes a simple example and shows how it could be implemented using multiple off-the-shelf containers. I'm left wondering what the value is, but I'm a heretic, so feel free to ignore me on this one. Still, you have to marvel at any approach that takes three concepts ("TripPlanner," "CabAgency," and "AirlineAgency"), turns them into nine-plus classes, and then adds dozens of lines of glue code and configuration XML before a single line of application logic is written.
'Archive' 카테고리의 다른 글
The Liskov Substitution Principle; LSP (리스코프의 치환원칙) (0) | 2012.01.24 |
---|---|
Chain Constructors (연쇄 생성자) (0) | 2012.01.24 |
Enum을 사용한 State 패턴 (0) | 2012.01.24 |
Anonymous class를 사용한 Command 패턴들의 Singleton (0) | 2012.01.24 |
한글 띄워쓰기 쉽게 교정하는 방법 (1) | 2012.01.17 |