共计 8548 个字符,预计需要花费 22 分钟才能阅读完成。
编程环境
Prettier
Prettier 是一个“有主见”的前端代码格式化工具。
在 VSCode 中安装好 Prettier 插件后,可能需要自定义一些配置,可在项目根目录下添加配置文件.prettierrc
:
{
"singleQuote": true,
"arrowParens": "avoid"
}
配置代码片段
在 VSCode 中,「文件」->「首选项」->「配置用户代码片段」->「新建全局代码片段文件 …」,输入自定义文件名,回车。在文件中写入:
{
"Print to console": {
"scope": "javascript,typescript",
"prefix": "cl",
"body": ["console.log($1);"],
"description": "Log output to console"
}
}
另外,需要对 VSCode 做个设置,在设置中搜索 snippet,把「控制活动代码段是否阻止快速建议」前面的√取消掉。
Node.js
到 Node.js 官网:https://nodejs.org/en/ 进行下载安装。
VSCode 默认终端
VSCode 默认是 PowerShell 终端,修改为 CMD 的操作步骤如下:
「终端」->「新建终端],在终端界面,点击 + 右边的「∨」->「选择默认配置文件」->「Command Prompt」,重启 VSCode。
安装 live-server
npm install live-server -g
在静态文件目录下执行:
live-server
live-server 运行在 8080 端口下,可以通过 127.0.0.1:8080
进行访问。
VSCode 生成 html 模板
在设置中搜索「Trigger Expansion On Tab」,前面打上√。然后就能用 ! + tab
自动生成 html 模板了。
JavaScript 组成
ECMAScript 是一种语言标准,规定了 JavaScript 基础语法核心知识,而 JavaScript 是网景公司对 ECMAScript 标准的一种实现。
Web APIs:
- DOM 操作文档,比如对页面元素进行移动、大小、添加制除等操作。
- BOM 操作浏览器,比如页面弹窗,检测窗口宽度、存储数据到浏览器等等。
JavaScript 权威网站:https://developer.mozilla.org/zh-CN/docs/Web/JavaScript
输入输出
输入语法
prompt(" 请输入姓名:")
输出语法
// 向 body 输出内容
document.write(" 输出的内容 ")
// 页面弹出警告对话框
alert(" 输出的内容 ")
// 控制台输出语法,程序员调试使用
console.log(" 控制台打印 ")
代码执行顺序
- 按 HTML 文档流顺序执行 JavaScript 代码
- alert()和 prompt()会跳过页面染先被执行
变量
变量声明
let age // 变量声明
console.log(age) // 此时 age 的值为 undefined
age = 18 // 变量赋值
console.log(age) // 此时 age 的值为 18
console.log(typeof age) // number 作为运算符(推荐)console.log(typeof age) // number 函数形式
let name, city // 一次性声明多个变量
let sex = 1 // 变量初始化
使用 let 声明的变量具有块作用域,在代码块 {} 中声明的变量无法在代码块的外部访问。
未使用 let 或 const 关键字声明的变量会在 global 范围内自动创建。
数据类型
有 6 种:
- number 数值(BigInt)
- string 字符串
- boolean 布尔值
- null 空值
- undifined 未定义
- symbol 符号
- object 对象
前 5 个属于基本数据类型,object 属于引用数据类型。基本数据类型的值直接在 栈内存 中存储;而对象是保存到 堆内存 中的,每创建一个新对象,就会在堆内存中开辟出一个新空间,而变量保存的是对象内存地址。
JavaScript 不区分整数和浮点数,统一用 number 表示。
console.log(10.0 === 10) // true
console.log(Math.abs(1 / 3 - (1 - 2 / 3)) < 0.0001) // true 浮点数比较
// NaN(Not a Number)这个特殊 Number 与所有其他值都不相等,包括它自己。// NaN 是粘性的,任何对 NaN 的操作都会返回 NaN
console.log(NaN === NaN) // false
console.log(NaN + 1) // NaN
console.log(isNaN(NaN)) // true 唯一能判断 NaN 的方法
console.log(9007199254740993) // 9007199254740992
// 比 2^53 大的整数,可使用内置 BigInt 类型,表示方法是在整数后加一个 n
var a = 9007199254740993n
var b = BigInt(9007199254740993) // 使用 BigInt() 把 Number 转成 BigInt
var c = BigInt('0x20000000000001')// 把字符串转成 BigInt
console.log(a === b, a === c) // false true
null 和 undefined:
- null 表示一个“空”值
- undefined 表示值“未定义”
- 大多数情况下,都应该用 null
- undefined 仅在判断函数参数是否传递的情况下有用
console.log(null + 1) // 1
let money
console.log(money + 100) // NaN 对 undefined 进行运算操作,结果是 NaN
console.log(money + '100') // undefined100 用 undefined 连接字符串,将得到一个含 undefined 的字符串
模板字符串:
let age = 18
// 外面用反引号,里面用 ${变量名}
console.log(` 我 ${age}岁了 `) // 我 18 岁了
字符串是不可变的,对字符串某个索引赋值,不会有任何错误,但也没有任何效果:
let s = 'abc'
s[1] = 'd'
console.log(s) // abc
JavaScript 有两种比较运算符:
- == 会自动转换数据类型再比较
- === 不会自动转换数据类型,如果数据类型不一致,返回 false,如果一致,再比较
不要使用 == 比较,始终坚持使用 === 比较。
strict 模式
在 strict 模式下运行的 JavaScript 代码,强制申明变量,未申明变量就使用,将导致运行错误。
'use strict'
age = 18 // 报错 age is not define
常量
用 const 关键字声明的变量是只读的。一旦一个变量被赋值为 const,它就不能被重新赋值。
const G = 9.8
console.log(G)
但是,使用 const 分配给变量的对象(包括数组和函数)仍然是可变的。使用 const 声明只能防止变量标识符的重新分配。为了确保数据不被改变,JavaScript 提供了一个函数 Object.freeze。
const MATH_CONSTANTS = {PI: 3.14,}
Object.freeze(MATH_CONSTANTS)
类型转换
隐式转换
规则:
- + 号两边只要有一个是字符串,都会把另外一个转成字符串
- 除了 + 以外的算术运算符,比如 – * / 等都会把数据转成数字类型
小技巧:
- + 号作为正号解析可以转换成数字型
-
任何数据和字符串相加结果都是字符串
console.log(+"111") // 转换为数字型
显式转换
转换为 string
null 和 undifined 没有 toString()方法,使用 String()方法会将 null 直接转换为 ”null”,将 undifined 直接转换为 ”undifined”。
转换为 number
Number()
- 如果是纯数字的字符串,则直接将其转换为数字
- 如果字符串中有非数字的内容,则转换为 NaN
- 如果字符串是一个空串或全是空格,则转换为 0
- true 转成 1,false 转成 0
- null 转成 0
- undifined 转成 NaN
parseInt()
将一个字符串中的有效整数内容取出来,然后转换为 number。
parseFloat()
作用和 parseInt() 类似,不同的是它可以获得有效的小数。
转换为 boolean
- 除了 0 和 NaN,其余都是 true
- 除了空串,其余都是 true
- null 和 undefined 都会转换为 false
- 对象转换为 true
运算符
赋值
??= 为空赋值,只有当变量的值为 null 或 undefined 时才会对变量进行赋值:
let a = 100
a ??= 200
console.log(a) //100
a = null
a ??= 200
console.log(a) //200
?. 是链判断运算符,允许开发人员读取深度嵌套在对象链中的属性值,而不必验证每个引用。当引用为空时,表达式停止计算并返回 undefined。
let s1 = {
name: '张三',
address: {city: '北京',},
}
let s2 = {
name: '李四',
address: null,
}
function test(stu) {console.log(stu.address?.city)
}
test(s1) // 北京
test(s2) //undefined
逻辑中断
短路:只存在于 && 和 || 中,当满足一定条件会让右边代码不执行。
- &&:左边为 false 就短路
- ||:左边为 true 就短路
原因:通过左边能得到整个式子的结果,因此没必要再判断右边。
运算结果:无论 && 还是 ||,运算结果都是最后被执行的表达式值,一般用在变量赋值。
console.log(11 && 22) //22
console.log(11 || 22) //11
展开运算符
展开运算符(…)能将一个数组进行展开,典型应用场景:求数组最大值、合并数组等。
const arr = [1, 3, 4]
console.log(Math.max(...arr)) //4
const arr1 = [1, 3, 4]
const arr2 = [2, 8, 9]
const arr = [...arr1, ...arr2]
console.log(arr) //[1, 3, 4, 2, 8, 9]
let obj = {name: '张三', age: 18}
let obj2 = {...obj} // 复制对象
console.log(obj2)
Unicode 编码
在字符串中使用转义字符输入 Unicode 编码,u 四位编码
console.log("u2620")
在网页中使用 Unicode 编码,&# 编码;(编码是 10 进制的)
<h1>☠</h1>
数组
let arr = [1, 2, 3, 4] // 声明数组
console.log(arr[0]) // 1
console.log(arr.length) // 数组长度
arr.length = 6 // 改变数组长度会导致 Array 大小变化
console.log(arr) // [1, 2, 3, 4, <2 empty items>]
console.log(arr.slice(1, 3)) // [2, 3] 切片操作
console.log(arr.slice(1)) // [2, 3, 4, <2 empty items>]
新增
let arr = [1, 2, 3]
arr.push("4")
console.log(arr) //[1, 2, 3, "4"]
arr.unshift(5)
console.log(arr) //[5, 1, 2, 3, "4"]
删除
let arr = [1, 2, 3, 4]
arr.pop()
console.log(arr) //[1, 2, 3]
arr.shift()
console.log(arr) //[2, 3]
let arr2 = [1, 2, 3, 4, 5, 6]
arr2.splice(1, 2) // 只删除,不添加
console.log(arr2) // [1, 4, 5, 6]
arr2.splice(1, 2, '7', '8', '9') // 删除后添加
console.log(arr2) // [1, '7', '8', '9', 6]
arr2.splice(1, 0, 10) // 只添加,不删除
console.log(arr2) // [1, 10, '7', '8', '9', 6]
去重
const arr = [1, 2, 1, 66, 2, 1, 9, 5]
for (let i = 0; i < arr.length; i++) {const index = arr.indexOf(arr[i], i + 1)
if (index !== -1) {arr.splice(index, 1)
i--
}
}
console.log(arr)
const newArr = []
for (let i of arr) {if (newArr.indexOf(i) === -1) {newArr.push(i)
}
}
console.log(newArr)
排序
let arr = [2, 4, 3, 5, 1]
arr.sort() // 默认升序
console.log(arr) //[1, 2, 3, 4, 5]
arr.sort((a, b) => b - a) // 降序
console.log(arr) //[5, 4, 3, 2, 1]
let arr2 = [
{
name: " 张三 ",
age: 18,
},
{
name: " 李四 ",
age: 20,
},
{
name: " 王五 ",
age: 19,
},
]
arr2.sort((a, b) => b.age - a.age) // 按 age 降序
console.log(arr2)
合并
let arr = [1, 2]
let arr2 = [3, 4]
let arr3 = arr.concat(arr2, 5, 6)
console.log(arr3) // [1, 2, 3, 4, 5, 6]
console.log(arr3.join('-')) // 1-2-3-4-5-6
遍历
let arr = [1, 2, 7, 8]
let r1 = arr.every((item) => item > 5)
console.log(r1) //false
let r2 = arr.some((item) => item > 5)
console.log(r2) //true
let r3 = arr.filter((item) => item > 5)
console.log(r3) //[7, 8]
let r4 = arr.map((item) => "age=" + item)
console.log(r4) //['age=1', 'age=2', 'age=7', 'age=8']
let r5 = arr.reduce((item1, item2) => item1 + item2) // 累加
console.log(r5) //18
arr.forEach((item, index) => {console.log(index, item)
})
注意:在 forEach 里面 return 不会终止迭代。
迭代器
let arr = [1, 2, 7, 8]
for (let i of arr) {console.log(i) //1 2 7 8
}
let arr = [1, 2, 7, 8]
for (let i of arr.entries()) {console.log(i) //entries()是键值对,keys()是键,values()是值}
伪数组转数组
function f() {a = Array.from(arguments)
console.log(a.map((item) => "age=" + item))
}
f(18, 19, 20) //['age=18', 'age=19', 'age=20']
搜索
let arr = [1, 13, 2, 44]
console.log(arr.indexOf(13)) //1 找不到返回 -1
console.log(arr.includes(13)) //true
console.log(arr.find((item) => item > 10)) //13
console.log(arr.findLast((item) => item > 10)) //44
console.log(arr.findIndex((item) => item > 10)) //1
console.log(arr.findLastIndex((item) => item > 10)) //3
函数
命名建议
常用动词约定 | 含义 |
---|---|
can | 判断是否可执行某个动作 |
has | 判断是否含有某个值 |
is | 判断是否为某个值 |
get | 获取某个值 |
set | 设置某个值 |
load | 加载某些数据 |
作用域
变量有一个坑,特殊情况:如果函数内部,变量没有声明,直接赋值,会当全局变量看,但是强烈不推荐。
function fn() {num = 10}
fn()
console.log(num) //10
原型对象
创建的每一个函数,解析器都会向函数中添加一个属性 prototype,这个属性对应着一个对象,这个对象就是所谓的 原型对象。
当函数以构造函数调用时,它所创建的对象中都会有一个隐含的属性,指向该构造函数的原型对象,可以通过__proto__来访问该属性。
当访问对象的一个属性或方法时,会先在对象自身中寻找,有则直接使用,没有则会去原型对象中寻找(找到直接使用)。
原型的原型
原型对象也是对象,它也有原型。
function Person() {}
// 向原型中添加 name 属性
Person.prototype.name = " 李小龙 "
var p = new Person()
console.log(p.hasOwnProperty("name")) // false
console.log(p.__proto__.__proto__.hasOwnProperty("hasOwnProperty")) // true
toString()
当打印一个对象时,实际上是输出对象 toString()方法的返回值,可为对象添加一个 toString()自来定义输出。
function Person() {}
Person.prototype.toString = function () {return " 我是李小龙 "}
var p = new Person()
console.log(p.toString())
call()和 apply()
这两个方法都是函数对象的方法,需要通过函数对象来调用。
在调用 call()和 apply()时可将一个对象指定为第一个参数,此时这个对象将会成为函数执行时的 this。
两者区别:
- call()将实参在对象之后依次传递
- apply()需要将实参封装到一个数组中统一传递
function func(a, b) {alert(a + b)
}
var obj = {name: " 李小龙 "}
func.call(obj, 1, 2)
func.apply(obj, [3, 4])
bind()
用来创建一个新的函数,bind 可以为新函数绑定 this,也可以为新函数绑定参数。
function fn(a, b, c) {console.log(a, b, c, this)
}
const obj = {age: 18}
const newFn = fn.bind(obj, 1, 2, 3)
newFn()
arguments 对象
在调用函数时,浏览器会传递进两个隐含的参数:
- 函数的上下文对象 this
-
封装实参的对象 arguments
- arguments 是个类数组对象(不是数组),可通过索引来操作数据,也可以获取长度
- 它里边有个属性叫 callee,这个属性对应一个函数对象,就是当前正在执行的函数对象
function func() {console.log(arguments[0], arguments.length, arguments.callee)
}
func("obj", 1, 2)
对象
创建对象
// 使用对象字面量创建对象
let obj = {
name: " 李小龙 ",
age: 18,
}
console.log(obj) // {name: '李小龙', age: 18}
// 使用 new 关键字创建对象
const obj = new Object()
const obj2 = new Object({name: " 李小龙 "})
// 使用构造函数创建对象
function Pig(name, age, sex) {
this.name = name
this.age = age
this.sex = sex
}
const Peppa = new Pig(" 佩奇 ", 3, " 女 ")
const George = new Pig(" 乔治 ", 4, " 男 ")
使用 new 关键字调用的函数,是构造函数(专门用来创建对象的函数)。
修改对象
在使用变量存储对象时,很容易因为改变变量指向的对象,提高代码的复杂度。所以通常情况下,声明存储对象的变量时会使用 const。
遍历对象
let mySymbol = Symbol()
let obj = {
name: '李小龙',
age: 18,
[mySymbol]: '我是特殊属性',
}
for (let k in obj) {console.log(k, '=', obj[k])
}
符号添加的属性不能枚举。
冻结对象
const obj = {
name: " 佩奇 ",
age: 10,
hobby: {one: " 唱歌 ",},
}
Object.freeze(obj) // 浅冻结,使第一层数据不可修改
// 深冻结数据
function deepFreeze(obj) {Object.freeze(obj)
for (let i in obj) {if (obj.hasOwnProperty(i)) {if (typeof obj[i] === "object") {deepFreeze(obj[i])
}
}
}
}