Heim  >  Artikel  >  Web-Frontend  >  So verwenden Sie natives JS, um die benötigten Plug-Ins zu kapseln

So verwenden Sie natives JS, um die benötigten Plug-Ins zu kapseln

php中世界最好的语言
php中世界最好的语言Original
2018-03-06 11:51:374490Durchsuche

Heute möchte ich mit Ihnen unseren Bedarf an Plug-Ins bei der Arbeit besprechen. Wenn wir Code schreiben, muss nicht der gesamte Geschäfts- oder Logikcode extrahiert und wiederverwendet werden. Zuerst müssen wir prüfen, ob wir einen Teil des häufig wiederholten Codes für die zukünftige Verwendung in eine separate Datei abstrahieren müssen. Schauen wir uns noch einmal an, ob unsere Geschäftslogik dem Team dienen kann. Plug-Ins werden nicht beiläufig geschrieben, sondern nach ihrer eigenen Geschäftslogik abstrahiert. Es gibt kein einheitliches Plug-In, sondern nur Plug-Ins. Der Grund, warum sie Plug-Ins genannt werden, besteht darin, dass sie sofort verwendet werden können oder wir nur einige Konfigurationsparameter hinzufügen müssen, um sie zu erreichen die Ergebnisse, die wir brauchen. Wenn diese Bedingungen erfüllt sind, werden wir die Erstellung eines Plug-Ins in Betracht ziehen.

Bedingungen für die Plug-in-Kapselung

Ein wiederverwendbares Plug-in muss die folgenden Bedingungen erfüllen:

Der Umfang des Plug-ins selbst ist unabhängig von der aktuellen Version des Benutzers Umfang, das heißt, die internen privaten Variablen des Plug-Ins können sich nicht auf die Umgebungsvariablen des Benutzers auswirken.

Zusätzlich zu den implementierten Grundfunktionen müssen Plug-Ins über Standardeinstellungsparameter verfügen Plug-Ins müssen einige APIs bereitstellen, und Benutzer können die Standardparameter der Plug-In-Funktion ändern, um benutzerdefinierte Plug-In-Effekte zu erzielen.

Plug-Ins unterstützen Kettenaufrufe >

Plug-Ins müssen Überwachungseingänge bereitstellen und bestimmte Elemente überwachen, damit die Elemente und Plug-Ins auf Plug-In-Effekte reagieren.

Informationen zu den Bedingungen für die Plug-in-Kapselung finden Sie in diesem Artikel: Native

JavaScript

Plug-in Writing Guide

Was ich erklären möchte, ist, wie ich mein Plug-in umsetze. in der Kapselung Schritt für Schritt. Ich beginne also mit einfachen Methodenfunktionen. Äußere Verpackung von Plug-Ins

Umhüllung mit Funktionen

Das sogenannte Plug-In ist eigentlich eine Reihe von Funktionen, die in einem Verschluss gekapselt sind. Ich erinnere mich, als ich anfing, js zu schreiben, habe ich die gewünschte Logik in eine Funktion geschrieben und dann je nach Bedarf verschiedene Parameter übergeben.

Zum Beispiel möchte ich die Methode zum Addieren zweier Zahlen implementieren:


Dies ist eine einfache Implementierung der gewünschten Funktion. Wenn Sie nur eine so einfache Logik implementieren, ist es nicht nötig, etwas Besonderes zu tun. Die js-Funktion selbst kann die meisten Probleme lösen. In unserer tatsächlichen Arbeit und Anwendung sind die allgemeinen Anforderungen jedoch viel komplexer.

Wenn Ihnen das Produkt zu diesem Zeitpunkt mitteilt, dass ich nicht nur zwei Zahlen addieren muss, sondern auch Funktionen wie Subtraktion, Multiplikation, Division, Rest usw. benötige. Was sollen wir zu diesem Zeitpunkt tun?
function add(n1,n2) {
    return n1 + n2;
}
// 调用
add(1,2)
// 输出:3
Natürlich werden Sie denken, warum das so schwierig ist. Schreiben Sie einfach alle diese Funktionen auf und fertig. Dann fügen Sie sie alle in eine js-Datei ein. Rufen Sie bei Bedarf einfach an.



