牛客最新前端笔试题解析(一) this指向题目解析及扩展

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: 牛客最新前端笔试题解析(一) this指向题目解析及扩展

前言


这篇文章主要来解析牛客笔试题部分的 this 指向题目。首先我们先从宏观上了解一下 JavaScript 中得 this 指向问题。


牛客最新前端JS笔试百题


JavaScriptthis 共有四种绑定(包括隐式绑定丢失)加 ES6 新增得箭头函数绑定,如果把这几种绑定全学会了,this 指向完全不成问题。 this 指向更详细的讲解,可以去看前面写过的一篇博文,里面还有 38 道题目,详细完整的讲解了 this 指向问题。


传送门: 《2w字大章 38道面试题》彻底理清JS中this指向问题


  • 默认绑定: 非严格模式下 this 指向全局对象,严格模式下 this 会绑定为 undefined
  • 隐式绑定: 满足 XXX.fn() 格式,fnthis 指向 XXX。如果存在链式调用, this 永远指向最后调用它的那个对象
  • 隐式绑定丢失:起函数别名,通过别名运行;函数作为参数会造成隐式绑定丢失。
  • 显式绑定: 通过 call/apply/bind 修改 this 指向
  • new绑定: 通过 new 来调用构造函数,会生成一个新对象,并且把这个新对象绑定为调用函数的 this
  • 箭头函数绑定: 箭头函数没有 this ,它的 this 是通过作用域链查到外层作用域的 this ,且指向函数定义时的 this 而非执行时


题目一. 隐式绑定与隐式绑定丢失


var x = 1;
var obj = {
    x: 3,
    fun:function () {
        var x = 5;
        return this.x;
    }
};
var fun = obj.fun;
console.log(obj.fun(), fun());
复制代码


解析


JavaScript 对于引用类型,其地址指针存放在栈内存中,真正的本体是存放在堆内存中的。fun = obj.fun 相当于将 obj.fun 指向得堆内存指针赋值给了 fun,此后 fun 执行与 obj 不会有任何关系,发生隐式绑定丢失。


  • obj.fun(): 隐式绑定,fun 里面的 this 指向 obj,打印 3
  • fun(): 隐式绑定丢失: fun 默认绑定,非严格模式下,this 指向 window,打印 1


答案


3 1
复制代码


题目二: 隐式绑定丢失


var person = {
  age: 18,
  getAge: function() {
    return this.age;
  }
};
var getAge = person.getAge
console.log(getAge())
复制代码


简单的隐式绑定丢失问题


答案


undefined
复制代码


题目三: 隐式绑定丢失


var obj = {
    name:"zhangsan",
    sayName:function(){
        console.log(this.name);
    }
}
var wfunc = obj.sayName;
obj.sayName();
wfunc();
var name = "lisi";
obj.sayName();
wfunc();
复制代码


简单的隐式绑定问题,不多做赘述了。


答案


zhangsan
undefined
zhangsan
lisi
复制代码


题目四:new绑定


var a = 5;
function test() { 
    a = 0; 
    console.log(a); 
    console.log(this.a); 
    var a;
    console.log(a); 
}
new test();
复制代码


解析


使用new来构建函数,会执行如下四部操作:


  1. 创建一个空的简单JavaScript对象(即{});
  2. 为步骤1新创建的对象添加属性__proto__,将该属性链接至构造函数的原型对象 ;
  3. 将步骤1新创建的对象作为this的上下文 ;
  4. 如果该函数没有返回对象,则返回this


通过 new 来调用构造函数,会生成一个新对象,并且把这个新对象绑定为调用函数的 this


  • console.log(a): 打印变量 a 的值,当前 testAO 中存在 a 变量,打印 0
  • console.log(this.a): new 绑定 this 指向新的实例对象,当前题目没有给实例对象添加 a 属性,打印 undefined
  • console.log(a): 同第一个,打印 0


答案


0
undefined
0
复制代码


题目五:箭头函数与显式绑定


