面试必问但日常不愿意看的题

news2025/1/20 6:02:15

  • 1,做道 this 相关的题,看你对 js 的 this 掌握的如何
  • 2,BFC 这样答才完美
    • 1,什么是 BFC?其规则是什么?
    • 2,如何触发 BFC
    • 3,BFC 到底可以解决什么问题呢
  • 3,作用域
  • 4,作用域链
  • 5,闭包

1,做道 this 相关的题,看你对 js 的 this 掌握的如何

var name = 'Jerry';
var a = {
  name: 'Tom',
  say: function () {
    console.log(this.name);
  },
};
var fun = a.say;
fun();
a.say();
var b = {
  name: 'xiaowang',
  say: function (fun) {
    fun();
  },
};
b.say(a.say);
b.say = a.say;
b.say();

2,BFC 这样答才完美

BFC 算是前端面试的高频考点了,但是每次回答的都不尽人意(至少我是这样的)。

回答这个问题我们可以从三点来进行

  • 1,什么是 BFC?其规则是什么?
  • 2,怎么触发 BFC
  • 3,BFC 它能够解决什么问题

1,什么是 BFC?其规则是什么?

BFC 的全称是 Block formatting context 对应中文是:块级格式化上下文,它是一个独立的渲染区域,我们可以把 BFC 理解为一个封闭的容器,内部的元素无论怎么布局都不会影响外部,容器内的样式布局自然也不会受外界的影响。

它内部的规则有:

  • 1,BFC 就是一个块级元素,块级元素会在垂直方向一个接一个排列
  • 2,BFC 就是页面中的一个隔离的独立容器,容器里的标签不会影响到外部标签
  • 3,BFC 区域不会与浮动的容器发生重叠
  • 4,属于同一个 BFC 的两个相邻元素的外边距会发生重叠,垂直方向的距离由两个元素中 margin 的较大值决定
  • 5,计算 BFC 的高度时,浮动元素也会参与计算

2,如何触发 BFC

我们可以通过添加一些 css 属性来触发,常见的有:

  • overflower 除了 visible 以外的值
  • position 的值设置为 absolute 或者 fixed
  • display 的值设置为 inline-block 或者 flex

3,BFC 到底可以解决什么问题呢

  • 1,它可以阻止元素被浮动元素覆盖,比如两栏布局

    <div class="left">left</div>
    <div class="right">right</div>
    .left {
      width: 100px;
      height: 100px;
      border: 1px solid aqua;
      float: left;
    }
    .right {
      background-color: rgb(255, 200, 209);
    }

由图可以看出,由于左边的 div 浮动后脱离文档流不占空间了,就会导致右边的 div 到了最左边,同时左侧浮动的 div 还会覆盖在上面,这时候我们就可以通过把右侧的 div 设置为一个 BFC,比如可以给它添加 display: flex 来触发,就可以解决右侧被左侧覆盖的问题

.right {
  width: auto;
  background-color: rgb(255, 200, 209);
  display: flex;
}
  • 2,解决父元素没有高度,子元素设置为浮动元素时,产生的父元素高度塌陷的问题,比如一个容器内的两个 div 都是浮动元素,此时给父元素设置背景色是没有任何效果的,因为此时父元素的高度为 0
<div class="p">
  <div>1</div>
  <div>2</div>
</div>

.p div {
  width: 100px;
  height: 100px;
  border: 1px solid aqua;
  float: left;
}
.p {
  background-color: rgb(157, 255, 0);
}

这时候我们就可以添加一个可以触发父元素 BFC 功能的属性,因为 BFC 有个规则是计算 BFC 的高度时,浮动元素也参与计算,所以触发 BFC 后,父元素的高度就会被撑开,也就是会产生清除浮动的效果

.p {
  background-color: rgb(157, 255, 0);
  overflow: hidden;
}
  • 3,可以解决 margin 边距重叠的问题,比如一个容器里有两个 div,这两个 div 分别设置了自己的 margin,一个是 10px,一个是 20px,正常两个盒子的垂直距离是 30px,不然,我们会发现两个盒子之间的垂直距离是 20px,这就是 margin 塌陷问题