OK, jetzt sind alle Funktionen implementiert, die wir brauchen. Und wir haben diese Funktionen auch in ein js geschrieben. Wenn Sie es alleine verwenden, können Sie klar erkennen, ob Sie etwas definiert haben, welchen Inhalt Sie geschrieben haben und auf welcher Seite ich es benötige. Anschließend können Sie diese js-Datei direkt importieren und fertig.

Wenn es sich jedoch um ein Team aus mehr als zwei Personen handelt oder Sie mit anderen zusammenarbeiten, um Code zu schreiben, weiß die andere Person zu diesem Zeitpunkt nicht, ob Sie die Add-Methode geschrieben haben, und definiert diese dann auch Methode hinzufügen. Dann kommt es zu einem Namenskonflikt zwischen Ihnen, der im Allgemeinen als globale Variablenverschmutzung bezeichnet wird.
// 加
function add(n1,n2) {
    return n1 + n2;
}
// 减
function sub(n1,n2) {
    return n1 - n2;
}
// 乘
function mul(n1,n2) {
    return n1 * n2;
}
// 除
function div(n1,n2) {
    return n1 / n2;
}
// 求余
function sur(n1,n2) {
    return n1 % n2;
}

Verwenden Sie globale Objekte zum Umschließen

, um dieses Problem der globalen Variablenverschmutzung zu lösen. Zu diesem Zeitpunkt können wir ein js-Objekt definieren, um unsere Toolfunktionen zu empfangen.

Bei der oben genannten Methode wird vereinbart, dass das Plug-in den Namen „Plugin“ trägt, sodass sich die Teammitglieder an die Benennungsregeln halten müssen, wodurch das Problem der globalen Umweltverschmutzung bis zu einem gewissen Grad gelöst wurde. Vereinbaren Sie in der Teamzusammenarbeit einfach die Namensregeln und informieren Sie andere Studierende. Natürlich ist es nicht ausgeschlossen, dass jemand Ihr Projekt übernimmt und nicht weiß, dass diese globale Variable definiert wurde. Anschließend definiert er sie erneut und weist Ihrem Objekt einen Wert zu. Natürlich können Sie dies tun, um das Namenskonfliktproblem zu lösen:

var plugin = {
    add: function(n1,n2){...},//加
    sub: function(n1,n2){...},//减
    mul: function(n1,n2){...},//乘
    div: function(n1,n2){...},//除
    sur: function(n1,n2){...} //余
}
// 调用
plugin.add(1,2)

oder

