JavaScript基础语法
HTML:标记语言JavaScript:编程语言 JavaScript的组成
- ECMASCRIPT: 定义了javascript的语法规范,描述了语言的基本语法和数据类型
- BOM (Browser Object Model): 浏览器对象模型
- 有一套成熟的可以操作浏览器的 API,通过 BOM 可以操作浏览器。比如: 弹出框、浏览器跳转、获取分辨率等
- DOM (Document Object Model): 文档对象模型
- 有一套成熟的可以操作页面元素的 API,通过 DOM 可以操作页面中的元素。比如: 增加个 div,减少个 div,给 div 换个位置等
总结: JS 就是通过固定的语法去操作 浏览器 和 标签结构 来实现网页上的各种效果
JS 中的注释
单行注释
- 可以直接写两个
/,也可以按ctrl + /
// 我是一个单行注释
多行注释
-
可以直接写
/**/然后在两个星号中间写注释- 各个编辑器的快捷键不一样,
vscode是ctrl + shift + a
- 各个编辑器的快捷键不一样,
/*
注释的代码不会执行
alert('我是一个弹出层')
alert('我是一个弹出层')
*/
变量(重点)
规则: 必须遵守的,不遵守就是错
- 一个变量名称可以由 数字、字母、英文下划线(_) 、美元符号($) 组成
- 严格区分大小写
- 不能由数字开头
- 不能是 保留字 或者 关键字
- 不要出现空格
规范: 建议遵守的(开发者默认),不遵守不会报错
- 变量名尽量有意义(语义化)
- 遵循驼峰命名规则,由多个单词组成的时候,从第二个单词开始首字母大写
- 不要使用中文
数据类型(重点)
- 是指我们存储在内存中的数据的类型
- 我们通常分为两大类 基本数据类型 和 复杂数据类型
基本数据类型
- 数值类型(number)
- 一切数字都是数值类型(包括二进制,十进制,十六进制等)
NaN(not a number),一个非数字
- 字符串类型(string)
- 被引号包裹的所有内容(可以是单引号也可以是双引号)
- 布尔类型(boolean)
- 只有两个(
true或者false)
- null类型(null)
- 只有一个,就是
null,表示空的意思
- undefined类型(undefined)
- 只有一个,就是
undefined,表示没有值的意思
复杂数据类型
- 对象类型(object)
- 函数类型(function)
判断数据类型
- 既然已经把数据分开了类型,那么我们就要知道我们存储的数据是一个什么类型的数据
- 使用
typeof关键字来进行判断
// 第一种使用方式
var n1 = 100; console.log(typeof n1);
// 第二种使用方式var s1 = 'abcdefg';
console.log(typeof(s1));
判断一个变量是不是数字
- 可以使用
isNaN这个方法来判断一个变量是不是数字 isNaN:is not a number
// 如果变量是一个数字var n1 = 100;console.log(isNaN(n1)); //=> false
// 如果变量不是一个数字
var s1 = 'Jack'
console.log(isNaN(s1)); //=> true
数据类型转换
- 数据类型之间的转换,比如数字转成字符串,字符串转成布尔,布尔转成数字等
其他数据类型转成数值
- Number(变量)
可以把一个变量强制转换成数值类型
可以转换小数,会保留小数
可以转换布尔值
遇到不可转换的都会返回
NaN
- parseInt(变量)
从第一位开始检查,是数字就转换,知道一个不是数字的内容
开头就不是数字,那么直接返回
NaN不认识小数点,只能保留整数
- parseFloat(变量)
从第一位开始检查,是数字就转换,知道一个不是数字的内容
开头就不是数字,那么直接返回
NaN认识一次小数点
- 除了加法以外的数学运算
运算符两边都是可运算数字才行
如果运算符任何一遍不是一个可运算数字,那么就会返回
NaN加法不可以用
其他数据类型转成字符串
- 变量.toString()
有一些数据类型不能使用
toString()方法,比如undefined和null
- String(变量)
所有数据类型都可以
- 使用加法运算
在 JS 里面,
+由两个含义字符串拼接: 只要
+任意一边是字符串,就会进行字符串拼接加法运算:只有
+两边都是数字的时候,才会进行数学运算
其他数据类型转成布尔
- Boolean(变量)
在 js 中,只有
''、0、null、undefined、NaN,这些是false其余都是
true
运算符
- 就是在代码里面进行运算的时候使用的符号,不光只是数学运算,我们在
js里面还有很多的运算方式
数学运算符
- +
只有符号两边都是数字的时候才会进行加法运算
只要符号任意一边是字符串类型,就会进行字符串拼接
- —
会执行减法运算
会自动把两边都转换成数字进行运算
3 *
会执行乘法运算
会自动把两边都转换成数字进行运算
- /
会执行除法运算
会自动把两边都转换成数字进行运算
- %
会执行取余运算
会自动把两边都转换成数字进行运算
赋值运算符
- =
就是把
=右边的赋值给等号左边的变量名
var num = 100就是把 100 赋值给 num 变量
那么 num 变量的值就是 100
- +=
var a = 10; a += 10; console.log(a); //=> 20
a += 10等价于a = a + 10
- -=
var a = 10;
a -= 10; console.log(a); //=> 0
a -= 10等价于a = a - 10
- *=
var a = 10; a *= 10; console.log(a); //=> 100
a *= 10等价于a = a * 10
- /+
var a = 10; a /= 10; console.log(a); //=> 1
a /= 10等价于a = a / 10
- %=
var a = 10; a %= 10; console.log(a); //=> 0
a %= 10等价于a = a % 10
比较运算符
-
==
- 比较符号两边的值是否相等,不管数据类型
1 == '1'两个的值是一样的,所以得到
true
-
===
- 比较符号两边的值和数据类型是否都相等
1 === '1'两个值虽然一样,但是因为数据类型不一样,所以得到
false
- !=
- 比较符号两边的值是否不等
1 != '1'因为两边的值是相等的,所以比较他们不等的时候得到
false
- !==
- 比较符号两边的数据类型和值是否不等
1 !== '1'因为两边的数据类型确实不一样,所以得到
true
-
=
- 比较左边的值是否 大于或等于 右边的值
1 >= 1结果是true
1 >= 0结果是true
1 >= 2结果是false
- <=
- 比较左边的值是否 小于或等于 右边的值
1 <= 2结果是true
1 <= 1结果是true
1 <= 0结果是false
-
- 比较左边的值是否 大于 右边的值
1 > 0结果是true
1 > 1结果是false
1 > 2结果是false
- <
- 比较左边的值是否 小于 右边的值
1 < 2结果是true
1 < 1结果是false
1 < 0结果是false
逻辑运算符
-
&&
- 进行 且 的运算
符号左边必须为 true 并且右边也是 true,才会返回 true
只要有一边不是 true,那么就会返回 false
true && true 结果是 true
true && false 结果是 false
false && true 结果是 false
false && false 结果是 fals
- ||
- 进行 或 的运算
符号的左边为 true 或者右边为 true,都会返回 true
只有两边都是 false 的时候才会返回 false
true || true 结果是 true
true || false 结果是 true
false || true 结果是 true
false || false 结果是 false
- !
- 进行 取反 运算
本身是
true的,会变成false本身是
false的,会变成true
!true结果是false
!false结果是true
自增自减运算符
1. ++
1- 进行自增运算
2- 分成两种,前置++ 和 后置++
3- 前置++,会先把值自动 +1,在返回
var a = 10;
console.log(++a);
// 会返回 11,并且把 a 的值变成 11
4 - 后置++,会先把值返回,在自动+1
var a = 10;
console.log(a++);
// 会返回 10,然后把 a 的值变成 11
2. --
1- 进行自减运算
2- 分成两种,前置-- 和 后置--
3- 和 ++ 运算符道理一样
分支结构
- 我们的
js代码都是顺序执行的(从上到下) - 逻辑分支就是根据我们设定好的条件来决定要不要执行某些代码
IF 条件分支结构(重点)
if 语句
- 通过一个
if语句来决定代码是否执行 - 语法:
if (条件) { 要执行的代码 } - 通过
()里面的条件是否成立来决定{}里面的代码是否执行
// 条件为 true 的时候执行 {} 里面的代码 if (true) {
alert('因为条件是 true,我会执行') }
// 条件为 false 的时候不执行 {} 里面的代码
if (false) {
alert('因为条件是 false,我不会执行') }
if else 语句
- 通过
if条件来决定,执行哪一个{}里面的代码 - 语法:
if (条件) { 条件为 true 的时候执行 } else { 条件为 false 的时候执行 } - 两个
{}内的代码一定有一个会执行
// 条件为 true 的时候,会执行 if 后面的 {}
if (true) {
alert('因为条件是 true,我会执行')
} else {
alert('因为条件是 true,我不会执行')
}
// 条件为 false 的时候,会执行 else 后面的 {}
if (false) {
alert('因为条件为 false,我不会执行') } else {
alert('因为条件为 false,我会执行') }
if else if ... 语句
-
可以通过
if和else if来设置多个条件进行判断 -
语法:
if (条件1) { 条件1为 true 的时候执行 } else if (条件2) { 条件2为 true 的时候执行 } -
会从头开始依次判断条件
-
如果第一个条件为 `true` 了,那么就会执行后面的 `{}` 里面的内容 -
如果第一个条件为 `false`,那么就会判断第二个条件,依次类推
-
多个 {} ,只会有一个被执行,一旦有一个条件为 true 了,后面的就不在判断了
// 第一个条件为 true,第二个条件为 false,最终会打印 “我是代码段1”
if (true) {
alert('我是代码段1')
} else if (false) {
alert('我是代码段2')
}
// 第一个条件为 true,第二个条件为 true,最终会打印 “我是代码段1”
// 因为只要前面有一个条件满足了,就不会继续判断了
if (true) {
alert('我是代码段1')
} else if (true) {
alert('我是代码段2')
}
// 第一个条件为 false,第二个条件为 true,最终会打印 “我是代码段2”
// 只有前一个条件为 false 的时候才会继续向后判断
if (false) {
alert('我是代码段1')
} else if (true) {
alert('我是代码段2')
}
// 第一个条件为 false,第二个条件为 false,最终什么也不会发生
// 因为当所有条件都为 false 的时候,两个 {} 里面的代码都不会执行
if (false) {
alert('我是代码段1')
} else if (false) {
alert('我是代码段2')
}
if else if … else 语句
和之前的 if else if ... 基本一致,只不过是在所有条件都不满足的时候,执行最后 else 后面的 {}
// 第一个条件为 false,第二个条件为 false,最终会打印 “我是代码段3”
// 只有前面所有的条件都不满足的时候会执行 else 后面的 {} 里面的代码
// 只要前面有一个条件满足了,那么后面的就都不会执行了
if (false) {
alert('我是代码段1')
} else if (false) {
alert('我是代码段2')
} else {
alert('我是代码段3')
}
SWITCH 条件分支结构(重点)
- 也是条件判断语句的一种
- 是对于某一个变量的判断
- 语法:
switch (要判断的变量) {
case 情况1:
情况1要执行的代码
break
case 情况2:
情况2要执行的代码
break
case 情况3:
情况3要执行的代码
break
default:
上述情况都不满足的时候执行的代码
}
要判断某一个变量 等于 某一个值得时候使用
三元运算
- 三元运算,就是用 两个符号 组成一个语句
- 三元运算只是对 if else 语句的一个简写形式
- 语法:
条件 ? 条件为 true 的时候执行 : 条件为 false 的时候执行
var age = 18;
age >= 18 ? alert('已经成年') : alert('没有成年')
循环结构
- 循环结构,就是根据某些给出的条件,重复的执行同一段代码
- 循环必须要有某些固定的内容组成
- 初始化
- 条件判断
- 要执行的代码
- 自身改变
WHILE 循环
while,中文叫 当…时,其实就是当条件满足时就执行代码,一旦不满足了就不执行了- 语法
while (条件) { 满足条件就执行 } - 因为满足条件就执行,所以我们写的时候一定要注意,就是设定一个边界值,不然就一直循环下去了
// 1. 初始化条件
var num = 0;
// 2. 条件判断
while (num < 10) {
// 3. 要执行的代码
console.log('当前的 num 的值是 ' + num)
// 4. 自身改变
num = num + 1
}
- 如果没有自身改变,那么就会一直循环不停了
DO WHILE 循环
- 是一个和
while循环类似的循环 while会先进行条件判断,满足就执行,不满足直接就不执行了- 但是
do while循环是,先不管条件,先执行一回,然后在开始进行条件判断 - 语法:
do { 要执行的代码 } while (条件)
// 下面这个代码,条件一开始就不满足,但是依旧会执行一次 do 后面 {} 内部的代码
var num = 10
do {
console.log('我执行了一次')
num = num + 1
} while (num < 10)
FOR 循环
- 和
while和do while循环都不太一样的一种循环结构 - 道理是和其他两种一样的,都是循环执行代码的
- 语法:
for (var i = 0; i < 10; i++) { 要执行的代码 }
// 把初始化,条件判断,自身改变,写在了一起
for (var i = 1; i <= 10; i++) {
// 这里写的是要执行的代码
console.log(i)
}
// 控制台会依次输出 1 ~ 10
- 这个只是看起来不太舒服,但是用起来比较好用
BREAK 终止循环
- 在循环没有进行完毕的时候,因为我设置的条件满足,提前终止循环
- 比如:我要吃五个包子,吃到三个的时候,不能在吃了,我就停止吃包子这个事情
- 要终止循环,就可以直接使用
break关键字
> for (var i = 1; i <= 5; i++) {
> // 没循环一次,吃一个包子
> console.log('我吃了一个包子')
> // 当 i 的值为 3 的时候,条件为 true,执行 {} 里面的代码终止循环
> // 循环就不会继续向下执行了,也就没有 4 和 5 了
> if (i === 3) {
> break
> }
> }
CONTINUE 结束本次循环
- 在循环中,把循环的本次跳过去,继续执行后续的循环
- 比如:吃五个包子,到第三个的时候,第三个掉地下了,不吃了,跳过第三个,继续吃第四个和第五个
- 跳过本次循环,就可以使用
continue关键字
for (var i = 1; i <= 5; i++) {
// 当 i 的值为 3 的时候,执行 {} 里面的代码
// {} 里面有 continue,那么本次循环后面的代码就都不执行了
// 自动算作 i 为 3 的这一次结束了,去继续执行 i = 4 的那次循环了
if (i === 3) {
console.log('这个是第三个包子,掉地下了,我不吃了')
continue
}
console.log('我吃了一个包子')
}
函数 (上)
- 我们代码里面所说的函数和我们上学的时候学习的什么三角函数、二次函数之类的不是一个东西
函数的概念
- 对于
js来说,函数就是把任意一段代码放在一个 盒子 里面 - 在我想要让这段代码执行的时候,直接执行这个 盒子 里面的代码就行
- 先看一段代码
// 这个是我们以前写的一段代码
for (var i = 0; i < 10; i++) {
console.log(i)
}
// 函数,这个 {} 就是那个 “盒子”
function fn() {
// 这个函数我们以前写的代码
for (var i = 0; i < 10; i++) {
console.log(i)
}
}
函数的两个阶段
- 按照我们刚才的说法,两个阶段就是 放在盒子里面 和 让盒子里面的代码执行
函数定义阶段
- 定义阶段就是我们把代码 放在盒子里面
- 我们就要学习怎么 放进去,也就是书写一个函数
- 我们有两种定义方式 声明式 和 赋值式
声明式
- 使用
function这个关键字来声明一个函数 - 语法:
function fn() {
// 一段代码
}
// function: 声明函数的关键字,表示接下来是一个函数了
// fn: 函数的名字,我们自己定义的(遵循变量名的命名规则和命名规范)
// (): 必须写,是用来放参数的位置(一会我们再聊)
// {}: 就是我们用来放一段代码的位置(也就是我们刚才说的 “盒子”)
赋值式
- 其实就是和我们使用
var关键字是一个道理了 - 首先使用
var定义一个变量,把一个函数当作值直接赋值给这个变量就可以了 - 语法:
var fn = function () {
// 一段代码
}
// 不需要在 function 后面书写函数的名字了,因为在前面已经有了
函数调用阶段
- 就是让 盒子里面 的代码执行一下
- 让函数执行
- 两种定义函数的方式不同,但是调用函数的方式都以一样的
调用一个函数
函数调用就是直接写 函数名() 就可以了
// 声明式函数
function fn() {
console.log('我是 fn 函数')
}
// 调用函数
fn()
// 赋值式函数
var fn2 = function () {
console.log('我是 fn2 函数')
}
// 调用函数
fn()
- 注意: 定义完一个函数以后,如果没有函数调用,那么写在 {} 里面的代码没有意义,只有调用以后才会执行
调用上的区别
- 虽然两种定义方式的调用都是一样的,但是还是有一些区别的
- 声明式函数: 调用可以在 定义之前或者定义之后
// 可以调用
fn()
// 声明式函数
function fn() {
console.log('我是 fn 函数')
}
// 可以调用
fn()
赋值式函数: 调用只能在 定义之后
// 会报错
fn()
// 赋值式函数
var fn = function () {
console.log('我是 fn 函数')
}
// 可以调用
fn()
函数的参数
- 我们在定义函数和调用函数的时候都出现过
() - 现在我们就来说一下这个
()的作用 - 就是用来放参数的位置
- 参数分为两种 行参 和 实参
// 声明式
function fn(行参写在这里) {
// 一段代码
}
fn(实参写在这里)
// 赋值式函数
var fn = function (行参写在这里) {
// 一段代码
}
fn(实参写在这里)
行参和实参的作用
-
行参
1.就是在函数外部不能使用
2 每写一个单词,就相当于在函数内部定义了一个可以使用的变量(遵循变量名的命名规则和命名规范)
3 多个单词之间以
,分隔
// 书写一个参数
function fn(num) {
// 在函数内部就可以使用 num 这个变量
}
var fn1 = function (num) {
// 在函数内部就可以使用 num 这个变量
}
// 书写两个参数
function fun(num1, num2) {
// 在函数内部就可以使用 num1 和 num2 这两个变量
}
var fun1 = function (num1, num2) {
// 在函数内部就可以使用 num1 和 num2 这两个变量
}
4.如果只有行参的话,那么在函数内部使用的值个变量是没有值的,也就是 `undefined``
5 行参的值是在函数调用的时候由实参决定的
2实参
. 在函数调用的时候给行参赋值的
也就是说,在调用的时候是给一个实际的内容的
function fn(num) {
// 函数内部可以使用 num
}
// 这个函数的本次调用,书写的实参是 100
// 那么本次调用的时候函数内部的 num 就是 100
fn(100)
// 这个函数的本次调用,书写的实参是 200
// 那么本次调用的时候函数内部的 num 就是 200
fn(200)
函数内部的行参的值,由函数调用的时候传递的实参决定
多个参数的时候,是按照顺序一一对应的
function fn(num1, num2) {
// 函数内部可以使用 num1 和 num2
}
// 函数本次调用的时候,书写的参数是 100 和 200
// 那么本次调用的时候,函数内部的 num1 就是 100,num2 就是 200
fn(100, 200)
参数个数的关系
.行参比实参少
- 因为是按照顺序一一对应的
行参少就会拿不到实参给的值,所以在函数内部就没有办法用到这个值
function fn(num1, num2) {
// 函数内部可以使用 num1 和 num2
}
// 本次调用的时候,传递了两个实参,100 200 和 300
// 100 对应了 num1,200 对应了 num2,300 没有对应的变量
// 所以在函数内部就没有办法依靠变量来使用 300 这个值
fn(100, 200, 300)
行参比实参多
- 因为是按照顺序一一对应的
- 所以多出来的行参就是没有值的,就是
undefined
function fn(num1, num2, num3) {
// 函数内部可以使用 num1 num2 和 num3
}
// 本次调用的时候,传递了两个实参,100 和 200
// 就分别对应了 num1 和 num2
// 而 num3 没有实参和其对应,那么 num3 的值就是 undefined
fn(100, 200)
函数的return
return返回的意思,其实就是给函数一个 返回值 和 终断函数
终断函数
- 当我开始执行函数以后,函数内部的代码就会从上到下的依次执行
- 必须要等到函数内的代码执行完毕
- 而
return关键字就是可以在函数中间的位置停掉,让后面的代码不在继续执行
> function fn() {
> console.log(1)
> console.log(2)
> console.log(3)
>
> // 写了 return 以后,后面的 4 和 5 就不会继续执行了
> return
> console.log(4)
> console.log(5)
> }
>
> // 函数调用
> fn()
返回值
- 函数调用本身也是一个表达式,表达式就应该有一个值出现
- 现在的函数执行完毕之后,是不会有结果出现的
// 比如 1 + 2 是一个表达式,那么 这个表达式的结果就是 3
console.log(1 + 2) // 3
function fn() {
// 执行代码
}
// fn() 也是一个表达式,这个表达式就没有结果出现
console.log(fn()) // undefined
return 关键字就是可以给函数执行完毕一个结果
function fn() {
// 执行代码
return 100
}
// 此时,fn() 这个表达式执行完毕之后就有结果出现了
console.log(fn()) // 100
- 我们可以在函数内部使用
return关键把任何内容当作这个函数运行后的结果
函数的优点
- 函数就是对一段代码的封装,在我们想调用的时候调用
- 函数的几个优点
- 封装代码,使代码更加简洁
- 复用,在重复功能的时候直接调用就好
- 代码执行时机,随时可以在我们想要执行的时候执行
预解析
- 预解析 其实就是聊聊
js代码的编译和执行 js是一个解释型语言,就是在代码执行之前,先对代码进行通读和解释,然后在执行代码- 也就是说,我们的
js代码在运行的时候,会经历两个环节 解释代码 和 执行代码
解释代码
-
因为是在所有代码执行之前进行解释,所以叫做 预解析(预解释)
-
需要解释的内容有两个
- 声明式函数
- 在内存中先声明有一个变量名是函数名,并且这个名字代表的内容是一个函数
- 声明式函数
-
var关键字- 在内存中先声明有一个变量名
看下面一段代码
fn() console.log(num)
function fn() { console.log('我是 fn 函数') }
var num = 100 经过预解析之后可以变形为
function fn() { console.log('我是 fn 函数') } var num
fn() console.log(num) num = 100
- 赋值式函数会按照
var关键字的规则进行预解析
函数(下)
作用域
- 什么是作用域,就是一个变量可以生效的范围
- 变量不是在所有地方都可以使用的,而这个变量的使用范围就是作用域
全局作用域
- 全局作用域是最大的作用域
- 在全局作用域中定义的变量可以在任何地方使用
- 页面打开的时候,浏览器会自动给我们生成一个全局作用域
window - 这个作用域会一直存在,直到页面关闭就销毁了
// 下面两个变量都是存在在全局作用域下面的,都是可以在任意地方使用的
var num = 100
var num2 = 200
局部作用域
- 局部作用域就是在全局作用域下面有开辟出来的一个相对小一些的作用域
- 在局部作用域中定义的变量只能在这个局部作用域内部使用
- 在
JS中只有函数能生成一个局部作用域,别的都不行 - 每一个函数,都是一个局部作用域
// 这个 num 是一个全局作用域下的变量 在任何地方都可以使用
var num = 100
function fn() {
// 下面这个变量就是一个 fn 局部作用域内部的变量
// 只能在 fn 函数内部使用
var num2 = 200
}
fn()
变量使用规则
- 有了作用域以后,变量就有了使用范围,也就有了使用规则
- 变量使用规则分为两种,访问规则 和 赋值规则
访问规则
-
当我想获取一个变量的值的时候,我们管这个行为叫做 访问
-
获取变量的规则:
- 首先,在自己的作用域内部查找,如果有,就直接拿来使用
- 如果没有,就去上一级作用域查找,如果有,就拿来使用
- 如果没有,就继续去上一级作用域查找,依次类推
- 如果一直到全局作用域都没有这个变量,那么就会直接报错(该变量 is not defined)
-
变量的访问规则 也叫做 作用域的查找机制
-
作用域的查找机制只能是向上找,不能向下找
function fn() {
var num = 100
}
fn()
console.log(num) // 发现自己作用域没有,自己就是全局作用域,没有再上一级了,直接报错
赋值规则
- 当你想给一个变量赋值的时候,那么就先要找到这个变量,在给他赋值
- 变量赋值规则:
- 先在自己作用域内部查找,有就直接赋值
- 没有就去上一级作用域内部查找,有就直接赋值
- 还没有再去上一级作用域查找,有就直接赋值
- 如果一直找到全局作用域都没有,那么就把这个变量定义为全局变量,再给他赋值
function fn() {
num = 100
}
fn()
// fn 调用以后,要给 num 赋值
// 查看自己的作用域内部没有 num 变量
// 就会向上一级查找
// 上一级就是全局作用域,发现依旧没有
// 那么就会把 num 定义为全局的变量,并为其赋值
// 所以 fn() 以后,全局就有了一个变量叫做 num 并且值是 100
console.log(num) // 100
递归函数
- 什么是递归函数
- 在编程世界里面,递归就是一个自己调用自己的手段
- 递归函数: 一个函数内部,调用了自己,循环往复
// 下面这个代码就是一个最简单的递归函数
// 在函数内部调用了自己,函数一执行,就调用自己一次,在调用再执行,循环往复,没有止尽
function fn() {
fn()
}
fn()
- 其实递归函数和循环很类似
- 需要有初始化,自增,执行代码,条件判断的,不然就是一个没有尽头的递归函数,我们叫做 死递归
简单实现一个递归
- 我们先在用递归函数简单实现一个效果
- 需求: 求 1 至 5 的和
- 先算 1 + 2 得 3
- 再算 3 + 3 得 6
- 再算 6 + 4 得 10
- 再算 10 + 5 得 15
- 结束
开始书写,写递归函数先要写结束条件(为了避免出现 “死递归”)
function add(n) {
// 传递进来的是 1
// 当 n === 5 的时候要结束
if (n === 5) {
return 5
}
}
add(1)
再写不满足条件的时候我们的递归处理
function add(n) {
// 传递进来的是 1
// 当 n === 5 的时候要结束
if (n === 5) {
return 5
} else {
// 不满足条件的时候,就是当前数字 + 比自己大 1 的数字
return n + add(n + 1)
}
}
add(1)
简单了解对象
- 对象是一个复杂数据类型
- 其实说是复杂,但是没有很复杂,只不过是存储了一些基本数据类型的一个集合
var obj = {
num: 100,
str: 'hello world',
boo: true
}
- 这里的
{}和函数中的{}不一样 - 函数里面的是写代码的,而对象里面是写一些数据的
- 对象就是一个键值对的集合
{}里面的每一个键都是一个成员- 也就是说,我们可以把一些数据放在一个对象里面,那么他们就互不干扰了
- 其实就是我们准备一个房子,把我们想要的数据放进去,然后把房子的地址给到变量名,当我们需要某一个数据的时候,就可以根据变量名里面存储的地址找到对应的房子,然后去房子里面找到对应的数据
创建一个对象
- 字面量的方式创建一个对象
// 创建一个空对象
var obj = {}
// 像对象中添加成员
obj.name = 'Jack'
obj.age = 18
- 内置构造函数的方式创建对象
/ 创建一个空对象
var obj = new Object()
// 向对象中添加成员
obj.name = 'Rose'
obj.age = 20
-
Object是js内置给我们的构造函数,用于创建一个对象使用的