JavaScript有三种声明函数的方法

(1)function命令

function print(s) {
console.log(s);
}

(2)函数表达式

1.var print = function(s) {
console.log(s);
}; 2.var print = function x(){
console.log(typeof x);
}; x
// ReferenceError: x is not defined print()
// function 3.var f = function f() {};

将匿名函数赋予变量,如果不匿名,也只在内部有效

函数表达式定义函数最后要加上分号代表结束

(3)Function构造函数

可以传递任意数量的参数给Function构造函数,但只有最后一个参数被当做函数体

Function构造函数可以不使用new命令,返回结果完全一样

var add = new Function(
'x',
'y',
'return x + y'
); // 等同于
function add(x, y) {
return x + y;
}

此方法比较不常用  第一种方法倒更熟悉

如果一个函数被多次声明,后面的声明就会覆盖前面的声明

调用函数时和其他语言大致相同,使用圆括号运算符

第一等公民 

Js把函数看成一种值,与其他值(数值,字符串,布尔值等)地位相同

凡是可以使用值的地方,就能使用函数,函数只是一个可以执行的值

function add(x, y) {
return x + y;
} // 将函数赋值给一个变量
var operator = add; // 将函数作为参数和返回值
function a(op){
return op;
}
a(add)(1, 1)
// 2

函数提升

 

函数的属性和方法

name属性:返回函数名字  如果采用表达式定义函数,且function后有名字,则返回那个名字

length属性:返回函数预期传入的参数个数

toString():返回一个字符串,内容是函数的源码,以及注释

对于var命令来说,局部变量只能在函数内部声明,在其他区块中声明,一律都是全局变量

函数内部也有变量提升,var命令声明的对象,不管在什么位置,变量声明都会被提升到函数体的头部

函数本身的作用域

函数本身作用域就是声明时所在的作用域,与运行时所在作用域无关

在定义时绑定作用域

var a = 1;
var x = function () {
console.log(a);
}; function f() {
var a = 2;
x();
} f() // 1

函数的参数  

可以省略,但没办法只省略靠前的参数,而保留靠后的参数,如果一定要省略靠前的参数,只有显式传入undefined

参数传递对原值的影响

如果是原始类型的值,无影响,只是一份拷贝

如果传入的是对象,则修改其属性的话,原值会受到影响

但把对象整个替换掉,不会影响原始值,因为相当于指向其他地址

同名参数

如果有同名参数,则取最后出现的那个值

function f(a, a) {
console.log(a); //a为最后一个a
} f(1) // undefined 相当于f(1,undefined)

arguments对象

由于js允许函数提供不定数目的参数,所以需要一种机制,可以在函数体内部读取所有参数,这就是arguments对象的由来

arguments对象包含函数运行时所有参数,使用类似数组

如果在非严格模式下还可以在函数内部修改参数的值,严格模式下只读

通过它的length属性,可以判断函数调用时到底带了几个参数

与数组关系:

是对象,不能使用数组方法,除非转换为数组

var args = Array.prototype.slice.call(arguments);

// 或者
var args = [];
for (var i = 0; i < arguments.length; i++) {
args.push(arguments[i]);
}

arguments自带一个callee属性1,返回它所对应的原函数,可以用这个写递归

arguments每个函数运行时都有,无需显式声明

函数的闭包

闭包即能够读取其他函数内部变量的函数,可以简单理解为“定义在一个函数内部的函数”

用处:1.读取函数内部变量 2.记住诞生的环境

function createIncrementor(start) {
return function () {
return start++;
};
} var inc = createIncrementor(5); inc() // 5
inc() // 6
inc() // 7

 3.封装对象的私有属性和方法,感觉像类 

function Person(name) {
var _age;
function setAge(n) {
_age = n;
}
function getAge() {
return _age;
} return {
name: name,
getAge: getAge,
setAge: setAge
};
} var p1 = Person('张三');
p1.setAge(25);
p1.getAge() // 25

立即调用的函数表达式,声明时就调用(IIFE)

(function(){ /* code */ }());
// 或者
(function(){ /* code */ })();

 所以以下写法都行

var i = function(){ return 10; }();
true && function(){ /* code */ }();
0, function(){ /* code */ }();
!function () { /* code */ }();
~function () { /* code */ }();
-function () { /* code */ }();
+function () { /* code */ }();

通常只对匿名函数使用

目的:1.不必命名,避免污染全局变量 2。在IIFE内部形成了一个单独的作用域,可以封装一些外部无法读取的私有变量

eval命令 

eval命令接受一个字符串作为参数,并将这个字符串当做语句执行

eval('var a = 1;');
a // 1

如果无法执行就报错

如果eval的参数不是字符串,那么会原样返回

eval的作用域是当前作用域,因此小心修改当前作用域变量的值

eval本质是在当前作用域中注入代码,由于安全风险和不利于js引擎优化执行速度,一般不推荐使用