function fun () {
    return () => {
        return () => {
            return () => {
                console.log(this.name)
            }
        }
    }
}
var f = fun.call({name: 'foo'})
var t1 = f.call({name: 'bar'})()()
var t2 = f().call({name: 'baz'})()
var t3 = f()().call({name: 'qux'})
复制代码


  1. 箭头函数没有 this ,它的 this 是通过作用域链查到外层作用域的 this ,且指向函数定义时的 this 而非执行时。
  2. 箭头函数,不能通过 call\apply\bind 来修改 this 指向,但可以通过修改外层作用域的 this 来达成间接修改。
  3. JavaScript 是静态作用域,即函数的作用域在函数定义的时候就决定了,而箭头函数的 this 是通过作用域链查到的,因此箭头函数定义后,它的作用域链就定死了。


  • f = fun.call({name: 'foo'}): 将 fun 函数的 this 指向 {name: 'foo'},并返回一个箭头函数,因此箭头函数的 this 也指向 {name: 'foo'}
  • t1 = f.call({name: 'bar'})()(): 对第一层箭头函数执行 call 操作,无效,当前 this 仍指向 {name: 'foo'},第二层、第三层都是箭头函数,第三层的 this 也指向 {name: 'foo'},打印 foo
  • 后续 t2 t3 分别对第二层、第三层箭头函数使用 call ,无效,最终都打印 foo


答案


foo
foo
foo
复制代码


题目六:箭头函数


let obj1 = {
    a: 1,
    foo: () => {
        console.log(this.a)
    }
}
// log1
console.log(obj1.foo())
const obj2 = obj1.foo
// log2
console.log(obj2())
复制代码


解析


  1. obj1.foo 为箭头函数,obj1 为对象,无法提供外层作用域,因此 obj.foo 里面的 this 指向 window


  • obj1.foo(): 箭头函数,this 指向 window,打印 undefined
  • obj2 隐式绑定丢失: 打印 undefined


答案


undefined
undefined
复制代码


题目七: 综合题(推荐看)


var name = 'global';
var obj = {
    name: 'local',
    foo: function(){
        this.name = 'foo';
        console.log(this.name);
    }.bind(window)
};
var bar = new obj.foo();
setTimeout(function() {
    console.log(window.name);
}, 0);
console.log(bar.name);
var bar3 = bar2 = bar;
bar2.name = 'foo2';
console.log(bar3.name);
复制代码


解析


这个题的整体出题质量还是挺高的,首先咱们来把涉及到的知识罗列一下:


  1. bind 是显式绑定,会修改 this 指向,但 bind() 函数不会立即执行函数,会返回一个新函数
  2. setTimeout 是异步任务,同步任务执行完毕后才会执行异步任务
  3. 绑定优先级: new绑定 > 显式绑定 > 隐式绑定 > 默认绑定


解析


  • obj.foo 将它的 this 通过 bind 显式的绑定为 window,但 bind 不会立即执行
  • var bar = new obj.foo(): new 绑定优先级大于 bind ,因此 bind 失效了,此时 this 指向 new 实例,因此 obj.foo 内部的 console 打印 foo
  • barnew obj.foo() 的实例,console.log(bar.name) 打印 foo
  • setTimeout 异步任务,等到同步执行完毕再来调用它的回调
  • bar3 = bar2 = bar,将 bar2,bar3,bar 的地址都指向 bar 所指向的空间。
  • bar2.name = 'foo2',修改地址指向堆内存的值
  • console.log(bar3.name): 由于三个变量指向同一块地址,bar3 修改了 namebar3 也随之改变,打印 foo2
  • setTimeout 的回调执行,打印 global


答案


foo
foo
foo2
global
复制代码


题目八: 综合题目七修改


题目六虽然出的很有质量,但是我感觉有几个地方考察的让人不满足,咱们来改一下题目。


改法一: 去除 new 绑定


var name = 'global';
var obj = {
    name: 'local',
    foo: function(){
        this.name = 'foo';
        console.log(this.name);
    }.bind(window)
};
obj.foo();
setTimeout(function() {
    console.log(this.name);
}, 0);
console.log(name);
复制代码


  • obj.foo 没有做修改,它的 this 通过 bind 显式的绑定为 window,但 bind 不会立即执行
  • obj.foo() 执行,显式绑定优先级大于隐式绑定,因此此时 foo 函数 this -> window
  • this.name 相当于 window.name,修改了全局的 name 变量值,打印 foo
  • setTimeout 回调等同步代码执行完毕
  • 全局的 name 已经修改为 foo ,打印 foo
  • 执行 setTimeout 的回调,默认绑定,打印 foo


答案


foo
foo
foo
复制代码


改法一: 修改 bind 为 call


