目录总览
1、严格模式
JavaScript 除了提供正常模式外,还提供了严格模式
ES5 的严格模式是采用具有限制性 JavaScript 变体的一种方式,即在严格的条件下运行 JS 代码
严格模式在 IE10 以上版本的浏览器才会被支持,旧版本浏览器会被忽略
严格模式对正常的 JavaScript 语义做了一些更改:
消除了 Javascript 语法的一些不合理、不严谨之处,减少了一些怪异行为
消除代码运行的一些不安全之处,保证代码运行的安全
提高编译器效率,增加运行速度
禁用了在 ECMAScript 的未来版本中可能会定义的一些语法,为未来新版本的 Javascript 做好铺垫。比如一些保留字如:class, enum, export, extends, import, super 不能做变量名
1.1、开启严格模式
严格模式可以应用到整个脚本或个别函数中。
因此在使用时,我们可以将严格模式分为为脚本开启严格模式和为函数开启严格模式两种情况
1.1.2、为脚本开启严格模式
为整个脚本文件开启严格模式,需要在所有语句之前放一个特定语句
"use strict"
或'use strict'
1 <script>'user strict' ; console .log ("这是严格模式。" );</script>
因为"use strict"
加了引号,所以老版本的浏览器会把它当作一行普通字符串而忽略。
有的 script 基本是严格模式,有的 script 脚本是正常模式,这样不利于文件合并,所以可以将整个脚本文件放在一个立即执行的匿名函数之中。这样独立创建一个作用域而不影响其他 script 脚本文件。
1 2 3 4 5 6 7 <script> (function ( ){ 'use strict' ; var num = 10 ; function fn ( ) {} })(); </script>
1.1.2、为函数开启严格模式
若要给某个函数开启严格模式,需要把"use strict"
或'use strict'
声明放在函数体所有语句之前
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 <body> <!-- 为整个脚本(script标签)开启严格模式 --> <script > 'use strict' ; </script > <script > (function ( ) { 'use strict' ; })(); </script > <!-- 为某个函数开启严格模式 --> <script > function fn ( ) { 'use strict' ; } function fun ( ) { } </script > </body>
将"use strict"
放在函数体的第一行,则整个函数以 “严格模式”运行。
2、严格模式中的变化
严格模式对 JavaScript 的语法和行为,都做了一些改变
2.1、变量规定
在正常模式中,如果一个变量没有声明就赋值,默认是全局变量
严格模式禁止这种用法,变量都必须先用 var 命令声明,然后再使用
严禁删除已经声明变量,例如,``delete x` 语法是错误的
1 2 3 4 5 6 7 8 9 10 11 12 <body > <script > 'use strict' var num = 10 console .log (num) </script > </body >
2.2、严格模式下 this 指向问题
以前在全局作用域函数中的this
指向window
对象
严格模式下全局作用域中函数中的this
是 undefined
以前构造函数时不加 new
也可以调用,当普通函数,this
指向全局对象
严格模式下,如果构造函数不加 new
调用,this
指向的是 undefined
,如果给它赋值,会报错
new
实例化的构造函数指向创建的对象实例
定时器this
还是指向window
事件、对象还是指向调用者
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 <body > <script > 'use strict' function fn ( ) { console .log (this ) } fn () function Star ( ) { this .sex = '男' } var ldh = new Star () console .log (ldh.sex ) setTimeout (function ( ) { console .log (this ) }, 2000 ) </script > </body >
2.3、函数变化
函数不能有重名的参数
函数必须声明在顶层,新版本的 JavaScript 会引入“块级作用域”(ES6 中已引入)。为了与新版本接轨,不允许在非函数的代码块内声明函数
1 2 3 4 5 6 7 8 9 10 11 12 <body> <script > 'use strict' ; function fn (a, a ) { console .log (a + a); }; function fn ( ) {} </script > </body>
3、高阶函数
高阶函数是对其他函数进行操作的函数,它接收函数作为参数或将函数作为返回值输出
接收函数作为参数
1 2 3 4 5 6 7 8 9 10 11 12 13 <body > <div > </div > <script > function fn (a, b, callback ) { console .log (a + b) callback && callback () } fn (1 , 2 , function ( ) { console .log ('我是最后调用的' ) }) </script > </body >
将函数作为返回值
1 2 3 4 5 <script > function fn ( ) { return function ( ) {} } </script >
此时 fn 就是一个高阶函数
函数也是一种数据类型,同样可以作为参数,传递给另外一个参数使用。最典型的就是作为回调函数
同理函数也可以作为返回值传递回来
4、闭包 4.1、变量作用域 变量根据作用域的不同分为两种:全局变量和局部变量
函数内部可以使用全局变量
函数外部不可以使用局部变量
当函数执行完毕,本作用域内的局部变量会销毁。
4.2、什么是闭包 闭包指有权访问另一个函数作用域中的变量的函数
简单理解:一个作用域可以访问另外一个函数内部的局部变量
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 <body > <script > function fn1 ( ) { var num = 10 function fn2 ( ) { console .log (num) } fn2 () } fn1 () </script > </body >
4.3、在 chrome 中调试闭包
打开浏览器,按 F12 键启动 chrome 调试工具。
设置断点。
找到 Scope 选项(Scope 作用域的意思)。
当我们重新刷新页面,会进入断点调试,Scope 里面会有两个参数(global 全局作用域、local 局部作用域)。
当执行到 fn2() 时,Scope 里面会多一个 Closure 参数 ,这就表明产生了闭包。
4.4、闭包的作用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 <body> <script > function fn ( ) { var num = 10 ; return function ( ) { console .log (num); } } var f = fn (); f (); </script > </body>
4.5、闭包练习 4.5.1、点击 li 输出索引号 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 <body > <ul class ="nav" > <li > 榴莲</li > <li > 臭豆腐</li > <li > 鲱鱼罐头</li > <li > 大猪蹄子</li > </ul > <script > var lis = document .querySelector ('.nav' ).querySelectorAll ('li' ) for (var i = 0 ; i < lis.length ; i++) { lis[i].index = i lis[i].onclick = function ( ) { console .log (this .index ) } } for (var i = 0 ; i < lis.length ; i++) { ;(function (i ) { lis[i].onclick = function ( ) { console .log (i) } })(i) } </script > </body >
4.5.2、定时器中的闭包 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 <body > <ul class ="nav" > <li > 榴莲</li > <li > 臭豆腐</li > <li > 鲱鱼罐头</li > <li > 大猪蹄子</li > </ul > <script > var lis = document .querySelector ('.nav' ).querySelectorAll ('li' ) for (var i = 0 ; i < lis.length ; i++) { ;(function (i ) { setTimeout (function ( ) { console .log (lis[i].innerHTML ) }, 3000 ) })(i) } </script > </body >
5、递归 如果一个函数在内部可以调用其本身,那么这个函数就是递归函数
简单理解: 函数内部自己调用自己,这个函数就是递归函数
由于递归很容易发生”栈溢出”错误,所以必须要加退出条件 return
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 <body > <script > var num = 1 function fn ( ) { console .log ('我要打印6句话' ) if (num == 6 ) { return } num++ fn () } fn () </script > </body >
6、浅拷贝和深拷贝
浅拷贝只是拷贝一层,更深层次对象级别的只拷贝引用
深拷贝拷贝多层,每一级别的数据都会拷贝
Object.assign(target,....sources)
ES6 新增方法可以浅拷贝
6.1、浅拷贝 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 var obj = { id : 1 , name : 'andy' , msg : { age : 18 } }; var o = {}for (var k in obj){ o[k] = obj.[k]; } console .log (o);Object .assign (o,obj);
6.2、深拷贝 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 var obj = { id : 1 , name : 'andy' , msg : { age : 18 } color : ['pink' ,'red' ] }; var o = {};function deepCopy (newobj,oldobj ){ for (var k in oldobj){ var item = obldobj[k]; if (item instanceof Array ){ newobj[k] = []; deepCopy (newobj[k],item) }else if (item instanceof Object ){ newobj[k] = {}; deepCopy (newobj[k],item) }else { newobj[k] = item; } } } deepCopy (o,obj);
7、 正则表达式 正则表达式是用于匹配字符串中字符组合的模式。在 JavaScript 中,正则表达式也是对象。
正则表通常被用来检索、替换那些符合某个模式(规则)的文本,例如验证表单:用户名表单只能输入英文字母、数字或者下划线, 昵称输入框中可以输入中文(匹配)。此外,正则表达式还常用于过滤掉页面内容中的一些敏感词(替换),或从字符串中获取我们想要的特定部分(提取)等 。
7.1、特点
实际开发,一般都是直接复制写好的正则表达式
但是要求会使用正则表达式并且根据自身实际情况修改正则表达式
7.2、创建正则表达式 在 JavaScript 中,可以通过两种方式创建正则表达式
通过调用 RegExp 对象的构造函数创建
通过字面量创建
7.2.1、通过调用 RegExp 对象的构造函数创建 通过调用 RegExp 对象的构造函数创建
1 var 变量名 = new RegExp (/表达式/ )
7.2.2、通过字面量创建 通过字面量创建
注释中间放表达式就是正则字面量
7.2.3、测试正则表达式 test
test()
正则对象方法,用于检测字符串是否符合该规则,该对象会返回true
或false
,其参数是测试字符串
regexObj
写的是正则表达式
str
我们要测试的文本
就是检测str
文本是否符合我们写的正则表达式规范
示例
1 2 3 <body> <script > </script > </body>
7.3、正则表达式中的特殊字符 7.3.1、边界符 正则表达式中的边界符(位置符)用来提示字符所处的位置,主要有两个字符
边界符
说明
^
表示匹配行首的文本(以谁开始)
$
表示匹配行尾的文本(以谁结束)
如果^ 和 $ 在一起,表示必须是精确匹配
1 2 3 4 5 6 7 8 9 10 11 12 13 14 var rg = /abc/ console .log (rg.test ('abc' ))console .log (rg.test ('abcd' ))console .log (rg.test ('aabcd' ))var reg = /^abc/ console .log (reg.test ('abc' )) console .log (reg.test ('abcd' )) console .log (reg.test ('aabcd' )) var reg1 = /^abc$/
7.3.2、字符类
字符类表示有一系列字符可供选择,只要匹配其中一个就可以了
所有可供选择的字符都放在方括号内
①[] 方括号
后面的字符串只要包含 abc
中任意一个字符,都返回true
②[-]方括号内部 范围符
方括号内部加上 - 表示范围,这里表示 a - z 26 个英文字母都可以
③[^] 方括号内部 取反符 ^
方括号内部加上 ^ 表示取反,只要包含方括号内的字符,都返回 false
注意和边界符 ^ 区别,边界符写到方括号外面
④ 字符组合 1 ;/[a-z1-9]/ .test ('andy' )
方括号内部可以使用字符组合,这里表示包含 a 到 z 的 26 个英文字母和 1 到 9 的数字都可以
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 <body > <script > var rg = /[abc]/ console .log (rg.test ('andy' )) console .log (rg.test ('baby' )) console .log (rg.test ('color' )) console .log (rg.test ('red' )) var rg1 = /^[abc]$/ console .log (rg1.test ('aa' )) console .log (rg1.test ('a' )) console .log (rg1.test ('b' )) console .log (rg1.test ('c' )) console .log (rg1.test ('abc' )) console .log ('------------------' ) var reg = /^[a-z]$/ console .log (reg.test ('a' )) console .log (reg.test ('z' )) console .log (reg.test (1 )) console .log (reg.test ('A' )) var reg1 = /^[a-zA-Z0-9_-]$/ console .log (reg1.test ('a' )) console .log (reg1.test ('B' )) console .log (reg1.test (8 )) console .log (reg1.test ('-' )) console .log (reg1.test ('_' )) console .log (reg1.test ('!' )) console .log ('----------------' ) var reg2 = /^[^a-zA-Z0-9_-]$/ console .log (reg2.test ('a' )) console .log (reg2.test ('B' )) console .log (reg2.test (8 )) console .log (reg2.test ('-' )) console .log (reg2.test ('_' )) console .log (reg2.test ('!' )) </script > </body >
7.3.3、量词符 量词符用来设定某个模式出现的次数
量词
说明
*
重复零次或更多次
+
重复一次或更多次
?
重复零次或一次
{n}
重复 n 次
{n,}
重复 n 次或更多次
{n,m}
重复 n 到 m 次
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 <body> <script > var reg = /^a{3,}$/ ; console .log (reg.test ('' )); console .log (reg.test ('a' )); console .log (reg.test ('aaaa' )); console .log (reg.test ('aaa' )); var reg = /^a{3,6}$/ ; console .log (reg.test ('' )); console .log (reg.test ('a' )); console .log (reg.test ('aaaa' )); console .log (reg.test ('aaa' )); console .log (reg.test ('aaaaaaa' )); </script > </body>
7.3.4、用户名验证 功能需求:
如果用户名输入合法, 则后面提示信息为 : 用户名合法,并且颜色为绿色
如果用户名输入不合法, 则后面提示信息为: 用户名不符合规范, 并且颜色为绿色
分析:
用户名只能为英文字母,数字,下划线或者短横线组成, 并且用户名长度为 6~16 位.
首先准备好这种正则表达式模式 /$[a-zA-Z0-9-_]{6,16}^/
当表单失去焦点就开始验证.
如果符合正则规范, 则让后面的 span 标签添加 right 类.
如果不符合正则规范, 则让后面的 span 标签添加 wrong 类.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 <body > <input type ="text" class ="uname" /> <span > 请输入用户名</span > <script > var reg = /^[a-zA-Z0-9_-]{6,16}$/ var uname = document .querySelector ('.uname' ) var span = document .querySelector ('span' ) uname.onblur = function ( ) { if (reg.test (this .value )) { console .log ('正确的' ) span.className = 'right' span.innerHTML = '用户名格式输入正确' } else { console .log ('错误的' ) span.className = 'wrong' span.innerHTML = '用户名格式输入不正确' } } </script > </body >
7.4、括号总结
大括号 量词符 里面面表示重复次数
中括号 字符集合 匹配方括号中的任意字符
小括号 表示优先级
1 2 3 4 5 6 7 var reg = /^[abc]$/ var reg = /^abc{3}$/ var reg = /^(abc){3}$/
在线测试正则表达式:https://c.runoob.com/
7.5、预定义类 预定义类指的是 某些常见模式的简写写法
预定类
说明
\d
匹配 0-9 之间的任一数字,相当于[0-9]
\D
匹配所有 0-9 以外的字符,相当于[ ^ 0-9]
\w
匹配任意的字母、数字和下划线,相当于[A-Za-z0-9_ ]
\W
除所有字母、数字、和下划线以外的字符,相当于[ ^A-Za-z0-9_ ]
\s
匹配空格(包括换行符,制表符,空格符等),相当于[\t\t\n\v\f]
\S
匹配非空格的字符,相当于[ ^ \t\r\n\v\f]
7.5.1、表单验证 分析:
1.手机号码: /^1[3|4|5|7|8][0-9]{9}$/
,1[3456789]\d{9}
2.QQ: [1-9][0-9]{4,}
(腾讯 QQ 号从 10000 开始)
3.昵称是中文: ^[\u4e00-\u9fa5]{2,8}$
4.邮箱:[0-9a-zA-Z_.-]+[@][0-9a-zA-Z_.-]+([.][a-zA-Z]+){1,2}
1 2 3 4 5 6 7 8 <body > <script > var reg = /^\d{3,4}-\d{7,8}$/ </script > </body >
7.6、正则表达式中的替换 7.6.1、replace 替换 replace()
方法可以实现替换字符串操作,用来替换的参数可以是一个字符串或是一个正则表达式
1 stringObject.replace (regexp / substr, replacement)
第一个参数: 被替换的字符串或者正则表达式
第二个参数:替换为的字符串
返回值是一个替换完毕的新字符串
1 2 3 4 var str = 'andy和red' var newStr = str.replace ('andy' , 'baby' )var newStr = str.replace (/andy/ , 'baby' )
7.6.2、正则表达式参数
switch
按照什么样的模式来匹配,有三种
g
: 全局匹配
i
:忽略大小写
gi
: 全局匹配 + 忽略大小写
8、ES6 的新增语法 8.1、let ES6 中新增的用于声明变量的关键字。
1 2 3 4 if (true ) { let a = 10 } console .log (a)
注意:使用 let 关键字声明的变量才具有块级作用域,使用 var 声明的变量不具备块级作用域特性。
1 2 console .log (a);let a = 20 ;
1 2 3 4 5 var num = 10 if (true ) { console .log (num) let num = 20 }
8.2、const 作用:声明常量,常量就是值(内存地址)不能变化的量。
1 2 3 4 if (true ) { const a = 10 } console .log (a)
1 2 const PI = 3.14 PI = 100
1 2 3 4 5 6 7 const ary = [100 , 200 ]ary[0 ] = 'a' ary[1 ] = 'b' console .log (ary) ary = ['a' , 'b' ]
8.3、let、const、var 的区别 1.使用 var 声明的变量,其作用域为该语句所在的函数内,且存在变量提升现象。 2.使用 let 声明的变量,其作用域为该语句所在的代码块内,不存在变量提升。 3.使用 const 声明的是常量,在后面出现的代码中不能再修改该常量的值。
let
var
const
函数级作用域
块级作用域
块级作用域
变量提升
不存在变量提升
不存在变量提升
值可更改
值可更改
值不可更改
8.4、解构赋值 ES6 中允许从数组中提取值,按照对应位置,对变量赋值。对象也可以实现解构。
8.4.1、数组解构 1 2 3 4 let [a, b, c] = [1 , 2 , 3 ]console .log (a)console .log (b)console .log (c)
如果解构不成功,变量的值为 undefined。
1 2 let [foo] = []let [bar, foo] = [1 ]
8.4.2、对象解构 1 2 3 4 let person = { name : 'zhangsan' , age : 20 }let { name, age } = personconsole .log (name) console .log (age)
1 2 3 let { name : myName, age : myAge } = person console .log (myName) console .log (myAge)
8.5、箭头函数 ES6 中新增的定义函数的方式。
函数体中只有一句代码,且代码的执行结果就是返回值,可以省略大括号
1 2 3 4 function sum (num1, num2 ) { return num1 + num2 } const sum = (num1, num2 ) => num1 + num2
如果形参只有一个,可以省略小括号
1 2 3 4 function fn (v ) { return v } const fn = v => v
箭头函数不绑定 this 关键字,箭头函数中的 this,指向的是函数定义位置的上下文 this。
1 2 3 4 5 6 7 8 9 const obj = { name : '张三' }function fn ( ) { console .log (this ) return () => { console .log (this ) } } const resFn = fn.call (obj)resFn ()
对象没有作用域,利用构造函数创建对象有作用域
8.6、剩余参数 剩余参数语法允许我们将一个不定数量的参数表示为一个数组。
1 2 3 4 5 function sum (first, ...args ) { console .log (first) console .log (args) } sum (10 , 20 , 30 )
ES6 之前是 arguments
剩余参数和解构配合使用
1 2 3 4 let students = ['wangwu' , 'zhangsan' , 'lisi' ]let [s1, ...s2] = studentsconsole .log (s1) console .log (s2)
8.7、Array 的扩展方法 8.7.1、扩展运算符(展开语法) 扩展运算符可以将数组或者对象转为用逗号分隔的参数序列。
1 2 3 4 let ary = [1 , 2 , 3 ];...ary console .log (...ary); console .log (1 , 2 , 3 )
扩展运算符可以应用于合并数组。
1 2 3 4 5 6 let ary1 = [1 , 2 , 3 ]let ary2 = [3 , 4 , 5 ]let ary3 = [...ary1, ...ary2]ary1.push (...ary2)
将类数组或可遍历对象转换为真正的数组
1 2 let oDivs = document .getElementsByTagName ('div' )oDivs = [...oDivs]
8.7.2、构造函数方法: Array.from() 将类数组或可遍历对象转换为真正的数组
1 2 3 4 5 6 7 let arrayLike = { 0 : 'a' , 1 : 'b' , 2 : 'c' , length : 3 } let arr2 = Array .from (arrayLike)
方法还可以接受第二个参数,作用类似于数组的 map 方法,用来对每个元素进行处理,将处理后的值放入返回的数组。
1 2 3 4 5 6 let arrayLike = { 0 : 1 , 1 : 2 , length : 2 } let newAry = Array .from (arrayLike, item => item * 2 )
8.7.3、实例方法:find() 用于找出第一个符合条件的数组成员,如果没有找到返回 undefined
1 2 3 4 5 6 7 8 9 10 11 let ary = [ { id : 1 , name : '张三' }, { id : 2 , name : '李四' } ] let target = ary.find ((item, index ) => item.id == 2 )
8.7.4、实例方法: findlndex() 用于找出第一个符合条件的数组成员的位置,如果没有找到返回-1
1 2 3 let ary = {1 , 5 , 10 , 15 };let index = ary.findIndex ((value, index ) => value > 9 );console .log (index);
8.7.5、实例方法: includes() 表示某个数组是否包含给定的值,返回布尔值。
1 2 3 ;[1 , 2 , 3 ] .includes (2 ) [(1 , 2 , 3 )].includes (4 )
8.8、String 的扩展方法 8.8.1、模板字符串 ES6 新增的创建字符串的方法,使用反引号定义。
模板字符串中可以解析变量。
1 2 let name = '张三' let sayHello = `hello,my name is ${name} `
模板字符串可以换行
1 2 3 4 5 6 7 8 9 10 let result = { name : 'zhangsan' , age : 20 , sex : '男' } let html = `<div> <span>${result.name} </span> <span>${result.age} </span> <span>${result.sex} </span> </div>`
模板字符串可以调用函数
1 2 3 4 5 const sayHello = function ( ) { return '测试' } let greet = `${sayHello} 测试` console .log (greet)
8.8.2、实例方法: startsWith()和 endsWith() startsWith()
:表示参数字符串是否在原字符串的头部,返回布尔值endsWith()
:表示参数字符串是否在原字符串的尾部,返回布尔值
1 2 3 let str = 'Hello world!' str.startsWith ('Hello' ) str.endsWith ('!' )
8.8.3、实例方法:repeat() repeat 方法表示将原字符串重复 n 次,返回一个新字符串。
1 2 'x' .repeat (3 ) 'hello' .repeat (2 )
8.9、Set 数据结构 ES6 提供了新的数据结构 Set。它类似于数组,但是成员的值都是唯一的,没有重复的值。
Set 本身是一个构造函数,用来生成 Set 数据结构。
Set 函数可以接受一个数组作为参数,用来初始化。
1 const set = new Set ([1 , 2 , 3 , 4 , 4 ])
可以用来数组去重
8.9.1、实例方法
add(value)
: 添加某个值,返回 Set 结构本身
delete(value)
: 删除某个值,返回一个布尔值,表示删除是否成功
has(value)
: 返回一个布尔值,表示该值是否为 Set 的成员
clear()
: 清除所有成员,没有返回值
1 2 3 4 5 const s = new Set ()s.add (1 ).add (2 ).add (3 ) s.delete (2 ) s.has (1 ) s.clear ()
8.9.2、遍历 Set 结构的实例与数组一样,也拥有 forEach 方法,用于对每个成员执行某种操作,没有返回值。
1 s.forEach (value => console .log (value))