js学习(4) 函数
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) 函数的更多相关文章
- js学习之函数表达式及闭包
来自<javascript高级程序设计 第三版:作者Nicholas C. Zakas>的学习笔记(七) 直接切入主题~ 定义函数的方式有两种: 函数声明 function functio ...
- js学习之函数的参数传递
我们都知道在 ECMAScript 中,数据类型分为原始类型(又称值类型/基本类型)和引用类型(又称对象类型):这里我将按照这两种类型分别对函数进行传参,看一下到底发生了什么. 参数的理解 首先,我们 ...
- js学习之函数声明与函数表达式区别[原创]
作为一名js初学者,与大家分享下.Javascript中有函数声明提升的功能,会优先编译函数声明部分.比如, ff(); function ff(){ alert("hello world. ...
- js学习之函数
1/.js中函数就是对象. 2/以表达式方式定义的函数一般不要函数名以使代码紧凑. 3/js中函数声明的方式会被默认提到外部脚本或最前面,所以在其定义前的代码也可调用. 然而以表达式方式的则不行. 4 ...
- js学习笔记 -- 函数
js函数有类似javaMethod用法 Math.max.apply( Math.max.call( Array map,reduce,filter,sort , , , , , , , , ]; v ...
- JS学习之函数内部属性和方法
知识点:arguments和this对象.caller属性.apply()和call()方法 arguments对象:函数内部对象,传入函数中所有参数的集合,类数组对象 属性:callee 指 ...
- js学习-自定义函数、对象的字面量、json对象学习小结
一.自定义对象的构造: var student=new Object(); //object是顶级对象,使用构造函数的方法创建一个对象,此处的意思是创建了一个学生的空对象 student.name=& ...
- js学习:函数
概述 函数的声明 JavaScript 有三种声明函数的方法 function 命令 function命令声明的代码区块,就是一个函数.function命令后面是函数名,函数名后面是一对圆括号,里面是 ...
- node.js学习(2)函数
1 简答函数 2 匿名函数 3 回调函数
- JS学习之函数的属性和方法
随机推荐
- H3C交换机配置vlan
一,内存二,硬盘(分区,数据量大小)三,电源线,网络线四,raid(raid0,raid1,raid5)五,装系统(系统版本,分区)六,配置网络 1.创建用户 system-view #进入配置loc ...
- tcpdump抓包常用参数和用法
tcpdump 与wireshark Wireshark(以前是ethereal)是Windows下非常简单易用的抓包工具.但在Linux下很难找到一个好用的图形化抓包工具.还好有Tcpdump.我们 ...
- USB HID设备报告描述符详解(转)
转自:http://group.ednchina.com/93/198.aspx. 参考:USB HID usage table 概述: 报告在这里意思是数据传输(data transfer),而 ...
- MySQL Antelope和Barracuda的区别分析
Antelope是innodb-base的文件格式,Barracude是innodb-plugin后引入的文件格式,同时Barracude也支持Antelope文件格式.两者区别在于: 文件格式 支持 ...
- mybatis的简单使用调用mapper接口
mybatis 是apache下的一个面向sql编程的半自动化的ORM持久层的框架.特点:面向sql编程,达到高性能的使用目的. 下面是简单使用 现导入jar包,只有mybatis和数据库驱动包(这里 ...
- 可空类型(Nullable)
C# 单问号 ? 与 双问号 ?? ? : 单问号用于对 int,double,bool 等无法直接赋值为 null 的数据类型进行 null 的赋值,意思是这个数据类型是 NullAble 类型的. ...
- java中Arrays的用法
Arrays是针对数组的工具类,可以进行 排序,查找,复制填充等功能. 大大提高了开发人员的工作效率. 一 数组复制 与使用System.arraycopy进行数组复制类似的, Arrays提供了一个 ...
- Spring的生命周期
转:https://blog.csdn.net/liuxilil/article/details/4676088 Spring的生命周期. 容器启动,实例化所有实现了BeanFactoyPostPro ...
- Mac OS X系统 用dd命令将iso镜像写入u盘
一. Mac下将ISO写入U盘可使用命令行工具dd,操作如下: 1.找出U盘挂载的路径,使用如下命令:diskutil list2.将U盘unmount(将N替换为挂载路径):diskutil unm ...
- spring 之 类型转换
在spring中, 提供了至少三种的 类型转换方式: ConversionServiceFactoryBean, FormattingConversionServiceFactoryBean, C ...