笔记目录
6 数组
- 作用:数组(Array)可以把一组相关的数据一起存放,并提供方便的访问(获取)方式。
- 概念:数组是一组数据的集合,其中的每个数据被称作数组元素,在数组中可以存放任意类型元素。数组是一种将一组数据存储在单个变量名下的方式。
6.1 创建数组
- new 关键字
var arr =new Array(); // 创建了一个空的数组
- 数组字面量
//创建一个空的数组
var arr=[];
//创建一个有元素的数组
var arr=[1, 2, 3, 4];
//再数组里面可以放任意的数据类型
var arr=[1, 2, "3", "张三", true];
6.2 提取数组元素
6.2.1 某元素:数组的索引(下标)
- 概念:用来访问数组元素的序号,序号从0开始。
- 方法:
数组名[索引]
var arr=[1,3,5,"张三","李四"]
arr[3] // 张三
arr[5] // undefined
6.2.2 全部元素:遍历数组
- 遍历:就是把数组中的元素全部访问一次。
- 注意:i必须从0开始,因为数组索引是从0开始的
var arr = [1, 2, 3, 4, 5];
for (var i = 0; i < 5; i++) { // i:0 1 2 3 4
console.log(arr[i]);
}
6.3 数组的长度
arr.length
// 1. 求数组 [2, 6, 1, 7, 4] 里面所有元素的和以及平均值
var arr = [2, 6, 1, 7, 4];
var sum = 0;
var average = 0;
for (var i = 0; i < arr.length; i++) {
sum += arr[i];
// 我们加的是数组元素 arr[i] ,不是计数器 i
}
average = sum / arr.length;
console.log(sum, average); // 想要输出多个变量,用逗号分隔即可
// 2. 求数组[2,6,1,77,52,25,7]中的最大值
var arr = [2, 6, 1, 77, 52, 25, 7, 99];
var max = arr[0];
for (var i = 1; i < arr.length; i++) {
if (arr[i] > max) {
max = arr[i];
}
}
console.log('该数组里面的最大值是:' + max); // 99
6.3.1 追加/替换元素 索引
var arr=["red","yellow","blue"];
arr[3]="pink";
// 后面追加元素 结果为arr=["red","yellow","blue","pink"];
arr[0]="yellow";
// 替换元素 结果为arr=["yellow","yellow","blue"];
6.3.2 选取元素到新数组 newArr[newArr.length]=arr[i];
// 将 arr=[2,0,6,1,77,0,52,0,25,7] 中大于10的元素单独存在一个数组中
var arr=[2,0,6,1,77,0,52,0,25,7];
var newArr=[]; // 当前newArr.length = 0
for(var i = 0; i < arr.length; i++)
{
if(arr[i]>10)
{
newArr[newArr.length]=arr[i];
}
}
console.log(newArr)
// [77, 52, 25]
6.4 翻转数组
var arr = ['red', 'green', 'blue', 'pink', 'purple', 'hotpink'];
var newArr = [];
for (var i = arr.length - 1; i >= 0; i--) {
newArr[newArr.length] = arr[i]
}
console.log(newArr);
// ['hotpink', 'purple', 'pink', 'blue', 'green', 'red']
7 函数
- 概念:封装了一段可被重复调用执行的代码块。
- 使用步骤:1、声明;2、调用
// 1、声明
function 函数名() // 函数名一般为动词
{
// 函数体
}
// 2、调用,执行代码
函数名();
// 利用函数求1~100的和
function getSum(num1,num2)
{
var sum = 0;
for(var i = num1; i <= num2; i++)
{
sum += i;
}
console.log(sum);
}
getSum(1,100);
7.1 形参与实参
// 声明函数
function 函数名(形参1,形参2,...) { // 形参:形式参数
}
// 执行函数
函数名(实参1,实参2,...); // 实参:实际参数
- 函数形参和实参个数不匹配的情况
function getSum(num1,num2)
{
console.log(num1+num2);
}
// 实参的个数 = 形参的个数
getSum(1,2); // 3
// 实参的个数 > 形参的个数
getSum(1,2,3); // 3
// 实参的个数 < 形参的个数
getSum(1); // NaN (形参不传值会默认为undefined)
7.2 return 语句
函数只是实现某种功能,最终的结果需要返回给函数的调用者
function 函数名()
{
//执行语句
return 需要返回的结果;
}
function getResult() {
return 666;
}
getResult(); // getResult() = 666
console.log(getResult()) // 666,需要用console.log()打印函数的返回值
注意:
- 只要遇到return,就把需要返回的结果返回出来,同时,自动跳出函数
- return一次只能返回一个值
- 如果我们想要返回多个值,可以使用数组:return [num1,num2,num3];
- 函数如果没有return,则返回的是undefined
break,continue,return的区别:
- break:结束当前的循环体
- continue:跳出本次循环,继续执行下次循环
- return:不仅可以跳出函数,还可以返回return语句中的值
7.3 arguments 使用
所有函数都内置了一个arguments对象,当我们不确定有多少个实参传递的时候,可以用arguments来获取。arguments对象中存储了传递的所有实参。
arguments展示形式是一个伪数组,因此可以进行遍历。伪数组具有以下特点︰
- 具有length属性
- 按照索引方式储存数据
- 不具有真正数组的
push()
,pop()
等方法``
function fn() // 不写形参
{
console.log(arguments);
}
fn(1,2,3); // 传几个实参就自动存几个到函数的arguments,此时arguments的值为:[1,2,3]
function fn() {
for (var i = 0; i < arguments.length; i++) {
console.log(arguments[i]);
}
}
fn(1, 2,3);
/* 1
2
3 */
案例:
// 1. 利用函数求任意个数的最大值
function getMax() { // arguments = [1,2,3]
var max = arguments[0];
for (var i = 1; i < arguments.length; i++) {
if (arguments[i] > max) {
max = arguments[i];
}
}
return max;
}
console.log(getMax(1, 2, 3));
console.log(getMax(1, 2, 3, 4, 5));
console.log(getMax(11, 2, 34, 444, 5, 100));
// 3 5 444
// 2. 利用函数翻转任意数组 reverse 翻转
function reverse(arr) {
var newArr = [];
for (var i = arr.length - 1; i >= 0; i--) {
newArr[newArr.length] = arr[i];
}
return newArr;
}
var arr1 = reverse([1, 3, 4, 6, 9]);
console.log(arr1);
var arr2 = reverse(['red', 'pink', 'blue']);
console.log(arr2);
7.4 函数相互调用
// 用户输入年份,输出当前年份2月份的天数
function backDay() {
var year = prompt('请您输入年份:');
if (isRunYear(year)) {
alert('当前年份是闰年2月份有29天');
}
else {
alert('当前年份是平年2月份有28天');
}
}
backDay();
// 判断是否为闰年的函数
function isRunYear(year) {
// 如果是闰年我们返回true,否则返回false
var flag = false;
if (year % 4 == 0 && year % 100 != 0 || year % 400 == 0) {
flag = true;
}
return flag;
}
7.5 声明函数的方式
- 用函数关键字自定义函数(命名函数):
function fn() {
}
fn();
- 函数表达式(匿名函数):
var 变量名 = function(形参) {
}
fn(); // 调用函数的方法一样,使用`fun(实参)`
8 作用域
概念:代码名字(变量)的作用范围
目的:提高程序的可靠性,减少命名冲突
分类:全局作用域、局部作用域(es6之前)
- 全局作用域:整个script标签 / 一个单独的js文件
<script>
var num=10;
</script>
- 局部作用域:函数内部
function fun()
{
var num=10;
}
8.1 全局变量和局部变量
- 全局变量: 全局作用域 下的变量
- 局部变量: 局部作用域 下的变量
function fun(aru) // 该形参也是局部变量,外部无法调用
{
var num1 = 10; // num1为局部变量
num2 = 20; // num2为全局变量(没有声明,直接赋值),不建议使用
}
注意:
- 全局变量:浏览器关闭的时候才会销毁,占内存资源
- 局部变量:程序执行完毕就会销毁,节约资源
8.2 作用域链
作用域链:内部函数访问外部函数的变量,采用链式查找(从内层到外层,一层一层地往外查找)的方式确定取值,采用就近原则。
function fun1()
{
var num=10;
function fun2()
{
var num=20;
function fun3()
{
console.log(num); // 20(就近原则)
}
}
}
num=20
相对于是fun3全局变量,但相对于fun1是局部变量,不可调用num=10
相对于fun2和fun3都是全局变量,可调用
9 函数预解析
为什么会出现以下情况:
console.log(num);
var num = 10;
// undefined
fun(); // 先调用也没事
function fun()
{
console.log(11);
}
// 11
fun();
var fun = function()
{
console.log(22);
}
// 报错
-
JS引擎运行JS分为以下两步:
-
- 预解析:JS会把JS里面所有的 var变量 和
function函数
放到 当前作用域的最前面
- 预解析:JS会把JS里面所有的 var变量 和
-
代码执行:按照代码书写的顺序从上到下执行
-
预解析分为以下两步:
-
- 变量预解析(变量提前): 把所有的 变量声明 提到 当前的作用域 最前面,不提
赋值操作
- 变量预解析(变量提前): 把所有的 变量声明 提到 当前的作用域 最前面,不提
console.log(num);
var num=10; // undefined
// 实际预解析步骤如下:
var num; // 把var提到`当前的作用域`最前面
console.log(num);
num = 10; // 不提`赋值操作`,结果为undefined
fun();
var fun=function() {
console.log(22);
} // 报错
// 实际预解析步骤如下:
var fun; // 把var提前
fun(); // 未定义函数,报错
fun=function() {
console.log(22);
}
- **函数预解析(函数提前):** 把所有的函数声明提升到当前作用域的最前面,不调用函数``
fun(); // 先写也没事儿,只是位置提前,实际没有被调用
function fun() {
console.log(11);
} // 11
// 实际预解析步骤如下:
function fun() { // JS会自动把函数声明部分提前
console.log(11);
}
fun(); // 再调用,结果为11
预解析案例:
// 案例1
var num = 10;
fun();
function fun() {
console.log(num);
var num = 20;
}
// 相当于执行了以下操作
var num;
function fun() {
var num; // var提升到当前作用域
console.log(num);
num = 20;
}
num = 10; // 剩余的函数从上往下依次写下来
fun();
// 结果:undefined
// 案例2
var num = 10;
function fn() {
console.log(num);
var num = 20;
console.log(num);
}
fn();
// 相当于以下代码
var num;
function fn() {
var num;
console.log(num);
num = 20;
console.log(num);
}
num = 10;
fn();
// 结果:undefined 20
// 案例3
var a = 18;
f1();
function f1() {
var b = 9;
console.log(a);
console.log(b);
var a = '123';
}
// 相当于以下代码
var a;
function f1() {
var b;
var a;
b = 9;
console.log(a); // a取值应该往上找
console.log(b);
a = '123';
}
a = 18;
f1();
// 结果:undefined 9
// 案例4
f1();
console.log(c);
console.log(b);
console.log(a);
function f1() {
var a = b = c = 9;
console.log(a);
console.log(b);
console.log(c);
}
// 相当于以下代码
function f1() {
var a;
a = b = c = 9;
// 相当于 var a = 9; b = 9; c = 9; b 和 c 直接赋值,没有var 声明,应该当全局变量看
// 集体声明 var a = 9, b = 9, c = 9;
console.log(a);
console.log(b);
console.log(c);
}
f1();
console.log(c); // 直接赋值为全部变量,可调用
console.log(b);
console.log(a); // 函数内部定义的变量为局部变量,不可调用
// 结果:9 9 9 9 9 报错