最常用的场合是解析json数据的字符传递,不过正确做法应该使用原生的JSON.parse方法

js规定,凡是使用别名执行eval,eval内部一律是全局作用域

var a = 1;

function f() {
var a = 2;
var e = eval;
e('console.log(a)');
} f() // 1

  

 

  

 

  

  

  

js学习(4) 函数的更多相关文章

  1. js学习之函数表达式及闭包

    来自<javascript高级程序设计 第三版:作者Nicholas C. Zakas>的学习笔记(七) 直接切入主题~ 定义函数的方式有两种: 函数声明 function functio ...

  2. js学习之函数的参数传递

    我们都知道在 ECMAScript 中,数据类型分为原始类型(又称值类型/基本类型)和引用类型(又称对象类型):这里我将按照这两种类型分别对函数进行传参,看一下到底发生了什么. 参数的理解 首先,我们 ...

  3. js学习之函数声明与函数表达式区别[原创]

    作为一名js初学者,与大家分享下.Javascript中有函数声明提升的功能,会优先编译函数声明部分.比如, ff(); function ff(){ alert("hello world. ...

  4. js学习之函数

    1/.js中函数就是对象. 2/以表达式方式定义的函数一般不要函数名以使代码紧凑. 3/js中函数声明的方式会被默认提到外部脚本或最前面,所以在其定义前的代码也可调用. 然而以表达式方式的则不行. 4 ...

  5. js学习笔记 -- 函数

    js函数有类似javaMethod用法 Math.max.apply( Math.max.call( Array map,reduce,filter,sort , , , , , , , , ]; v ...

  6. JS学习之函数内部属性和方法

    知识点:arguments和this对象.caller属性.apply()和call()方法     arguments对象:函数内部对象,传入函数中所有参数的集合,类数组对象 属性:callee 指 ...

  7. js学习-自定义函数、对象的字面量、json对象学习小结

    一.自定义对象的构造: var student=new Object(); //object是顶级对象,使用构造函数的方法创建一个对象,此处的意思是创建了一个学生的空对象 student.name=& ...

  8. js学习:函数

    概述 函数的声明 JavaScript 有三种声明函数的方法 function 命令 function命令声明的代码区块,就是一个函数.function命令后面是函数名,函数名后面是一对圆括号,里面是 ...

  9. node.js学习(2)函数

    1 简答函数 2 匿名函数 3 回调函数

  10. JS学习之函数的属性和方法

随机推荐

  1. zombodb 数据类型映射

    zombodb 与es 数据类型的映射处理 通用数据类型映射 Postgres 类型 Elasticsearch JSON 映射定义 bytea {"type": "bi ...

  2. python 【winerror2】系统找不到指定的路径

    # _*_ coding:utf-8_*_from selenium import webdriver driver = webdriver.Firefox()driver.get("htt ...

  3. 1.2.1 Excel如何从身份证号中提取日期

    在对应的单元格中我们输入公式: =MID(B3,7,4)&"年"&MID(B3,11,2)&"月"&MID(B3,13,2)&a ...

  4. ResourceBundle类的方式来读取config.properties配置文件参数值

    //获取config.properties配置文件参数值 public static ResourceBundle resource = ResourceBundle.getBundle(" ...

  5. Guava 12:Guava EventBus源码剖析

    一.架构速读 传统上,Java的进程内事件分发都是通过发布者和订阅者之间的显式注册实现的.设计EventBus就是为了取代这种显示注册方式,使组件间有了更好的解耦.EventBus不是通用型的发布-订 ...

  6. Hanlp1.7版本的新增功能一览

    Hanlp1.7版本在去年下半年的时候就随大快的DKH1.6版本同时发布了,截至目前1.7大版本也更新到了1.7.1了.本篇分别就1.7.0和1.7.1中新增的功能做一个简单的汇总介绍. HanLP ...

  7. jenkins构建触发器详解-不登录触发远程构建详解

    利用jenkins的远程构建功能,我们可以使用任何脚本,甚至定制一个Web页来控制Job的执行,但是远程构建你如果直接使用的话,老是需要登录才能执行,如何避免登录?稍微折腾了一下,调通了. 1.首先去 ...

  8. Linux操作redis 使用(VMwareWorkstation)

    项目一般都部署到linux上面,记得刚出来的时候,第一家公司 服务器是windows系统,以后公司的项目都放在了linux上面,所以掌握linux的一些基本操作是一个程序员必备的知识,本次记录如何使用 ...

  9. Oracle 关于concat与双竖线用法的补充

    --只能连接2个字符串select concat('nod',' chen is ') from dual; --连接2个列名select concat(name,ip2) from vm_info; ...

  10. PO & SO Integration By IDOC in CNABB

    PO & SO Integration By IDOC in CNABB 话说博主来ABB一个多月时间了,虽然对ABB系统内的流程和配置不是很了解,但对ABB系统内使用的PO和SO通过idoc ...