0

我正在尝试为自己构建一个小助手库。首先,出于学习目的,然后我可以扩展它,以便在项目中派上用场。

我对原型引用、闭包和作用域有所了解。我还故意使用模块化模式来制作它,这样我toolbox就不会污染全局命名空间。我也知道我们可以将原型分配给构造函数,因此构造的对象将保存这些方法。

下面是内容toolbox.js

(function (window) { 
var toolbox = (function () { 
    var toolbox = function(it){
            return new pick(it);
    }
    var pick = function (it){
        var craft = document.getElementsByTagName(it);
        craft = Array.prototype.slice.call(craft, 0);
        return Array.prototype.push.apply(this, craft);
    }

    pick.prototype = toolbox.prototype = {
        raw: function(){
            return Array.prototype.slice.call(this, 0);
        }, 
        tell: function(secret){
            return secret;
        }
    }

    return toolbox;
}());
window.toolbox = toolbox;
})(window);

并调用toolbox

toolbox("div"); //returns the desired object with the div collection
toolbox("div").raw(); //returns a raw array with the divs
toolbox().tell("A secret"); //returns "A secret"
toolbox.tell("It's a secret"); //type error: the function has no method like tell (like hell it does...should)

但是像这样修改上面的代码:

var toolbox = (function () { 
    var toolbox = function(it){
    return new pick(it);
}
     ...
    toolbox.tell(secret){ return secret }
return toolbox;
}());

将工作。

所以我的问题是为什么toolbox.prototype = {} 不这样做,而pick.prototype = {}pick继承定义的方法?

我想同时实现这两个目标toolbox.tell("something");,并且toolbox("div").raw();不必直接将方法原型化到模块中。

请帮忙!我已经在谷歌上搜索了好几天来学习这些,现在我被困住了。非常感谢您的帮助!

更新

简而言之,jQuery 是如何做到这一点的:

(function( window, undefined ) {
    var jQuery = (function() {
        // Define a local copy of jQuery
        var jQuery = function( selector, context ) {
                // The jQuery object is actually just the init constructor 'enhanced'
                return new jQuery.fn.init( selector, context, rootjQuery );
            }   

        jQuery.fn = jQuery.prototype = {
            constructor: jQuery,
            init: function( selector, context, rootjQuery ) {
                //init stuff
            }
        };

        // Give the init function the jQuery prototype for later instantiation
        jQuery.fn.init.prototype = jQuery.fn;

        jQuery.extend = jQuery.fn.extend = function() {
            //extend stuff
        };

        jQuery.extend({
            //extend stuff
        });

    // Expose jQuery to the global object
    return jQuery;

    })();
window.jQuery = window.$ = jQuery;
})(window);
4

3 回答 3

2

原型中的东西和对象本身的东西是有区别的。

考虑以下示例:

var foo = function() { return 'I am foo'; }

foo.prototype.lie = function() { return 'I am not foo'; }
foo.lie(); //error, lie does not exist in foo

var bar = new foo;
bar.lie(); //it works!

prototype基本上是父母和孩子之间的中间地带。原型拥有父级的所有内容,以及您附加到它的所有内容 - 但是,父级没有自己的原型。而且,更令人困惑的是,如果您将内容分配给父对象而不是其原型,则额外的内容不会被复制给子对象。很奇怪,对吧?

解释最后一条语句(使用前面的示例):

foo.makeAPoem = function() { return 'Roses are red, violets are blue, coffe milk eggs'; }
var bar = new foo;
foo.makeAPoem(); //works
bar.makeAPoem(); //nope!

这正是您没有String.split例如的原因,但是'abcd'.split- 这些方法存在于原型中。

如果您想让某些东西在对象和原型中都有效,请将其分配给两者!

foo.jedi = foo.prototype.jedi = function() { return 'This is not the function you are looking for'; }
foo.jedi(); //works
bar.jedi(); //works!

所以现在,您可以选择继承的内容和保留在父项中的内容。如果您希望它仅对父级可用,则仅将其附加到父级。但是,如果您希望它仅在孩子上可用(例如jQuery.andSelf,您没有 jQuery().andSelf),请将其附加到原型中。如果您想要两者,请将其附加到两者。

于 2011-06-01T05:50:31.947 回答
0

您可以定义一个电源构造函数(我不确定它是否真的这样调用=)),例如:

var toolbox = (function(){
 // some private methods
   var test_1 = function(){ alert("test_1"); };
   var test_2 = function(){ alert("test_1"); };
 // return needed methods and props into the global namespace: 
   return {
    "test_1" : test_1,
    "test_2" : test_2
   }
})();

toolbox.test_1();
于 2011-06-01T05:38:56.070 回答
0

原因是 toolbox 从来没有像 pick 那样被实例化为构造函数。简单的解决方案。

代替

pick.prototype = toolbox.prototype = {
....
...
return toolbox;

只用

pick.prototype = {
....
....
for(var attr in pick.prototype) {
  toolbox[attr] = pick.prototype[i];
}

这会将方法直接添加到工具箱

于 2011-06-01T05:51:15.547 回答