JavaScript 函数进阶

  本文内容学习于:后盾人 (houdunren.com)

1.使用1et/const时不会压入 window,    直接function xx声明函数,会压入    window    

let hd = function() { console.log("后盾人");};
window.hd(); //window.hd is not a function

2.标准声明的函数优先级更高,解析器会优先提取函数并放在代码树顶端,所以标准声明函数位置不限制,所以下面的代码可以正常执行。

console.log(hd(3)); function hd(num) {
return ++num;};


3.标准声明优先级高于赋值声明

console.log(hd(3));//4
function hd(num) { return ++num;

var hd = function() { return "hd";};


4.立即执行函数,可以用来定义私有作用域防止污染全局作用域

"use strict";

(function () {
var web='houdunren';}) ();
console.log(web); //web is not defined

使用 let/const 有块作用域特性,所以使用以下方式也可以产生私有作用域

let web = 'houdunren';
console.log(web);

5.标准形式声明的函数也会提升到前面,优先级行于var变量提高(但变量函数定义不会被提升)


6.形参是在函数声明时设置的参数,实参指在调用函数时传递的值。形参数量大于实参时,没有传参的形参值为undefined实参数量大于形参时,多于的实参将忽略并不会报错


7.默认参数
1)以往默认参数的处理方式

//total:总价 year:年数
function avg(total, year) { year = year ll 1;
return Math.round(total / year);
console.log(avg(2000,3));


2)使用新版本默认参数方式如下(默认参数要放在最后面)

function avg(total, year = 1) { return Math.round(total / year)};
console.log(avg(2000,3));


8.arguments
arguments 是函数获得到所有参数集合,下面是使用 arguments 求和的例子

function sum() {
return [...arguments].reduce((total, num) => {
return (total += num);},0);
console.log(sum(2,3,4,2,6)); //17


更建议使用展示语法

function sum(...args) {
return args.reduce((a, b) => a + b);
console.log(sum(2,3,4,2,6)); //17

9.箭头函数
函数声明的简写形式,在使用递归调用、构造函数、事件处理器时不建议使用箭头函数。


1)无参数时使用空扩号即可

let sum = () => {
return 1 + 3;
console.log(sum());//4


2)函数体为单一表达式时不需要 return 返回处理,系统会自动返回表达式计算结果。

let sum = () => 1 + 3;

console.log(sum()); //4


3)多参数传递与普通声明函数一样使用逗号分隔


let hd = [1, 8, 3, 5].filter((item, index)=> {
return item <= 3;});
console.log(hd);


4)只有一个参数时可以省略括号

let hd = [1, 8,3, 5].filter(item => item <= 3); console.log(hd);


10.递归调用
递归指函数内部调用自身的方式。主要用于数量不确定的循环操作要有退出时机否则会陷入死循

通过阶乘来体验递归调用

function factorial(num =3) {
return num == 1 ? num : num * factorial(--num);

console.log(factorial(5)); //120 5*4*3*2


递归打印倒三角

function star(row = 5) {
if (row == 0) return "";
document.write("*".repeat(row) + "
"); star(--row);

累加计算方法

function sum(...num) {
return num.length == 0 ? 0 : num.pop() + sum(...num);
console.log(sum(l, 2,3,4,5,7,9)); //31


11.展开语法
展示语法或称点语法体现的就是收/放特性,做为值时是放,做为接收变量时是收。

let hd = [1,2,3];
let [a, b, c] = [...hd]; console.log(a);//1

console.log(b);//2

console.log(c);//3

[...hd] = [1,2,3];
console.log(hd); //[1,2,3]


使用展示语法可以替代 arguments 来接收任意数量的参数

function hd(...args) {
console.log(args);}
hd(1,2,3,"后盾人");//[1,2,3,"后盾人"]

也可以用于接收部分参数(多个参数时...参数必须放后面)

function hd(site, ...args) {
console.log(site, args);//后盾人(3)[1,2,3]
hd("后盾人",1,2,3);


使用 ... 可以接受传入的多个参数合并为数组,下面是使用点语法进行求合计算。

function sum(...params) {
console.log(params);
return params.reduce((pre, cur) => pre + cur);
console.log(sum(1,3,2,4));


12.标签函数
使用函数来解析标签字符串,第一个参数是字符串值的数组,其余的参数为标签变量。

function hd(str,...values) {
console.log(str);//["站点","_","",raw: Array(3)] console.log(values);//["后盾人","houdunren.com"]
let name = '后盾人',url = 'houdunren.com'; hd '站点${name}-${url}`;

13.this
调用函数时 this 会隐式传递给函数指项函数调用时的关联对象,也称之为函数的上下文。


13.1 函数调用
全局环境下this就是 window 对象的引用
使用严格模式时在全局函数内this为undefined


13.2 方法调用
函数为对象的方法时this 指向该对象

1)构造函数
函数当被 new时即为构造函数,一般构造函数中包含属性与方法。

函数中的上下文(this)指向到实例对象。构造函数主要用来生成对象,里面的 this 默认就是指当前对象

function User() {
this.name ="后盾人"; this.say=function() {
console.log(this);//User {name:"后盾人",say:?} return this.name;};}
let hd = new User(();
console.log(hd.say());//后盾人


2)对象字面量
下例中的 hd 函数不属于对象方法所以指向window show 属于对象方法执向 obj对象

let obj = {
site:"后盾人", show() {
console.log(this.site);//后盾人
console.log(`this in show method: ${this}`); //this in show method: [object object] function hd() {
console.log(typeof this.site); //undefined
console.log(`this in hd function: ${this}`); //this in hd function: [object Window]
hd();}
};
obj.show();

数组的map和foreach方法,允许提供一个函数作为参数。这个函数内部不应该使用this.

var o = {
v:'hello',
p:[ 'al', 'a2' ], f:function f() {
this.p.forEach(function (item){
console.log(this.v +''+ item);});}}
o.f()
// undefined al// undefined a2


解决办法1:解决这个问题的一种方法,就是前面提到的,使用中间变量固定

this var o = {
v: 'hello',
p: [ 'al', 'a2' ], f:function f() {
var that = this;
this.p.forEach(function (item){
console.log(that.v+''+item);});}}
o.f()
// hello al// hello a2

解决办法2:另一种方法是将this当作foreach方法的第二个参数,固定它的运行环境。

var o = { v: 'hello',
p: [ 'al', 'a2' ], f:function f() {
this.p.forEach(function (item) {
console.log(this.v+''+item);},this);}
o.f()
// hello al// hello a2
解决办法3:使用es6的箭头函数。 var o = {
v:'hello',
p: [ 'al', 'a2' ], f:function f() {
this.p.forEach((item)=>{
console.log(this.v+ii+ item);});}
o.f()
// hello al// hello a2


对于延时函数(setTimeout&setInterval)内部的回调函数的this指向全局对象window//默认情况下代码

function Person() {
this.age = o;
setTimeout(function() {
console.log(this);},3000);

var    =new Person();//3秒后返回    window 对象    
//通过bind绑定
function Person() {
this.age = o;
setTimeout((function() {
console.log(this);}).bind(this),3000);

var p =new Person();//3秒后返回构造函数新生成的对象 Person{...}

原型链中的方法的this仍然指向调用它的对象

var o = {
f :function (){
return this.a + this.b;

};
var p =Object.create(o); p.a = 1; p.b = 4;
console.log(p.f()); // 5

你可能感兴趣的:(JavaScript,javascript,前端,开发语言)