js基础
js三座大山:原型和原型链 作用域和闭包 异步和单线程
引用类型:对象 数组 函数 因为指针引用的是一个空间 所以叫做引用类型 目的是为了避免浪费很大的内存空间 引用类型可以无限制扩展属性
typeof (undefined string number boolean object function)只能区分值类型 但是不能区分引用类型 对象和数组{}[]null均打印object
变量计算:强制类型转换值类型转换
1.字符串拼接
var b = "110" +10
2.==
100 == "100"
0 == ""
null == undefined
3.if语句0 nan undefined null "" false
4.逻辑运算
!window.abc a=100 !!a true
js中的内置函数 不考虑浏览器的环境单纯作为语言来讲内置了构造函数 Number String Boolean Object Function Array Date RegExp Error内置函数的作用原型链
构造函数的特点 函数名称大写 默认会有return this
重点关注this 对象无限添加属性
![](https://img.haomeiwen.com/i10368875/8978cc7f231f9878.png)
![](https://img.haomeiwen.com/i10368875/0345e3453353a366.png)
原型规则 4条
1.所有的引用类型(数组 对象 函数)都具有对象特性,既可以任由扩展属性除了null
2.所有的引用类型(数组 对象 函数)都有一个proto属性!隐式原型!,属性值是一个普通对象 所有函数都有一个prototype属性,属性值也是个对象
3.所有引用类型(数组对象函数)proto属性值指向他的构造函数“prototype”属性值
4.试图在好一个对象的属性时 如果对象本身没有会去查找自身的proto隐式原型查找
![](https://img.haomeiwen.com/i10368875/84d156372c2cefdb.png)
for(i in obj) 高级浏览器屏蔽了来自原型的属性只拿自身的属性
obj.hasOwnProperty(i)属性是否为自身的
原型链
![](https://img.haomeiwen.com/i10368875/65e243216398a4b9.png)
![](https://img.haomeiwen.com/i10368875/0004bd4a460f88a9.png)
1.准确判断一个变量是数组类型
intanceof判断一个函数是否是一个变量的构造函数
[] intanceof Array
2.写一个原型链继承的例子
就是将构造函数的显示原型重新赋值一个对象
![](https://img.haomeiwen.com/i10368875/02047fa85c2dd6e1.png)
3.描述new一个对象的例子
创建一个新对象 this来引用这个对象然后就可以给this扩展属性 类似于a={} a,age =18
![](https://img.haomeiwen.com/i10368875/79018ee31df01928.png)
练习
![](https://img.haomeiwen.com/i10368875/8935f4a901263ba0.png)
![](https://img.haomeiwen.com/i10368875/eddb752f638b2da4.png)
json:也是js中的内置对象和math一样是内置对象而不是内置函数 JSON.stringfy和JSON.parses 也是一种数据格式
![](https://img.haomeiwen.com/i10368875/1f4460007ecf96ae.png)
1.Math.random在前端清除缓存作用
![](https://img.haomeiwen.com/i10368875/50872283ea6af1ce.png)
![](https://img.haomeiwen.com/i10368875/a080c5ffc8ef0f88.png)
Date.now()和getTime是一样的
this要在执行时才能确认值,定义时无法确认,执行时的意思是一个函数加括号 在这之前函数仍然处于定义状态如果3,4,5行写了错误 在前6行代码是不会报错的因为没有编译
箭头函数的this为定义时的对象也就是一直要找到外层函数看外层函数在什么对象的下面所以闭包中使用箭头函数会将this指向正确的对象 而闭包中的函数是普通函数调用所以指向window 以下的例子可以很好理解这句话
![](https://img.haomeiwen.com/i10368875/30e2a7c1b0b8d194.png)
1.构造函数 this指向new 后面的新对象
![](https://img.haomeiwen.com/i10368875/0165b28d29017427.png)
2.对象属性 下面的第一个例子 this指向调用者
![](https://img.haomeiwen.com/i10368875/b033e635bd5aa247.png)
3.普通函数 window
![](https://img.haomeiwen.com/i10368875/34afe1ca847c78f9.png)
4.call apply bind call和apply的区别在于后面参数传递apply为数组 而bind与这两个的区别在于返回的是函数需要.bind(aaa)()执行
![](https://img.haomeiwen.com/i10368875/e68901346177d7a7.png)
![](https://img.haomeiwen.com/i10368875/958bfd592ed18eff.png)
函数表达式
![](https://img.haomeiwen.com/i10368875/f2a470798118fcc5.png)
![](https://img.haomeiwen.com/i10368875/e5d6c84286697553.png)
前6行代码 console.log(this)并不能确定this的值
作用域:js没有块级作用域 只有函数和全局作用域
作用域链:当前作用域没有的变量即自由变量,于是去父级作用域去找,函数的父级作用域是函数定义时的作用域不是执行的时候
![](https://img.haomeiwen.com/i10368875/e3e423080498ef83.png)
闭包:函数 作为返回值
![](https://img.haomeiwen.com/i10368875/80eff17620e2c7c4.png)
打印100的原因 函数的this是执行时确定所以this的当前环境是window而函数变量是定义时将会去父级查找所以打印100
闭包的使用场景:函数作为参数传递 函数作为返回值
![](https://img.haomeiwen.com/i10368875/4ba53c3ba21cfa06.png)
1.变量提升变量定义/函数声明
![](https://img.haomeiwen.com/i10368875/d338039903632cb6.png)
声明提前 也就是相当于var age; age = 100 函数定义就是函数声明所以就是把整个函数定义提前 而变量是变量定义+变量赋值两个操作 所已 这就是为什么函数表达式不会起作用比如var a = function(){} 提前的将是 var a = undefined
arguments
10个a标签都初始化好之后放进页面中,监听函数的执行时间是不确定的是我们点击的时候当我们点击的时候i已经循环到10 所以结果将是每个标签点击都为10
![](https://img.haomeiwen.com/i10368875/1ae42f617366ff1f.png)
创建 函数作用域 声明了10个函数
![](https://img.haomeiwen.com/i10368875/1a5b95cc6ee436e8.png)
闭包的应用 封装变量收敛权限
判断用户是否是第一次加载对list进行封装 函数外部是拿不到list的 所以你将不会改到list收敛权限
![](https://img.haomeiwen.com/i10368875/82d078d08f7bf069.png)
执行上下文:一段<script>标签或者一个函数将会生成执行上下文
异步单线程:js是单线程所以才有的异步 alert(200)会阻塞程序的执行
对比异步和同步
![](https://img.haomeiwen.com/i10368875/8e2a4202f84a844e.png)
![](https://img.haomeiwen.com/i10368875/e93cb96dbb960300.png)
1.setTimeout 同步和异步的区别是阻塞
2.前端使用异步的场景有哪些,什么时候需要异步
在可能发生等待的时候 1.setTimeout/2.网络请求 ajax<img>加载/3.事件绑定不一样的地方是可以执行多次 等待过程中不能阻塞程序运行
![](https://img.haomeiwen.com/i10368875/994869e9228b2ab9.png)
![](https://img.haomeiwen.com/i10368875/59ad1eaeeb608ba1.png)
单线程 下图没有时间的设置和设置为0是一样的 单线程一次只能做一件事打印100 300 200
![](https://img.haomeiwen.com/i10368875/3301945f82ce38db.png)
其他知识点
![](https://img.haomeiwen.com/i10368875/afa8dc400ec30f85.png)
![](https://img.haomeiwen.com/i10368875/f3d500f8858eae4b.png)
![](https://img.haomeiwen.com/i10368875/a9d1bfd23d80cbe9.png)
jswebapi
回顾js基础 表面看来并不能用于工作中开发代码 也就是浏览器要遵循ecma标准也要遵循W3c标准(dom bom 事件绑定 ajax 存储)w3c没有规定js基础相关的 东西他和ecma是不重合的W3c不管变量类型 原型 作用域和异步 语法和原来是ecma来管 w3c只负责定义浏览器中js操作页面的api和全局变量
![](https://img.haomeiwen.com/i10368875/73cab0d0e40b0ae7.png)
![](https://img.haomeiwen.com/i10368875/4c4463031bab162d.png)
![](https://img.haomeiwen.com/i10368875/a5eefceb399a20a0.png)
dom本质:html和xml规则是一样的浏览器拿到html的页面的时候需要把这个页面渲染并且还要让浏览器能解析js 浏览器最喜欢处理结构化的东西拿到html的代码的时候就会将他结构化dom html代码就是字符串浏览器将他结构化成js/浏览器可识别的东西,如果js不能识别和操作那js将不能及逆行
![](https://img.haomeiwen.com/i10368875/26d948bab3dfbeda.png)
dom操作
1.dom是哪种基本的数据结构 树形的数据结构
2.dom操作常用的api
针对树结构 :新增节点/获取父节点/获取子节点/删除节点/遍历
移动的时候 p2将不会在原来的地方存在
![](https://img.haomeiwen.com/i10368875/a6f596aeed6a84e1.png)
![](https://img.haomeiwen.com/i10368875/a358f231b501c7aa.png)
3.dom的attr和property的区别
![](https://img.haomeiwen.com/i10368875/8968760e49d2af3d.png)
![](https://img.haomeiwen.com/i10368875/4de4e368624d202c.png)
attribute
![](https://img.haomeiwen.com/i10368875/573361fbb599274a.png)
attribute改的是html文档中的标签属性 而property改的是js对象内的属性是两个完全不一样的东西
![](https://img.haomeiwen.com/i10368875/9d92defee463f5a9.png)
nodeType 是数字类型 nodeName 是英文名字的描述
![](https://img.haomeiwen.com/i10368875/3b5e831dd4757bb7.png)
Bom拿到的是关于浏览器一些信息
navigator.userAgent
![](https://img.haomeiwen.com/i10368875/76744e14b3f9424c.png)
screen
![](https://img.haomeiwen.com/i10368875/5c0762edc76dc5de.png)
location
![](https://img.haomeiwen.com/i10368875/568a01987cf0a0bf.png)
history
![](https://img.haomeiwen.com/i10368875/f3b4a1b48b036180.png)
1.检查浏览器类型
navigator.userAgent()
2.解析url的各部分
location.href
location.protocol
location.pathname
location.serach
location.hash
![](https://img.haomeiwen.com/i10368875/6328bc6bf774e89e.png)
事件绑定(通用事件绑定 事件冒泡 代理)
面试的时候尽量不要说自己用框架做过什么东西
1.编写一个通用的事件监听函数
![](https://img.haomeiwen.com/i10368875/4c7d4bdab18be225.png)
ie低版本使用你的是attachEvent因为ie低版本使用的是自己的标准,
2.事件冒泡流程
不用事件冒泡做的 方式 很繁琐
![](https://img.haomeiwen.com/i10368875/d45ffc06c2ad930f.png)
事件冒泡:将时间绑定在div1上 冒泡的机制是p1 div1 body冒泡是事件向上 上图的效果点击激活是先alert激活 然后取消不满足我们的要求 也就是要阻止点击激活的时候的冒泡e.stopPropatation() propagation:蔓延
其余相同的时间将会冒泡到body
冒泡事件的应用 代理
虽然我们在div1上绑定的监听事件 但是target是告诉我们在哪里触发的事件即a标签
target 事件属性可返回事件的目标节点(触发该事件的节点),如生成事件的元素、文档或窗口
![](https://img.haomeiwen.com/i10368875/054184ec269622ad.png)
![](https://img.haomeiwen.com/i10368875/415fab6fabb7ca0a.png)
![](https://img.haomeiwen.com/i10368875/9cede5bbea58dd11.png)
![](https://img.haomeiwen.com/i10368875/2078b48efb8ebdcf.png)
代理的好处:代码简洁 减少浏览器内存占用
3.对于无限下拉加载图片的页面,如何给每个图片绑定事件 代理
事件冒泡流程:
dom属性结构 事件冒泡 阻止冒泡 代理
ajax(XMLHttpRequest 状态吗说明 跨域)
1.手动编写ajax 不依赖第三方库
3-10行为异步 异步的意思是最后执行 send之后xhr开始舰艇状态变化 ie低版本使用得是ActiveXObject
var xhr = new XHMLHttpRequest()
xhr.open('get','/a.txt',false);
xhr.onreadystatechange= function(){
if(xhr.readyState == 4){
![](https://img.haomeiwen.com/i10368875/f06d4f7919c642e5.png)
if(xhr.status == 200){
status是http标准的状态码
![](https://img.haomeiwen.com/i10368875/6463af001efb3b8c.png)
console.log(xhr.responseText)
}
}
}
xhr.send(null)
![](https://img.haomeiwen.com/i10368875/ab826d96f2c24c16.png)
2.跨域的几种实现方式跨域的原理(什么是跨域 jsonp)
浏览器同源策略 不允许ajax访问 其他接口 比如我们自己的网站用ajax来访问别人的接口
![](https://img.haomeiwen.com/i10368875/c8b996967ed1d96b.png)
https 端口是443
防盗链 就是我们从自己的网站使用百度的图片是加载不出来的
a标签
![](https://img.haomeiwen.com/i10368875/b097621b6b68c78b.png)
img标签用于打点统计统计网站可能是其他域
![](https://img.haomeiwen.com/i10368875/312a1462fe5610df.png)
![](https://img.haomeiwen.com/i10368875/61260ecd70eb5622.png)
jsonp实现原理
![](https://img.haomeiwen.com/i10368875/8f10c43d7852150d.png)
服务器动态拼一个js文件返回给我们
2服务端色湖之http header
![](https://img.haomeiwen.com/i10368875/84f8827cf5a10596.png)
存储
1.cookie sessionStorage localStorage的区别
cookie本身用于客户端和服务器端通信的 但是它有本地存储的功能于是被借用
![](https://img.haomeiwen.com/i10368875/f6ed1015aea88256.png)
![](https://img.haomeiwen.com/i10368875/7dcb2cf65d8a0633.png)
sessionStorage在浏览器中存储本地只要浏览器关闭将会清理localStorage不会
![](https://img.haomeiwen.com/i10368875/9d0979ca993e1c47.png)
区别:容量
cookie会被携带到ajax中
api易用性
开发环境
1.页面的加载过程
从输入url到得到html的详细过程:加载资源的形式/加载一个资源的过程/浏览器 渲染页面的过程
形式2种:输入url加载html加载html的静态资源js图片 浏览器根据DNS服务器得到域名的IP地址,向这个ip的机器发送http请求,服务器收到处理http请求 ,浏览器的到返回的内容,浏览器根据html结构生成dom树 再根据css生成cssom 在将dom树和cssom整合形成rendertree,最后浏览器根据rendertree开始渲染 遇到script会阻塞 渲染因为js有权利改变dom的结构
window.onload和DOMContentLoaded的区别
页面加载的几种样例
首先加载页面的11行代码html代码,然后浏览器开始一步步从头分析 分析道link标签时 去加载css然后生成cssom的一个东西浏览器就会知道在div上要添加这样的样式,紧接着在继续解析的时候因为css已经解析完了再去渲染div生成divdom结构的时候就会将样式一次性渲染出来
为何把css放在head中:加载完css浏览器 知道规则在渲染body的时候将会把样式直接渲染出来,如果放在最后的话将会第一次显示默认的情况过了一会按照样式渲染
![](https://img.haomeiwen.com/i10368875/cc452b1f9a26d226.png)
首先输入url加载12代码 然后浏览器开始 解析进入body渲染 到底9行的时候加载script 并执行这一行 js代码阻塞后续代码的渲染,js执行 完之后然后继续向下渲染
js放在最后 是因为不会阻塞html的结构渲染 第二是因为 script可以拿到所有的标签
![](https://img.haomeiwen.com/i10368875/6f055abef2e8ef5a.png)
首先加载html的内容 然后开始渲染加载到图片的 时候异步请求 但是并不会阻碍整个dom的渲染
window.onload必须加载完所有资源
DOMContentLoaded dom渲染完既可以执行不需要等待视频图片
![](https://img.haomeiwen.com/i10368875/3041152763053f2f.png)
2.性能优化
(1)多使用内存缓存 减少cpu计算 减少网络请求 入手(加载页面静态资源的压缩合并,静态资源缓存:本地缓存,使用cdn让资源加载更快因为cdn是不同区域的网络优化,ssr后端渲染,数据直接输出到html中,页面渲染 css放前面 js放后面 懒加载 下拉加载 减少dom操作多个操作合并在一起执行无论是查询还是插入 比如查询的时候存入变量 事件节流 尽早执行操作donmcontentloaded)
优化样例讲解
1.构建工具合并静态资源
2.缓存通过连接名称控制缓存 不更换资源的名字 浏览器会帮我们缓存
![](https://img.haomeiwen.com/i10368875/1cb851e3e7ea2339.png)
4.ssr后端渲染 jsp把数据直接输出到页面中 而不是ajax请求
![](https://img.haomeiwen.com/i10368875/2a64afd9df3efcaa.png)
5.缓存dom查询 第一个是每次循环都将执行dom查询做了10次
![](https://img.haomeiwen.com/i10368875/bf97a0c30e7ecb32.png)
![](https://img.haomeiwen.com/i10368875/a76f894c9c7c8fb2.png)
事件截流
每次键盘输入的时候后触发change操作,但是有的人打字很快我们要做的就是当输入完成之后再去触发change操作
每次都清除上次的setTimeout事件 也就是在短时间内上一次的事件将永远不会触发只会执行最后一次的setTimeout事件
![](https://img.haomeiwen.com/i10368875/a255fc6a36eccde6.png)
![](https://img.haomeiwen.com/i10368875/0d2a832c2eafc200.png)
3.安全性
xss跨站请求攻击 再发布文章的时候的时候插入一段js代码 获取查看者的cookie发送到我自己的服务器 基本原理就是在向网站发布东西的时候提交 js代码 最好的预防就是将<替换成 其他内容
xsrf 和之前css看的不要将请求放在链接中的一个意思增加验证流程
![](https://img.haomeiwen.com/i10368875/9910954c3a12c424.png)
简历 过程
项目经历和解决方案
个人博客
开源项目
es6语法
原型上面的this是指new新的对象
![](https://img.haomeiwen.com/i10368875/0aa51cbc4d40aa80.png)
Object.assign({最后生成的对象},{json对象}) 作用就是合并对象
![](https://img.haomeiwen.com/i10368875/3b908051aeb1bf08.png)
es6中的构造函数在new的时候就会自动立即执行
![](https://img.haomeiwen.com/i10368875/92ba823687644a07.png)
类里面的函数前面没有function showName(){} const a = class{}
![](https://img.haomeiwen.com/i10368875/566b2fb2d8537b7c.png)
类不会预解析 下面的代码会报错
![](https://img.haomeiwen.com/i10368875/e3654f47fbf12c28.png)
关于this
下面的this打印为undefined 矫正的方法为1.在构造函数里this.showName.bind(this)
![](https://img.haomeiwen.com/i10368875/16b3f37066a0d834.png)
严格模式下函数中的this将会是undefined 也就是解释了为什么在类中结构出来的函数中的this将会是undefined
![](https://img.haomeiwen.com/i10368875/9d7ae00d5fb91174.png)
![](https://img.haomeiwen.com/i10368875/3865376b2c2fe6ee.png)
类中的getter和setter set会监听到对象添加属性的操作 get会监听到获取的操作 用途实在底层封装框架
![](https://img.haomeiwen.com/i10368875/23b13311508b28cf.png)
类身上的方法为静态方法
![](https://img.haomeiwen.com/i10368875/a7476492590b46a2.png)
静态方法也会可以给子类调用
原方式函数继承
![](https://img.haomeiwen.com/i10368875/ffea1c1dfc0cef5c.png)
function Person(name){
this.name = name;
}
Person.prototype.showName = function(){ return ${this.nane}
}
function Student(name,skill){
//属性继承
Person.call(this,name);
this.skill = skill;
}
//方法继承
Student.prototype = new Person()
![](https://img.haomeiwen.com/i10368875/5dbabea3c36a58e4.png)
es6继承 extends
子类的构造函数必须执行一个函数就是super() 来执行父类的构造函数 否则会报错因为子类的构造函数将会把父类的构造函数给覆盖了
![](https://img.haomeiwen.com/i10368875/152baa96caa0de61.png)
![](https://img.haomeiwen.com/i10368875/13cb338697e7ea25.png)
添加自己的属性
![](https://img.haomeiwen.com/i10368875/44dec458c7e3a39c.png)
![](https://img.haomeiwen.com/i10368875/321e30c6d0996d9c.png)
继承:拖拽的例子
![](https://img.haomeiwen.com/i10368875/b8c3a40758fb9311.png)
es6中的块级作用域 解决点击li监听函数弹出不一样的数字问题(1.let是利用块级作用域if for后面的花括号就是一个块 2.使用立即执行函数创建函数作用域解决)
let 相当于 var 但是没有预解析 同一个作用域里不能重复定义,在for循环的时候是父级作用域括号里是f花括号是子
const 在定义的时候必须要先赋值
对象引用的关系是不可以再重新引用其他的对象地址 但是对象内部的属性可以
Object.freeze 定一个数组不能去修改const arr = Object.freeze(["aaa","bbb"])
arr.push("a")将会报错 不能扩展
![](https://img.haomeiwen.com/i10368875/d1061d2e61f3d9c6.png)
![](https://img.haomeiwen.com/i10368875/3c7b57457ed2e4fe.png)
下图将会报错
![](https://img.haomeiwen.com/i10368875/62d8e6594615196c.png)
没有块级作用域的时候我们是拿立即执行匿名函数来模拟的
解构赋值
let [a,b,c] = [12,5,6]
左右两边,结构格式要保持一致,数组的解构按从左至右依次赋值
解构json 解构对象的时候需要属性依次对应
let json = {
name="strive",
age:18
}
let {name,age}= json
给默认值 但是对null会无效 null会赋值给c
![](https://img.haomeiwen.com/i10368875/a1a634b62139fc12.png)
![](https://img.haomeiwen.com/i10368875/7c732391112d5081.png)
数组里可以交换位置
![](https://img.haomeiwen.com/i10368875/a5b1c81e96b4bfed.png)
window和top是相同的
解构还用在import {}
函数传参解构
![](https://img.haomeiwen.com/i10368875/1181bac201edec2f.png)
//函数传参给的默认值 如果不给的话将会报错 相当解构左右两边不等 如给了默认值当函数没有传参的时候将取默认值
function show({a = 1,b = 2} = {}){}
show()
4.字符串模板及新增
优点可以随意换行
![](https://img.haomeiwen.com/i10368875/67d1f19d3005f6ca.png)
str.indexOf 返回索引位置 -1
str.includes("a")返回 bool
![](https://img.haomeiwen.com/i10368875/2ebf121524203720.png)
检测 是否是以http开头
str.startsWith("http") str.endsWith
str.repeat(3) 重复3次
str.padStart(整个字符串长度,填充内容) 向前填充
"str".padEnd(6,"xxx")
![](https://img.haomeiwen.com/i10368875/d9b38ea3466532a7.png)
函数变化
函数的默认参数
这个传值了 只有不给的时候才是没传
![](https://img.haomeiwen.com/i10368875/fca0110521b876a4.png)
会报错 函数的参数将是默认定义的 和for循环有差别
![](https://img.haomeiwen.com/i10368875/5788efc56ae43e59.png)
扩展运算符 重置 剩余运算符...
对数组 和对象都管用
![](https://img.haomeiwen.com/i10368875/19aec6f728c6a727.png)
排序
![](https://img.haomeiwen.com/i10368875/a2011fb1c10a8fa4.png)
剩余 剩余的意思是只能放在最后一个
![](https://img.haomeiwen.com/i10368875/4d9c42350a291016.png)
复制数组 let arr = [1.2,3]
let b = [...arr]
![](https://img.haomeiwen.com/i10368875/a11d91834c3bbbc7.png)
函数真正的变化是箭头函数
当没有花括号的时候默认返回
![](https://img.haomeiwen.com/i10368875/6228bc59a22e5e6c.png)
完整的箭头函数
![](https://img.haomeiwen.com/i10368875/2d92f95d75808855.png)
会执行undefined setTimeout由window执行
![](https://img.haomeiwen.com/i10368875/06ee99ade18a62ec.png)
setTimeout 相当于一个数组 把方法存到数组里面 然后过段时间拿出来在全局环境下运行
![](https://img.haomeiwen.com/i10368875/42f58674ce74ab8f.png)
let和const定义的变量不属于window
箭头函数中的this会绑定定义函数时候坐在的对象,不再是运行是所在的对象了
![](https://img.haomeiwen.com/i10368875/aab49e608905e3b2.png)
arguments在箭头函数中不存在 利用扩展运算符
![](https://img.haomeiwen.com/i10368875/5c05ddb1829731d7.png)
箭头函数不能做构造函数使用 不能new
![](https://img.haomeiwen.com/i10368875/69846ecc8c6cc334.png)
数组的热闹循环
数组最常见的就是循环
arr.forEach
arr.forEach(()=>{},123)回调函数是每循环一次调用一次
![](https://img.haomeiwen.com/i10368875/73a5365ef4034bd7.png)
arr.filter
过滤:也是学要return的 过滤不想要的元素如果回调函数返回true该item项将回留下来
![](https://img.haomeiwen.com/i10368875/8fc861df4ed60908.png)
arr.some 数组里有一个选项符合条件就返回true 类似于查找的功能
![](https://img.haomeiwen.com/i10368875/275115755adf3d41.png)
arr.every 数组里面的所有元素都符合条件才会返回true
以上接受的参数一模一样 可以接受两个参数,第一个为回调函数 第二个是this的指向改变,第二个参数的几率很小
![](https://img.haomeiwen.com/i10368875/f6291283870719fc.png)
![](https://img.haomeiwen.com/i10368875/222c365a95081492.png)
箭头函数绑定的this是定义时的this 不会改变 一下的结果打印还是window
![](https://img.haomeiwen.com/i10368875/d3f5ba7c1de340a9.png)
arr.reduce 接受的参数 前一个和当前的item 项 index arr
reduce也是一个循环 求数组和
let arr = [1,2,3.4]
arr.reduce((pre,cur,index,arr)=>{ return pre+cur})
![](https://img.haomeiwen.com/i10368875/815b4a63c87f833b.png)
arr.redeceRight 打印81
![](https://img.haomeiwen.com/i10368875/f7a06c839e4157cf.png)
for of 循环 最后一个[key,value]为数组解构
![](https://img.haomeiwen.com/i10368875/5b368ef247b14339.png)
![](https://img.haomeiwen.com/i10368875/7812aabaea22aa73.png)
数组另增新的其他的东西
将类数组转为数组的方式,并且下标是0,1,2
类数组典型的特点是有length
![](https://img.haomeiwen.com/i10368875/522ad62b178b3c94.png)
![](https://img.haomeiwen.com/i10368875/ab6a03c50a607d4c.png)
![](https://img.haomeiwen.com/i10368875/1298117d866938eb.png)
2.Array.of()将一组值转为数组 类似于三点
![](https://img.haomeiwen.com/i10368875/460988ede30fa41f.png)
3.数组本身的方法
find找出第一个复合条件的数组成员,如果没找到返回undefined
let arr = [1,2,3]
arr.find((val.index,arr)=>{ return val>2}) //打印3
findIndex没找到返回-1返回第一个符合条件的位置
3fill (填充的东西 开始位置 结束为止)
![](https://img.haomeiwen.com/i10368875/107ac2dadafc1e2e.png)
4.arr,IndexOf
arr.incluldes("a") 返回truefalse
findIndex传得是函数 indexOf是value项 返回的将是第一个符合条件的索引项位置
对象
对象的简介语法
![](https://img.haomeiwen.com/i10368875/53ce29779888fcfd.png)
对象里面不能用箭头函数
![](https://img.haomeiwen.com/i10368875/cf5fe9beb1cbec2a.png)
Obejct.is比较两个值相等 NaN
Object.is(NaN,NaN)
![](https://img.haomeiwen.com/i10368875/68c2c21ff06fdd1d.png)
![](https://img.haomeiwen.com/i10368875/93da64931315ab1f.png)
Object.assign 用来合并 复制对象
let json = {a:1}
let json 2 = {b:1,//a:2} 后面会覆盖前面的
let json3 = {c:3}
Object.assign({},json1,json2,json3)
![](https://img.haomeiwen.com/i10368875/9132ad24f1bed7a9.png)
![](https://img.haomeiwen.com/i10368875/64f725557887f324.png)
Promise 主要的作用是解决异步回调问题,传统方式大部分用回调函数,事件
就是回调嵌套回调在嵌套回调
![](https://img.haomeiwen.com/i10368875/f611ff06322e4535.png)
![](https://img.haomeiwen.com/i10368875/da7e887cada71961.png)
![](https://img.haomeiwen.com/i10368875/3cdbcfd043c861a5.png)
![](https://img.haomeiwen.com/i10368875/02530ee0a1f5525f.png)
![](https://img.haomeiwen.com/i10368875/6a04bc75964a2b02.png)
![](https://img.haomeiwen.com/i10368875/fbe76bcd5f57965e.png)
Promise.all必须确保所有的promise都是resolve的状态即都是成功的状态
![](https://img.haomeiwen.com/i10368875/508a8054326ed4cc.png)
symbol和generator
let syml = Symbol("aa") 不能new 他不是构造函数
![](https://img.haomeiwen.com/i10368875/a95e7d0b6adac041.png)
json 使用for in循环 不能输出symbol的属性
![](https://img.haomeiwen.com/i10368875/83d4669a33213509.png)
![](https://img.haomeiwen.com/i10368875/84b925cb386df60b.png)
generator 也是个函数 目的是解决异步问题 深度嵌套问题同promise 现在使用async
区分generator函数和普通函数*
function * show(){
yield //产出
}
genrator是分步骤完成的 done是说该函数是否完成,遇到return 完成
![](https://img.haomeiwen.com/i10368875/3458792313781994.png)
![](https://img.haomeiwen.com/i10368875/666b6caf04fa10fb.png)
![](https://img.haomeiwen.com/i10368875/f32ef34c97a4616a.png)
以上是手动调用 for..of自动遍历generator 但是不会遍历return的东西只会遍历yield的东西 还可以配合解构赋值
![](https://img.haomeiwen.com/i10368875/0921b5864b05c978.png)
![](https://img.haomeiwen.com/i10368875/4618efa155254f5c.png)
b是 剩余参数组成的数组
![](https://img.haomeiwen.com/i10368875/330a4faaa854dc1f.png)
![](https://img.haomeiwen.com/i10368875/e4fcf77664d0b69b.png)
![](https://img.haomeiwen.com/i10368875/e223c53ab5f297e6.png)
解决异步 第一步的结果传到第二步请求中
![](https://img.haomeiwen.com/i10368875/3c01b5599f966e0a.png)
![](https://img.haomeiwen.com/i10368875/5a5a46df1516ecc7.png)
async
![](https://img.haomeiwen.com/i10368875/7e806d191bc2bf11.png)
promise解决多层回调
![](https://img.haomeiwen.com/i10368875/a768aeb40ef91c5f.png)
generator解决多层回调
![](https://img.haomeiwen.com/i10368875/a53a93556c382e70.png)
![](https://img.haomeiwen.com/i10368875/fc0459321f7af3dc.png)
![](https://img.haomeiwen.com/i10368875/7cb0d32e63f97afc.png)
![](https://img.haomeiwen.com/i10368875/e5b09d3a635ea742.png)
任何与网络请求有关的东西都try catch
以下三个网络请求没有必然关系的时候可以这么用 因为不能保证哪个网络请求先执行完后执行完
![](https://img.haomeiwen.com/i10368875/8235a9747701a10b.png)