if(!plugin){ //这里的if条件也可以用: (typeof plugin == 'undefined')
    var plugin = {        // 以此写你的函数逻辑    }
}

Auf diese Weise entsteht kein Namenskonflikt.

var plugin;
if(!plugin){
    plugin = {
        // ...
    }
}
Grundsätzlich kann man dies als Plug-In betrachten. Das globale Verschmutzungsproblem ist gelöst und die Methodenfunktion kann extrahiert und in einer separaten Datei abgelegt werden.

Verschlussverpackung verwenden

上面的例子,虽然可以实现了插件的基本上的功能。不过我们的plugin对象,是定义在全局域里面的。我们知道,js变量的调用,从全局作用域上找查的速度会比在私有作用域里面慢得多得多。所以,我们最好将插件逻辑写在一个私有作用域中。
实现私有作用域,最好的办法就是使用闭包。可以把插件当做一个函数,插件内部的变量及函数的私有变量,为了在调用插件后依旧能使用其功能,闭包的作用就是延长函数(插件)内部变量的生命周期,使得插件函数可以重复调用,而不影响用户自身作用域。
故需将插件的所有功能写在一个立即执行函数中:

;(function(global,undefined) {
    var plugin = {
        add: function(n1,n2){...}
        ...
    }
    // 最后将插件对象暴露给全局对象
    'plugin' in global && global.plugin = plugin;
})(window);

对上面的代码段传参问题进行解释一下:

在定义插件之前添加一个分号,可以解决js合并时可能会产生的错误问题;

undefined在老一辈的浏览器是不被支持的,直接使用会报错,js框架要考虑到兼容性,因此增加一个形参undefined,就算有人把外面的 undefined 定义了,里面的 undefined 依然不受影响;

window对象作为参数传入,是避免了函数执行的时候到外部去查找。

其实,我们觉得直接传window对象进去,我觉得还是不太妥当。我们并不确定我们的插件就一定用于浏览器上,也有可能使用在一些非浏览端上。所以我们还可以这么干,我们不传参数,直接取当前的全局this对象为作顶级对象用。

;(function(global,undefined) {
    "use strict" //使用js严格模式检查,使语法更规范
    var _global;
    var plugin = {
        add: function(n1,n2){...}
        ...
    }
    // 最后将插件对象暴露给全局对象
    _global = (function(){ return this || (0, eval)('this'); }());
    !('plugin' in _global) && (_global.plugin = plugin);
}());

如此,我们不需要传入任何参数,并且解决了插件对环境的依事性。如此我们的插件可以在任何宿主环境上运行了。

关于立即自执行函数,有两种写法:

/ 写法一
(function(){})()//

写法二

(function(){}())

使用模块化的规范包装

虽然上面的包装基本上已经算是ok了的。但是如果是多个人一起开发一个大型的插件,这时我们要该怎么办呢?多人合作,肯定会产生多个文件,每个人负责一个小功能,那么如何才能将所有人开发的代码集合起来呢?这是一个讨厌的问题。要实现协作开发插件,必须具备如下条件:

每功能互相之间的依赖必须要明确,则必须严格按照依赖的顺序进行合并或者加载

每个子功能分别都要是一个闭包,并且将公共的接口暴露到共享域也即是一个被主函数暴露的公共对象

关键如何实现,有很多种办法。最笨的办法就是按顺序加载js

<script type="text/javascript" src="part1.js"></script>
<script type="text/javascript" src="part2.js"></script>
<script type="text/javascript" src="part3.js"></script>...<script type="text/javascript" src="main.js"></script>

但是不推荐这么做,这样做与我们所追求的插件的封装性相背。
不过现在前端界有一堆流行的模块加载器,比如require、seajs,或者也可以像类似于Node的方式进行加载,不过在浏览器端,我们还得利用打包器来实现模块加载,比如browserify。不过在此不谈如何进行模块化打包或者加载的问题,如有问题的同学可以去上面的链接上看文档学习。
为了实现插件的模块化并且让我们的插件也是一个模块,我们就得让我们的插件也实现模块化的机制。
我们实际上,只要判断是否存在加载器,如果存在加载器,我们就使用加载器,如果不存在加载器。我们就使用顶级域对象。

if (typeof module !== "undefined" && module.exports) {
    module.exports = plugin;
} else if (typeof define === "function" && define.amd) {
    define(function(){return plugin;});
} else {
    _globals.plugin = plugin;
}

这样子我们的完整的插件的样子应该是这样子的:

// plugin.js
;(function(undefined) {
    "use strict"
    var _global;
    var plugin = {
        add: function(n1,n2){ return n1 + n2; },//加
        sub: function(n1,n2){ return n1 - n2; },//减
        mul: function(n1,n2){ return n1 * n2; },//乘
        div: function(n1,n2){ return n1 / n2; },//除
        sur: function(n1,n2){ return n1 % n2; } //余
    }
    // 最后将插件对象暴露给全局对象
    _global = (function(){ return this || (0, eval)(&#39;this&#39;); }());
    if (typeof module !== "undefined" && module.exports) {
        module.exports = plugin;
    } else if (typeof define === "function" && define.amd) {
        define(function(){return plugin;});
    } else {
        !(&#39;plugin&#39; in _global) && (_global.plugin = plugin);
    }
}());

我们引入了插件之后,则可以直接使用plugin对象。

with(plugin){
    console.log(add(2,1)) // 3
    console.log(sub(2,1)) // 1
    console.log(mul(2,1)) // 2
    console.log(div(2,1)) // 2
    console.log(sur(2,1)) // 0
}

插件的API

插件的默认参数

我们知道,函数是可以设置默认参数这种说法,而不管我们是否传有参数,我们都应该返回一个值以告诉用户我做了怎样的处理,比如:

function add(param){    
        var args = !!param ? Array.prototype.slice.call(arguments) : [];
            return args.reduce(function(pre,cur){        
            return pre + cur;
    }, 0);
}
console.log(add()) //不传参,结果输出0,则这里已经设置了默认了参数为空数组console.log(add(1,2,3,4,5)) //传参,结果输出15

则作为一个健壮的js插件,我们应该把一些基本的状态参数添加到我们需要的插件上去。
假设还是上面的加减乘除余的需求,我们如何实现插件的默认参数呢?道理其实是一样的。

// plugin.js
;(function(undefined) {
    "use strict"
    var _global;
    function result(args,fn){
        var argsArr = Array.prototype.slice.call(args);
        if(argsArr.length > 0){
            return argsArr.reduce(fn);
        } else {
            return 0;
        }
    }
    var plugin = {
        add: function(){
            return result(arguments,function(pre,cur){
                return pre + cur;
            });
        },//加
        sub: function(){
            return result(arguments,function(pre,cur){
                return pre - cur;
            });
        },//减
        mul: function(){
            return result(arguments,function(pre,cur){
                return pre * cur;
            });
        },//乘
        div: function(){
            return result(arguments,function(pre,cur){
                return pre / cur;
            });
        },//除
        sur: function(){
            return result(arguments,function(pre,cur){
                return pre % cur;
            });
        } //余
    }
    // 最后将插件对象暴露给全局对象
    _global = (function(){ return this || (0, eval)(&#39;this&#39;); }());
    if (typeof module !== "undefined" && module.exports) {
        module.exports = plugin;
    } else if (typeof define === "function" && define.amd) {
        define(function(){return plugin;});
    } else {
        !(&#39;plugin&#39; in _global) && (_global.plugin = plugin);
    }
}());
// 输出结果为:
with(plugin){
    console.log(add()); // 0
    console.log(sub()); // 0
    console.log(mul()); // 0
    console.log(div()); // 0
    console.log(sur()); // 0
    console.log(add(2,1)); // 3
    console.log(sub(2,1)); // 1
    console.log(mul(2,1)); // 2
    console.log(div(2,1)); // 2
    console.log(sur(2,1)); // 0
}

实际上,插件都有自己的默认参数,就以我们最为常见的表单验证插件为例:validate.js

(function(window, document, undefined) {
    // 插件的默认参数
    var defaults = {
        messages: {
            required: &#39;The %s field is required.&#39;,
            matches: &#39;The %s field does not match the %s field.&#39;,
            "default": &#39;The %s field is still set to default, please change.&#39;,
            valid_email: &#39;The %s field must contain a valid email address.&#39;,
            valid_emails: &#39;The %s field must contain all valid email addresses.&#39;,
            min_length: &#39;The %s field must be at least %s characters in length.&#39;,
            max_length: &#39;The %s field must not exceed %s characters in length.&#39;,
            exact_length: &#39;The %s field must be exactly %s characters in length.&#39;,
            greater_than: &#39;The %s field must contain a number greater than %s.&#39;,
            less_than: &#39;The %s field must contain a number less than %s.&#39;,
            alpha: &#39;The %s field must only contain alphabetical characters.&#39;,
            alpha_numeric: &#39;The %s field must only contain alpha-numeric characters.&#39;,
            alpha_dash: &#39;The %s field must only contain alpha-numeric characters, underscores, and dashes.&#39;,
            numeric: &#39;The %s field must contain only numbers.&#39;,
            integer: &#39;The %s field must contain an integer.&#39;,
            decimal: &#39;The %s field must contain a decimal number.&#39;,
            is_natural: &#39;The %s field must contain only positive numbers.&#39;,
            is_natural_no_zero: &#39;The %s field must contain a number greater than zero.&#39;,
            valid_ip: &#39;The %s field must contain a valid IP.&#39;,
            valid_base64: &#39;The %s field must contain a base64 string.&#39;,
            valid_credit_card: &#39;The %s field must contain a valid credit card number.&#39;,
            is_file_type: &#39;The %s field must contain only %s files.&#39;,
            valid_url: &#39;The %s field must contain a valid URL.&#39;,
            greater_than_date: &#39;The %s field must contain a more recent date than %s.&#39;,
            less_than_date: &#39;The %s field must contain an older date than %s.&#39;,
            greater_than_or_equal_date: &#39;The %s field must contain a date that\&#39;s at least as recent as %s.&#39;,
            less_than_or_equal_date: &#39;The %s field must contain a date that\&#39;s %s or older.&#39;
        },
        callback: function(errors) {
        }
    };
    var ruleRegex = /^(.+?)\[(.+)\]$/,
        numericRegex = /^[0-9]+$/,
        integerRegex = /^\-?[0-9]+$/,
        decimalRegex = /^\-?[0-9]*\.?[0-9]+$/,
        emailRegex = /^[a-zA-Z0-9.!#$%&&#39;*+/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$/,
        alphaRegex = /^[a-z]+$/i,
        alphaNumericRegex = /^[a-z0-9]+$/i,
        alphaDashRegex = /^[a-z0-9_\-]+$/i,
        naturalRegex = /^[0-9]+$/i,
        naturalNoZeroRegex = /^[1-9][0-9]*$/i,
        ipRegex = /^((25[0-5]|2[0-4][0-9]|1[0-9]{2}|[0-9]{1,2})\.){3}(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[0-9]{1,2})$/i,
        base64Regex = /[^a-zA-Z0-9\/\+=]/i,
        numericDashRegex = /^[\d\-\s]+$/,
        urlRegex = /^((http|https):\/\/(\w+:{0,1}\w*@)?(\S+)|)(:[0-9]+)?(\/|\/([\w#!:.?+=&%@!\-\/]))?$/,
        dateRegex = /\d{4}-\d{1,2}-\d{1,2}/;
    ... //省略后面的代码
})(window,document);
/*
 * Export as a CommonJS module
 */
if (typeof module !== &#39;undefined&#39; && module.exports) {
    module.exports = FormValidator;
}

当然,参数既然是默认的,那就意味着我们可以随意修改参数以达到我们的需求。插件本身的意义就在于具有复用性。
如表单验证插件,则就可以new一个对象的时候,修改我们的默认参数:

var validator = new FormValidator(&#39;example_form&#39;, [{
    name: &#39;req&#39;,
    display: &#39;required&#39;,
    rules: &#39;required&#39;
}, {
    name: &#39;alphanumeric&#39;,
    rules: &#39;alpha_numeric&#39;
}, {
    name: &#39;password&#39;,
    rules: &#39;required&#39;
}, {
    name: &#39;password_confirm&#39;,
    display: &#39;password confirmation&#39;,
    rules: &#39;required|matches[password]&#39;
}, {
    name: &#39;email&#39;,
    rules: &#39;valid_email&#39;
}, {
    name: &#39;minlength&#39;,
    display: &#39;min length&#39;,
    rules: &#39;min_length[8]&#39;
}, {
    names: [&#39;fname&#39;, &#39;lname&#39;],
    rules: &#39;required|alpha&#39;
}], function(errors) {
    if (errors.length > 0) {
        // Show the errors
    }
});

插件的钩子

我们知道,设计一下插件,参数或者其逻辑肯定不是写死的,我们得像函数一样,得让用户提供自己的参数去实现用户的需求。则我们的插件需要提供一个修改默认参数的入口。
如上面我们说的修改默认参数,实际上也是插件给我们提供的一个API。让我们的插件更加的灵活。如果大家对API不了解,可以百度一下API
通常我们用的js插件,实现的方式会有多种多样的。最简单的实现逻辑就是一个方法,或者一个js对象,又或者是一个构造函数等等。
** 然我们插件所谓的API,实际就是我们插件暴露出来的所有方法及属性。 **
我们需求中,加减乘除余插件中,我们的API就是如下几个方法:

...
var plugin = {
    add: function(n1,n2){ return n1 + n2; },
    sub: function(n1,n2){ return n1 - n2; },
    mul: function(n1,n2){ return n1 * n2; },
    div: function(n1,n2){ return n1 / n2; },
    sur: function(n1,n2){ return n1 % n2; } 
}
...

可以看到plubin暴露出来的方法则是如下几个API:

add

sub

mul

div

sur

在插件的API中,我们常常将容易被修改和变动的方法或属性统称为钩子(Hook),方法则直接叫钩子函数。这是一种形象生动的说法,就好像我们在一条绳子上放很多挂钩,我们可以按需要在上面挂东西。
实际上,我们即知道插件可以像一条绳子上挂东西,也可以拿掉挂的东西。那么一个插件,实际上就是个形象上的链。不过我们上面的所有钩子都是挂在对象上的,用于实现链并不是很理想。

相信看了这些案例你已经掌握了方法,更多精彩请关注php中文网其它相关文章!

相关阅读:

禁止页面缓存有哪些方法

在html里怎么添加flash视频格式(flv、swf)文件

怎样通过disabled和readonly将input设置为只读效果

meta的标签有哪些作用

Das obige ist der detaillierte Inhalt vonSo verwenden Sie natives JS, um die benötigten Plug-Ins zu kapseln. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn