Я читал много Javascript в последнее время, и я заметил, что весь файл обернут следующим образом в импортируемых файлах .js.
(function() {
...
code
...
})();
В чем причина этого, а не простой набор конструкторских функций?
Я читал много Javascript в последнее время, и я заметил, что весь файл обернут следующим образом в импортируемых файлах .js.
(function() {
...
code
...
})();
В чем причина этого, а не простой набор конструкторских функций?
Обычно это пространство имен (см. ниже) и контролирует видимость функций-членов и/или переменных. Подумайте об этом, как об определении объекта. Плагины jQuery обычно записываются так.
В Javascript вы можете встраивать функции. Итак, законно:
function outerFunction() {
function innerFunction() {
// code
}
}
Теперь вы можете вызвать outerFunction()
, но видимость innerFunction()
ограничена областью outerFunction()
, что означает, что она закрыта для outerFunction()
. Он в основном следует тому же принципу, что и переменные в Javascript:
var globalVariable;
function someFunction() {
var localVariable;
}
Соответственно:
function globalFunction() {
var localFunction1 = function() {
//I'm anonymous! But localFunction1 is a reference to me!
};
function localFunction2() {
//I'm named!
}
}
В приведенном выше сценарии вы можете вызывать globalFunction()
из любого места, но вы не можете вызвать localFunction1
или localFunction2
.
Что вы делаете, когда пишете (function() { ... code ... })()
, вы делаете код внутри литерала функции (что означает, что весь "объект" на самом деле является функцией). После этого вы вызываете свою функцию (окончательный ()
). Поэтому основным преимуществом этого, как я уже упоминал ранее, является то, что вы можете иметь частные методы/функции и свойства:
(function() {
var private_var;
function private_function() {
//code
}
})()
В первом примере globalFunction() была публичной функцией, которая может быть вызвана для доступа к общедоступным функциям, но в приведенном выше примере, как вы это называете? Здесь функция самозапуска делает код автоматически запускаемым при запуске. Так же, как вы можете добавить initMyStuff(); в начало любого .js файла, и он будет автоматически запускаться как часть глобальной области видимости, эта функция самозапуска также будет автоматически запускаться, хотя, поскольку она является неназванной функцией, ее нельзя вызывать несколько раз, например, initMyStuff().
Оптимально то, что вы также можете определять вещи внутри и выставлять их во внешний мир, поэтому (пример пространства имен, чтобы вы могли в принципе создать свою собственную библиотеку/плагин):
var myPlugin = (function() {
var private_var;
function private_function() {
}
return {
public_function1: function() {
},
public_function2: function() {
}
}
})()
Теперь вы можете вызвать myPlugin.public_function1()
, но вы не можете получить доступ к private_function()
! Это довольно похоже на определение класса. Чтобы понять это лучше, я рекомендую следующие ссылки для дальнейшего чтения:
ИЗМЕНИТЬ
Я забыл упомянуть. В этом финале ()
вы можете передать все, что захотите внутри. Например, когда вы создаете плагины jQuery, вы проходите в jQuery
или $
следующим образом:
(function(jQ) { ... code ... })(jQuery)
Итак, что вы здесь делаете, это определение функции, которая принимает один параметр (называемый jQ
, локальная переменная и известный только этой функции). Затем вы вызываете вызов функции и передаете параметр (также называемый jQuery
, но этот из внешнего мира и ссылка на собственно jQuery). Нет надобности делать это, но есть некоторые преимущества:
Раньше я описывал, как эти функции запускаются автоматически при запуске, но если они автоматически запускаются, кто передает аргументы? Этот метод предполагает, что все параметры определяются как глобальные переменные. Поэтому, если jQuery не был определен как глобальная переменная, этот пример не будет работать и не может быть вызван каким-либо другим способом, поскольку наш пример является анонимной функцией. Как вы могли догадаться, одна вещь, которую выполняет jquery.js во время ее инициализации, определяет глобальную переменную 'jQuery', а также более известную глобальную переменную $$, которая позволяет этому коду работать после включения jquery.js.
В своей простейшей форме этот метод нацелен на перенос кода внутри области функций.
Это помогает снизить шансы:
Он не обнаруживает, когда документ готов - это не что-то вроде document.onload
или window.onload
Он широко известен как Immediately Invoked Function Expression (IIFE)
или Immediately Invoked Function Expression (IIFE)
Self Executing Anonymous Function
.
var someFunction = function(){ console.log('wagwan!'); };
(function() { /* function scope starts here */
console.log('start of IIFE');
var myNumber = 4; /* number variable declaration */
var myFunction = function(){ /* function variable declaration */
console.log('formidable!');
};
var myObject = { /* object variable declaration */
anotherNumber : 1001,
anotherFunc : function(){ console.log('formidable!'); }
};
console.log('end of IIFE');
})(); /* function scope ends */
someFunction(); // reachable, hence works: see in the console
myFunction(); // unreachable, will throw an error, see in the console
myObject.anotherFunc(); // unreachable, will throw an error, see in the console
В приведенном выше примере любая переменная, определенная в функции (т.е. Объявленная с использованием var
), будет "закрыта" и доступна только в пределах области функции (как это делает Вивиан Палиат). Другими словами, эти переменные не видны/недоступны вне функции. Смотрите демо-версию.
Javascript имеет функцию определения области. "Параметры и переменные, определенные в функции, не видны вне функции и что переменная, определенная где угодно внутри функции, видима всюду внутри функции". (от "Javascript: хорошие части").
В конце концов, код, который был ранее представлен, можно также сделать следующим образом:
var someFunction = function(){ console.log('wagwan!'); };
var myMainFunction = function() {
console.log('start of IIFE');
var myNumber = 4;
var myFunction = function(){ console.log('formidable!'); };
var myObject = {
anotherNumber : 1001,
anotherFunc : function(){ console.log('formidable!'); }
};
console.log('end of IIFE');
};
myMainFunction(); // I CALL "myMainFunction" FUNCTION HERE
someFunction(); // reachable, hence works: see in the console
myFunction(); // unreachable, will throw an error, see in the console
myObject.anotherFunc(); // unreachable, will throw an error, see in the console
Однажды кто-то, вероятно, подумал: "Должен быть способ избежать наименования" myMainFunction ", поскольку все, что мы хотим, это немедленно выполнить его".
Если вы вернетесь к основам, вы узнаете, что:
expression
: что-то оценивающее значение. т.е. 3+11/x
statement
: строка (коды) кода делает что-то, но не оценивает значение. т.е. if(){}
Аналогично, выражения функций оцениваются до значения. И одно из последствий (я предполагаю?) Заключается в том, что они могут быть немедленно вызваны:
var italianSayinSomething = function(){ console.log('mamamia!'); }();
Итак, наш более сложный пример:
var someFunction = function(){ console.log('wagwan!'); };
var myMainFunction = function() {
console.log('start of IIFE');
var myNumber = 4;
var myFunction = function(){ console.log('formidable!'); };
var myObject = {
anotherNumber : 1001,
anotherFunc : function(){ console.log('formidable!'); }
};
console.log('end of IIFE');
}();
someFunction(); // reachable, hence works: see in the console
myFunction(); // unreachable, will throw an error, see in the console
myObject.anotherFunc(); // unreachable, will throw an error, see in the console
Следующим шагом является мысль "почему var myMainFunction =
если мы даже не используем его !?".
Ответ прост: попробуйте удалить это, например, ниже:
function(){ console.log('mamamia!'); }();
Это не сработает, потому что "объявления функций не вызываются".
Фокус в том, что, удалив var myMainFunction =
мы преобразовали выражение функции в объявление функции. См. Ссылки в разделе "Ресурсы" для получения дополнительной информации об этом.
Следующий вопрос: "Почему я не могу сохранить его как выражение функции с чем-то другим, кроме var myMainFunction =
?
Ответ "вы можете", и есть на самом деле много способов сделать это: добавив +
, a !
, a -
, или, может быть, обертывание в пару круглых скобок (как это теперь делается по соглашению), и больше я верю. В качестве примера:
(function(){ console.log('mamamia!'); })(); // live demo: jsbin.com/zokuwodoco/1/edit?js,console.
или
+function(){ console.log('mamamia!'); }(); // live demo: jsbin.com/wuwipiyazi/1/edit?js,console
или
-function(){ console.log('mamamia!'); }(); // live demo: jsbin.com/wejupaheva/1/edit?js,console
Поэтому, как только соответствующая модификация добавляется к тому, что когда-то было нашим "Альтернативным кодом", мы возвращаемся к тому же самому коду, что и тот, который используется в примере "Code Explained"
var someFunction = function(){ console.log('wagwan!'); };
(function() {
console.log('start of IIFE');
var myNumber = 4;
var myFunction = function(){ console.log('formidable!'); };
var myObject = {
anotherNumber : 1001,
anotherFunc : function(){ console.log('formidable!'); }
};
console.log('end of IIFE');
})();
someFunction(); // reachable, hence works: see in the console
myFunction(); // unreachable, will throw an error, see in the console
myObject.anotherFunc(); // unreachable, will throw an error, see in the console
Подробнее о Expressions vs Statements
:
Можно только удивляться: "что происходит, когда вы НЕ определяете переменную" правильно "внутри функции, т.е. Вместо этого выполняете простое назначение?"
(function() {
var myNumber = 4; /* number variable declaration */
var myFunction = function(){ /* function variable declaration */
console.log('formidable!');
};
var myObject = { /* object variable declaration */
anotherNumber : 1001,
anotherFunc : function(){ console.log('formidable!'); }
};
myOtherFunction = function(){ /* oops, an assignment instead of a declaration */
console.log('haha. got ya!');
};
})();
myOtherFunction(); // reachable, hence works: see in the console
window.myOtherFunction(); // works in the browser, myOtherFunction is then in the global scope
myFunction(); // unreachable, will throw an error, see in the console
В принципе, если переменной, которая не была объявлена в ее текущей области, присваивается значение, тогда "поиск цепи видимости происходит до тех пор, пока она не найдет переменную или не попадет в глобальную область (в какой момент она ее создаст)".
Когда в среде браузера (против серверной среды, такой как nodejs) глобальная область видимости определяется объектом window
. Следовательно, мы можем сделать window.myOtherFunction()
.
Мой совет "Хорошая практика" в этой теме - всегда использовать var
при определении чего-либо: будь то число, объект или функция и даже в глобальном масштабе. Это делает код намного проще.
Заметка:
block scope
(Update: локальные переменные области блока добавлены в ES6.)function scope
& global scope
(область window
в среде браузера) Узнайте больше о области Javascript Scopes
:
Как только вы получите эту концепцию IIFE
, она приведет к module pattern
, который обычно делается за счет использования этого шаблона IIFE. Повеселись :)
Javascript только в браузере действительно имеет несколько эффективных областей: область действия и глобальная область.
Если переменная не находится в области функций, она находится в глобальной области. И глобальные переменные, как правило, плохие, поэтому это конструкция, позволяющая хранить переменные библиотеки для себя.
Это называется замыканием. Он в основном запечатывает код внутри функции, чтобы другие библиотеки не мешали ему. Это похоже на создание пространства имен в скомпилированных языках.
Пример. Предположим, что я пишу:
(function() {
var x = 2;
// do stuff with x
})();
Теперь другие библиотеки не могут получить доступ к переменной x
, которую я создал для использования в моей библиотеке.
Вы можете использовать закрытие функций как datastrong > в больших выражениях, как в этом методе определения поддержки браузера для некоторых объектов html5.
navigator.html5={
canvas: (function(){
var dc= document.createElement('canvas');
if(!dc.getContext) return 0;
var c= dc.getContext('2d');
return typeof c.fillText== 'function'? 2: 1;
})(),
localStorage: (function(){
return !!window.localStorage;
})(),
webworkers: (function(){
return !!window.Worker;
})(),
offline: (function(){
return !!window.applicationCache;
})()
}
В дополнение к сохранению переменных локально, очень удобно использовать при написании библиотеки с использованием глобальной переменной, вы можете дать ей более короткое имя переменной для использования в библиотеке. Он часто используется при написании плагинов jQuery, поскольку jQuery позволяет отключить переменную $, указывающую на jQuery, используя jQuery.noConflict(). Если он отключен, ваш код все равно может использовать $и не прерывать, если вы просто выполните:
(function($) { ...code...})(jQuery);
Мы также должны использовать "use strict" в функции scope, чтобы убедиться, что код должен быть выполнен в "строгом режиме". Пример кода, показанного ниже
(function() {
'use strict';
//Your code from here
})();
В ES6 вы можете использовать обычный блок фигурных скобок, чтобы сохранить код вне глобального пространства имен (var
сильно обескуражен!):
{
'use strict';
your stuff here
}
Протестировано в Chrome, Safari, Safari iOS, Firefox. Я понятия не имею, имеет ли этот шаблон имя, будь то хорошая практика или почему я не мог сначала найти ссылку на него вообще (до указанной выше статьи).