admin管理员组文章数量:1352158
This problem is better illustrated with an example. I'll use Javascript (actually Coffeescript for syntax sake), but just because Javascript is just another LISP, right?
So, suppose I'm writing a web app which does (obviously) ajax requests. I implement a function to handle that:
ajaxRequest = (url, params, callback) ->
# implementation goes here
Now, suppose I have a grid that fetches data from the server. Somewhere in my code I must do something like this:
userGrid.onMustFetch = ->
ajaxRequest '/fetch/users', { surname: 'MacGyver' }, (data) ->
# fill grid with data
What exactly is the problem here? If I want to test the implementation of onMustFetch, I will not be able to do so, because inside onMustFetch, a dependency is being called, and the test environment cannot control the dependency.
To solve this problem, I inject the dependency into the function I want to test. That means changing onMustFetch to this:
userGrid.onMustFetch = (ajaxRequest) ->
ajaxRequest '/fetch/users', { surname: 'MacGyver' }, (data) ->
# fill grid with data
Now the test code can pass a mock of ajaxRequest to onMustFetch and successfully test the behavior.
Wunderbar, right? Wrong! Now I have a second problem, the problem of having to bind the right instance of ajaxRequest to the right instance of onMustFetch.
In a language like Java, I could use a Dependency Injection framework to do this for me, and my code would look like this:
class UserGrid {
private AjaxService ajaxService;
@Inject
public UserGrid(AjaxService ajaxService) {
this.ajaxService = ajaxService;
}
public void onMustFetch() {
HashMap<String, String> params = new HashMap<String, String>();
params.put("surname", "MacGyver");
ajaxService.request("/fetch/users", params, new AjaxCallback(data) {
// fill grid with data
});
}
}
Creepy, I know... but actually the DI framework does all the wiring, so at least that part of the problem is easier.
Now back to our web app and to Javascript. Even if I successfully manage to always invoke onMustFetch with the right ajaxRequest reference (after all in this case that is not so hard to do), there must be an easier way. When my code grows, the dependencies increase. I can imagine passing a reference of ajaxRequest around, but what about when I have a securityService, a browserService, a eventBusService, etc, etc, etc.?
Now the real question here: How do lisp like languages solve this problem of managing dependencies? (It seems to me the dependencies must keep being passed around all over the application, but I'm sure there must be a better way...)
This problem is better illustrated with an example. I'll use Javascript (actually Coffeescript for syntax sake), but just because Javascript is just another LISP, right?
So, suppose I'm writing a web app which does (obviously) ajax requests. I implement a function to handle that:
ajaxRequest = (url, params, callback) ->
# implementation goes here
Now, suppose I have a grid that fetches data from the server. Somewhere in my code I must do something like this:
userGrid.onMustFetch = ->
ajaxRequest '/fetch/users', { surname: 'MacGyver' }, (data) ->
# fill grid with data
What exactly is the problem here? If I want to test the implementation of onMustFetch, I will not be able to do so, because inside onMustFetch, a dependency is being called, and the test environment cannot control the dependency.
To solve this problem, I inject the dependency into the function I want to test. That means changing onMustFetch to this:
userGrid.onMustFetch = (ajaxRequest) ->
ajaxRequest '/fetch/users', { surname: 'MacGyver' }, (data) ->
# fill grid with data
Now the test code can pass a mock of ajaxRequest to onMustFetch and successfully test the behavior.
Wunderbar, right? Wrong! Now I have a second problem, the problem of having to bind the right instance of ajaxRequest to the right instance of onMustFetch.
In a language like Java, I could use a Dependency Injection framework to do this for me, and my code would look like this:
class UserGrid {
private AjaxService ajaxService;
@Inject
public UserGrid(AjaxService ajaxService) {
this.ajaxService = ajaxService;
}
public void onMustFetch() {
HashMap<String, String> params = new HashMap<String, String>();
params.put("surname", "MacGyver");
ajaxService.request("/fetch/users", params, new AjaxCallback(data) {
// fill grid with data
});
}
}
Creepy, I know... but actually the DI framework does all the wiring, so at least that part of the problem is easier.
Now back to our web app and to Javascript. Even if I successfully manage to always invoke onMustFetch with the right ajaxRequest reference (after all in this case that is not so hard to do), there must be an easier way. When my code grows, the dependencies increase. I can imagine passing a reference of ajaxRequest around, but what about when I have a securityService, a browserService, a eventBusService, etc, etc, etc.?
Now the real question here: How do lisp like languages solve this problem of managing dependencies? (It seems to me the dependencies must keep being passed around all over the application, but I'm sure there must be a better way...)
Share Improve this question edited Apr 17, 2012 at 19:15 Patrick McElhaney 59.4k41 gold badges137 silver badges170 bronze badges asked Apr 17, 2012 at 19:04 RobotFooRobotFoo 3092 silver badges8 bronze badges 2- 2 Lisp-like languages are different from each other. They have different object systems, etc. This question is really about Javascript: how to implement a Java design pattern (Dependency Injection) in Javascript. I'm removing the off-topic tags [clojure] [lisp] and [scheme]. – Kaz Commented Apr 17, 2012 at 19:10
- 5 This is not about javascript. Javascript just happens to be the way I am familiar with in order to express the problem. I am interested in the ways the lisp languages deal with the problem, so I would most appreciate if you put the tags back in. Feel free to rephrase the question in a way that makes sense for Lisp, Clojure or Scheme. – RobotFoo Commented Apr 17, 2012 at 22:13
2 Answers
Reset to default 6This is typically done using closures/currying. You pass in your dependencies as parameters. In JS you could do:
buildUserGrid = function(dependency){
return {
onMustFetch = function(){
depenency.doSomething();
},
doSomethingElse = function(){
dependency.doSomethingElse();
}
}
}
var userGrid = buildUserGrid(ajaxRequest);
userGrid.onMustFetch();
In Javascript I don't know why you couldn't use techniques similar to any OO language. A very basic implementation in JS (sorry, I don't know Coffescript)
// expects a function
var UserGrid = function(ajaxService) {
this.dependencies = ["ajaxService"];
// will be overwritten by the DI service, but can also be
// assigned manually as in java
this.ajaxService = ajaxService;
};
UserGrid.prototype.onMustFetch=function() {
var callback = function() { ... }
this.ajaxService('/fetch/users',{ surname: 'MacGyver' }, callback);
};
var diController = {
create: function(constr) {
var obj = new constr();
// just look at obj.dependencies to see what should be assigned, and map
// the implemenations as properties of obj. this could be
// as simple as a switch or a direct mapping of names to object types
// ... assign implementations to obj
return obj;
}
};
Create:
var userGrid = diController.create(UserGrid);
So diController
does the same thing as your java dependency injector. In java it can just figure out what type of object is needed using reflection. There's not much reflecting to be done in Javascript, so create a convention to tell the system what is needed. In this case I used an array called "dependencies" but you could use any construct you like.
本文标签: javascriptDependency injection in functional programmingStack Overflow
版权声明:本文标题:javascript - Dependency injection in functional programming - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743910497a2560308.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论