<div>
  <div class="t">1</div>
  <div class="b">2</div>
</div>
.t,
.b {
  width: 100px;
  height: 100px;
  background-color: rgb(245, 108, 73);
  margin: 10px;
}
.b {
  background-color: paleturquoise;
  margin: 20px;
}

这就是 margin 塌陷问题,此时 margin 边距的结果为两个 div 间的较大值,如果我们想让 margin 外边距变为正常的 30,可以触发一个 div 的 BFC,它的内部就会遵循 BFC 规则,这样就可以为元素包裹一个盒子形成一个完全独立的空间,做到里面的元素不受外部元素的影响

<div>
  <div class="t">1</div>
  <div style="overflow: hidden">
    <div class="b">2</div>
  </div>
</div>

3,作用域

作用城是指程序源代码中定义的范国, 分为全局作用域局部作用域也叫函数作用域, 作用域规定了如何设置变量,也就是确定当前执行代码对变量的访问权限, 函数作用城采用词法作用域,也就是静态作用域

  • 所谓词法作用域就是在函数定义的时候, 就已经确定了

    let value = 1;
    function fn({
      console.log(value);
    }
    function foo({
      let value = 2;
      fn();
    }
    foo();  // 1 函数是一个静态作用域
  • 变量对象

    变量对象是当前代码段中,所有的变量(变量函数 形参 arguments)组成的一个对象,变量对象是在执行上下文中被激活的,只有变量对象被激活了,在这段代码中才能使用所有的变量,变量对象分为全局变量对象局部变量对象 全局简称为 Variable Object VO 函数由于执行才被激活 称为 Active Object AO

    let value = 1;
    function fn({
      console.log(value);
    }
    function foo({
      let value = 2;
      fn();
    }
    foo();  // 1

    以上代码 VO 和 AO 示意图:

4,作用域链

在 js 中,函数存在一个隐式属性scopes,其实scopes存的就是变量的集合 VO,这个属性用来保存当前函数的执行上下文环境,由于在数据结构上是链式的,因此也被称作是作用域链,我们可以把它理解为一个数组,可以理解为是一系列的 AO 对象所组成的一个链式结构。

scopes属性在函数声明的时候产生的,在函数调用的时候更新,即在函数被调用的时候,将该函数的 AO 对象压入 scopes 中。

  • 作用域链的作用

    作用域链的作用是保证执行环境里有权访问的变量和函数是有序的,作用域链的变量只能向上访问,变量访问到 window 对象即被终止,作用域链向下访问变量是不被允许的 最直观的表现就是:函数内部可以访问到函数外部声明的变量

let value = 100;
function fn({
  console.log(value);  // 100  函数内部可以访问到函数外部声明的变量
  let res = 200;
}
fn();
console.log(res);  // res is not defined  函数外部访问不到函数内部的变量

画出以下代码的作用域链:

let global;
function fn({
  var a = 10;
  function foo({
    var b = 100;
  }
  foo();
}
fn();
// fn定义 fn执行 foo定义 foo执行
// 如果没有fn执行会有foo定义吗?没有,所以说b定义的时候和a执行的时候作用域链是一样的

函数执行完之后会销毁,所谓的销毁就是作用域链断开。

将以上代码改为典型的闭包:

let global;
function fn() {
  var a = 10;
  function foo() {
    console.log(a);
  }
  return foo;
}
let res = fn();
res();  // 10   fn()函数调用的时候,fn函数的作用域链会断掉,即a变量会被销毁,但是foo定义的时候,foo的作用域链会连接上,所以还是可以访问到a变量的。

5,闭包

它指的是有权访问另一个函数作用域中的变量的函数。具体来说,它可以被理解为定义在一个函数内部的函数,这样内部函数就可以访问到外部函数的局部变量。

闭包的存在是为了允许我们间接访问函数内部的变量,同时也能延长这些变量的使用寿命,并减少命名空间污染

两个函数嵌套,一个函数有权访问到另一个函数中的变量就形成了闭包,这个时候会造成内存的泄漏吗?

并不一定会造成内存泄漏,造成内存泄漏的必要条件是:内部的函数要保存到外面再进行执行的时候,才会延长fn函数的作用域链才会造成内存的泄漏

1,闭包的实战应用
1,防抖节流函数是闭包
2,单例模式是闭包
3,回调函数
// 回调函数是闭包吗?是
function add(num1, num2, callback{
  let sum = num1 + num2;
  if (typeof callback === "function") {
    callback(sum);
  }
}
add(1020function (sum{
  console.log(sum);
});  
// 30
4,手写js的方法,手写bind方法是闭包
let foo = {
  name: "xiaowang",
};
function getName() {
  console.log(this.name);
}
Function.prototype.myBind = function (obj) {
  let _t = this;
  return function () {
    return _t.call(obj);
  };
};
let getFooName = getName.myBind(foo);
getFooName();

5,定时器传参是闭包

function fn(a) {
  return fucntion (){
    console.log(a)
  }
}
setTimeout(fn(123), 1000)

6,利用闭包判断数据类型是闭包

function isType(type) {
  return function (target) {
    return `[object ${type}]` === Object.prototype.toString.call(target);
  };
}
const isArray = isType("Array");
console.log(isArray([1, 2, 3])); // true
console.log(isArray(1)); // false

7,封装私有变量和函数

function createPerson(name) {
  let age = 0;
  return {
    getName: function () {
      return name;
    },
    getAge: function () {
      return age;
    },
    setAge: function (newAge) {
      age = newAge;
    },
  };
}
let person = createPerson("xiaowang");
console.log(person.getName()); // xiaowang
console.log(person.getAge()); //0
person.setAge(18);
console.log(person.getAge()); //18
8,高阶函数
  • 高阶函数除了可以接受函数作为参数外,还可以把函数作为结果值返回
  • 实现对一个数组的求和
function sum(arr) {
  return arr.reduce(function (x, y) {
    return x + y;
  });
}
let res = sum([10, 20, 30, 40]);
console.log(res); //100
  • 但是,如果不需要立即求和,而是在后面的代码中,根据需要再计算怎么办?可以不返回求和的结果,而是返回求和的函数
function lazy_sum(arr) {
  let sum = function () {
    return arr.reduce(function (x, y) {
      return x + y;
    });
  };
  return sum;
}
let res = lazy_sum([10, 20, 30, 40]);
console.log(res()); //100

9,迭代器(执行一次函数往下取一个值)

let arr = ["xiaowang""xiaoli""xiaoma"];
function incre(arr) {
  let i = 0;
  return function () {
    return arr[i++];
  };
}
let next = incre(arr);
console.log(next()); // xiaowang
console.log(next()); // xiaoli
console.log(next()); // xiaoma
console.log(next()); // undefined
10, 缓存

比如求和操作,如果没有缓存,每次调用都要重新计算,采用缓存已经执行过的去查找,查找到了就直接返回,不需要重新计算

let fn = (function () {
  let cache = {};
  let calc = function (arr) {
    let sum = 0;
    for (let i = 0; i < arr.length; i++) {
      sum += arr[i];
    }
    return sum;
  };
  return function () {
    let args = Array.prototype.slice.call(arguments, 0);
    let key = args.join(",");
    let result,
        totalCache = cache[key];
    if (totalCache) {
      // 如果缓存有
      console.log("从缓存中取", cache);
      result = totalCache;
    } else {
      result = cache[key] = calc(args);
      console.log("存入缓存:", cache);
    }
    return result;
  };
})();
fn(1, 2, 3, 4, 5, 6); // 存入缓存: {1,2,3,4,5,6: 21}
fn(1, 2, 3, 4, 5, 6); // 从缓存中取 {1,2,3,4,5,6: 21}
fn(1, 2, 3, 4, 5, 6, 7, 8); // 存入缓存: {1,2,3,4,5,6: 21, 1,2,3,4,5,6,7,8: 36}
fn(1, 2, 3, 4, 5, 6, 7, 8); // 从缓存中取: {1,2,3,4,5,6: 21, 1,2,3,4,5,6,7,8: 36}
fn(1, 2, 3, 4, 5, 6, 7, 8); // 从缓存中取: {1,2,3,4,5,6: 21, 1,2,3,4,5,6,7,8: 36}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1475369.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

疫情物资智能管理:Java与SpringBoot的实践

✍✍计算机毕业编程指导师 ⭐⭐个人介绍&#xff1a;自己非常喜欢研究技术问题&#xff01;专业做Java、Python、微信小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。 ⛽⛽实战项目&#xff1a;有源码或者技术上的问题欢迎在评论区一起讨论交流&#xff01; ⚡⚡ Java、…

Python算法100例-2.6 分糖果

完整源代码项目地址&#xff0c;关注博主私信源代码后可获取 1.问题描述2.问题分析3.算法设计4.确定程序框架5.完整的程序6.运行结果 1&#xff0e;问题描述 10个小孩围成一圈分糖果&#xff0c;老师分给第1个小孩10块&#xff0c;第2个小孩2块&#xff0c;第3个小孩8块&…

区块链智能合约开发

一.区块链的回顾 1.区块链 区块链实质上是一个去中心化、分布式的可进行交易的数据库或账本 特征: 去中心化&#xff1a;简单来说&#xff0c;在网络上一个或多个服务器瘫痪的情况下&#xff0c;应用或服务仍然能够持续地运行&#xff0c;这就是去中心化。服务和应用部署在…

SpringMVC 学习(八)之文件上传与下载

目录 1 文件上传 2 文件下载 1 文件上传 SpringMVC 对文件的上传做了很好的封装&#xff0c;提供了两种解析器。 CommonsMultipartResolver&#xff1a;兼容性较好&#xff0c;可以兼容 Servlet3.0 之前的版本&#xff0c;但是它依赖了 commons-fileupload …

Eavesdropping(窃听机制)在机器学习中的用法

1. 简单翻译 考虑一个对任务 T 和 T’ 有用的特征 F&#xff0c;它在学习 T 时很容易学习&#xff0c;但在学习 T’ 时很难学习&#xff0c;因为 T’ 以更复杂的方式使用 F。网络学习 T 将学习 F&#xff0c;但网络学习 T’ 可能不会。如果网络学习 T’ 也学习 T&#xff0c;T…

每日汇评:黄金多头拒绝在美国宏观数据发布前放弃

周三早些时候&#xff0c;金价买家再次测试两周高点 2041美元&#xff1b; 美元延续反弹&#xff0c;但疲弱的国债收益率可能限制其上涨空间&#xff1b; 由于金价等待美国数据&#xff0c;4小时图表技术面似乎具有建设性&#xff1b; 金价正在复制周二亚洲交易中的价格走势&am…

江科大stm32 定时器 TIM输出比较--学习笔记

这几天遇到输出比较相关的问题&#xff0c;于是来学习下TIM输出比较部分知识点&#xff01; 输出比较简介 CNT是计数器的值&#xff0c;CCR寄存器是捕获/ 比较寄存器 简单的讲&#xff0c;输出比较就是用来输出PWM波形。 PWM简介 占空比&#xff1a;高电平占一个周期的比例。…

VScode打开keil5软件的内容

VScode想要打开keil5软件的内容&#xff0c;需要在此引入 具体可参考&#xff1a; VS Code环境下编辑、编译、下载Keil工程代码

本届挑战赛亚军方案:面向微服务架构系统中无标注、多模态运维数据的异常检测、根因定位与可解释性分析

CheerX团队来自于南瑞研究院系统平台研发中心&#xff0c;中心主要从事NUSP电力自动化通用软件平台的关键技术研究与软件研发。 选题分析 图1 研究现状 本次CheerX团队的选题紧密贴合了目前的运维现状。实际运维中存在多种问题导致运维系统的不可用。比如故障发生时&#xff…

【Maven】Maven 基础教程(一):基础介绍、开发环境配置

Maven 基础教程&#xff08;一&#xff09;&#xff1a;基础介绍、开发环境配置 1.Maven 是什么1.1 构建1.2 依赖 2.Maven 开发环境配置2.1 下载安装2.2 指定本地仓库2.3 配置阿里云提供的镜像仓库2.4 配置基础 JDK 版本2.5 配置环境变量 1.Maven 是什么 Maven 是 Apache 软件…

Docker部署Portainer图形化管理工具

文章目录 前言1. 部署Portainer2. 本地访问Portainer3. Linux 安装cpolar4. 配置Portainer 公网访问地址5. 公网远程访问Portainer6. 固定Portainer公网地址 前言 Portainer 是一个轻量级的容器管理工具&#xff0c;可以通过 Web 界面对 Docker 容器进行管理和监控。它提供了可…

2024最后一次Java面试,java高级开发面试经验

这里写自定义目录标题 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants 创建一个自定义列表如何创建一个…

Python Skip-Gram代码实战,Skip-Gram代码超简单讲解和步骤拆解,Word2vec代码构建思路,Skip-Gram代码实例,模板套用

1. Skip-Gram介绍 Skip-gram模型是Word2Vec模型的一种训练方法&#xff0c;它的目标是通过目标词预测上下文词。Skip-gram模型通过神经网络结构来学习每个单词的向量表示。 在Skip-gram模型中&#xff0c;每个单词被表示为一个固定维度的向量&#xff0c;该向量称为嵌入向量或词…

AStar算法(大物件寻路)

前言 A星(物件大小为一格)寻路&#xff0c;都很熟悉了吧&#xff0c;网上源码一堆&#xff0c;随便抄; 这章需要讲述 大物件的A星寻路&#xff0c;何为大物件&#xff0c;就是 比如 物件 为4个格子&#xff1b; 这样&#xff0c;原来的A星 没法直接用了&#xff0c;必须得改装…

【Java程序员面试专栏 算法思维】四 高频面试算法题:回溯算法

一轮的算法训练完成后,对相关的题目有了一个初步理解了,接下来进行专题训练,以下这些题目就是汇总的高频题目,本篇主要聊聊回溯算法,主要就是排列组合问题,所以放到一篇Blog中集中练习 题目关键字解题思路时间空间岛屿数量网格搜索分别向上下左右四个方向探索,遇到海洋…

微信小程序引入Vant插件

Vant官网&#xff1a;Vant Weapp - 轻量、可靠的小程序 UI 组件库 先查看官网的版本 新建一个package.json页面&#xff0c;代码写上&#xff1a;&#xff08;我先执行的npm安装没出package页面&#xff0c;所以先自己创建了一个才正常&#xff09; {"dependencies"…

【Spring底层原理高级进阶】基于Spring Boot和Spring WebFlux的实时推荐系统的核心:响应式编程与 WebFlux 的颠覆性变革

&#x1f389;&#x1f389;欢迎光临&#x1f389;&#x1f389; &#x1f3c5;我是苏泽&#xff0c;一位对技术充满热情的探索者和分享者。&#x1f680;&#x1f680; &#x1f31f;特别推荐给大家我的最新专栏《Spring 狂野之旅&#xff1a;底层原理高级进阶》 &#x1f680…

Bicycles(变形dijkstra,动态规划思想)

Codeforces Round 918 (Div. 4) G. Bicycles G. Bicycles 题意&#xff1a; 斯拉夫的所有朋友都打算骑自行车从他们住的地方去参加一个聚会。除了斯拉维奇&#xff0c;他们都有一辆自行车。他们可以经过 n n n 个城市。他们都住在城市 1 1 1 &#xff0c;想去参加位于城市…

c++实现栈和队列类

c实现栈和队列类 栈(Stack)Stack示意图Stack.cpp 队列(queue)queue 示意图queue.cpp 栈(Stack) Stack示意图 Stack.cpp #pragma once #include "ListStu.cpp"template<typename T> class Stack { public: /* * void push(T& tDate)* 参数一 &#xff1a;…

Android和Linux的开发差异

最近开始投入Android的怀抱。说来惭愧&#xff0c;08年就听说这东西&#xff0c;当时也有同事投入去看&#xff0c;因为恶心Java&#xff0c;始终对这玩意无感&#xff0c;没想到现在不会这个嵌入式都快要没法搞了。为了不中年失业&#xff0c;所以只能回过头又来学。 首先还是…