var name = 'global';
var obj = {
    name: 'local',
    foo: function(){
        this.name = 'foo';
        console.log(this.name);
    }
};
obj.foo.call(window);
var bar = new obj.foo();
setTimeout(function() {
    console.log(window.name);
}, 0);
console.log(bar.name);
var bar3 = bar2 = bar;
bar2.name = 'foo2';
console.log(bar3.name);
复制代码


callbind 的区别就在于 call 函数会立即执行


因此 obj.foo.call(window) 会立即执行函数,同时也修改了全局的 name 值。其他部分与上面所讲类似,不多做赘述了。


答案


foo
foo
foo
foo2
foo
复制代码


往期精彩文章




相关文章
|
1月前
|
机器学习/深度学习 编解码 前端开发
探索无界:前端开发中的响应式设计深度解析####
【10月更文挑战第29天】 在当今数字化时代,用户体验的优化已成为网站与应用成功的关键。本文旨在深入探讨响应式设计的核心理念、技术实现及最佳实践,揭示其如何颠覆传统布局限制,实现跨设备无缝对接,从而提升用户满意度和访问量。通过剖析响应式设计的精髓,我们将一同见证其在现代Web开发中的重要地位与未来趋势。 ####
46 7
|
1月前
|
编解码 前端开发 UED
探索无界:前端开发中的响应式设计深度解析与实践####
【10月更文挑战第29天】 本文深入探讨了响应式设计的核心理念,即通过灵活的布局、媒体查询及弹性图片等技术手段,使网站能够在不同设备上提供一致且优质的用户体验。不同于传统摘要概述,本文将以一次具体项目实践为引,逐步剖析响应式设计的关键技术点,分享实战经验与避坑指南,旨在为前端开发者提供一套实用的响应式设计方法论。 ####
64 4
|
1月前
|
监控 前端开发 JavaScript
探索微前端架构:构建可扩展的现代Web应用
【10月更文挑战第29天】本文探讨了微前端架构的核心概念、优势及实施策略,通过将大型前端应用拆分为多个独立的微应用,提高开发效率、增强可维护性,并支持灵活的技术选型。实际案例包括Spotify和Zalando的成功应用。
|
1月前
|
缓存 前端开发 JavaScript
"面试通关秘籍:深度解析浏览器面试必考问题,从重绘回流到事件委托,让你一举拿下前端 Offer!"
【10月更文挑战第23天】在前端开发面试中,浏览器相关知识是必考内容。本文总结了四个常见问题:浏览器渲染机制、重绘与回流、性能优化及事件委托。通过具体示例和对比分析,帮助求职者更好地理解和准备面试。掌握这些知识点,有助于提升面试表现和实际工作能力。
66 1
|
1月前
|
前端开发 JavaScript 开发者
揭秘前端高手的秘密武器:深度解析递归组件与动态组件的奥妙,让你代码效率翻倍!
【10月更文挑战第23天】在Web开发中,组件化已成为主流。本文深入探讨了递归组件与动态组件的概念、应用及实现方式。递归组件通过在组件内部调用自身,适用于处理层级结构数据,如菜单和树形控件。动态组件则根据数据变化动态切换组件显示,适用于不同业务逻辑下的组件展示。通过示例,展示了这两种组件的实现方法及其在实际开发中的应用价值。
40 1
|
2月前
|
人工智能 资源调度 数据可视化
【AI应用落地实战】智能文档处理本地部署——可视化文档解析前端TextIn ParseX实践
2024长沙·中国1024程序员节以“智能应用新生态”为主题,吸引了众多技术大咖。合合信息展示了“智能文档处理百宝箱”的三大工具:可视化文档解析前端TextIn ParseX、向量化acge-embedding模型和文档解析测评工具markdown_tester,助力智能文档处理与知识管理。
|
2月前
|
缓存 前端开发 JavaScript
前端的全栈之路Meteor篇(二):容器化开发环境下的meteor工程架构解析
本文详细介绍了使用Docker创建Meteor项目的准备工作与步骤,解析了容器化Meteor项目的目录结构,包括工程准备、环境配置、容器启动及项目架构分析。提供了最佳实践建议,适合初学者参考学习。项目代码已托管至GitCode,方便读者实践与交流。
|
2月前
|
JSON 前端开发 JavaScript
前端模块打包器的深度解析
【10月更文挑战第13天】前端模块打包器的深度解析
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
76 2
|
2月前
|
缓存 Java 程序员
Map - LinkedHashSet&Map源码解析
Map - LinkedHashSet&Map源码解析
78 0

推荐镜像

更多