JS 编程题

𝑔𝑜𝑜𝑔𝑥ℎ2023年2月18日
大约 9 分钟

1. 写出下面这段代码的输出结果是?

var out = 25,
    inner = {
        out : 20,
        func: function () {
            var out = 30;
            return this.out;
        }
    }
console.log((inner.func, inner.func)());
console.log(inner.func());
console.log((inner.func)());
console.log((inner.func = inner.func)());

//输出结果 :

25
20
20
25

解析:

本题考点两个: 1, 作用域;2. 运算符(赋值运算符,逗号运算符)

第一个考的就是逗号运算符,逗号运算符就是运算前面的,返回最后一个;

例如:var i= 1;j = 2; k = 5; console.log((i++, j++, k)),最后输出的是5;

回到题目,逗号运算符会返回inner.func,而inner.func是一个匿名函数,而这个匿名函数是属于window的,所以里面的this指向的是window,输出25;

第二,第三个输出都是20,因为this指向的是inner

最后一个考的是等号运算符,inner.func = inner.func其实返回的是运算的结果,也就是返回的是匿名函数,所以this也是指向window,输出的是25

2. 以下代码片段,输出的结果是?简要解释一下

var name = '1';
var object = {
    name: '2',
    getNameFunc: function () {
        return function () {
            return this.name;
        }
    }
}
console.log(object.getNameFunc()())

//输出结果: 1

解释:

object.getNameFunc()执行完后,返回的是一个匿名函数,所以里面的this指向的就是window,最终结果输出的是1;

3. 下面代码的输出是什么?

function sayHi() {
    console.log(name);
    console.log(age);
    var name = '张三'let age = 21;
}
sayHi();

答案: undefined 和 ReferenceError

在函数中,我们首先使用var关键字声明了name变量。

这意味着变量在创建阶段会被提升(JavaScript会在创建变量创建阶段为其分配内存空间),默认值为undefined,直到我们实际执行到使用该变量的行。

我们还没有为name变量赋值,所以它仍然保持undefined的值。

使用let关键字(和const)声明的变量也会存在变量提升,但与var不同,初始化没有被提升。

在我们声明(初始化)它们之前,它们是不可访问的。 这被称为“暂时死区”。

当我们在声明变量之前尝试访问变量时,JavaScript会抛出一个ReferenceError。

4. 下面代码输出是什么?

for (var i = 0; i < 3; i++) {
  setTimeout(() => console.log(i), 1);
}

for (let i = 0; i < 3; i++) {
  setTimeout(() => console.log(i), 1);
}

答案:3 3 3 and 0 1 2

由于JavaScript中的事件执行机制,setTimeout函数真正被执行时,循环已经走完。

由于第一个循环中的变量i是使用var关键字声明的,因此该值是全局的。

在循环期间,我们每次使用一元运算符++都会将i的值增加1。

因此在第一个例子中,当调用setTimeout函数时,i已经被赋值为3。

在第二个循环中,使用let关键字声明变量i:使用let(和const)关键字声明的变量是具有块作用域的(块是{}之间的任何东西)。

在每次迭代期间,i将被创建为一个新值,并且每个值都会存在于循环内的块级作用域。

5. 下面代码的输出是什么?

const shape = {
  radius: 10,
  diameter() {
    return this.radius *2;
  },
perimeter: () => 2* Math.PI * this.radius
};

shape.diameter();
shape.perimeter();

答案: 20 and NaN

请注意,diameter是普通函数,而perimeter是箭头函数。

对于箭头函数,this关键字指向是它所在上下文(定义时的位置)的环境,与普通函数不同!

这意味着当我们调用perimeter时,它不是指向shape对象,而是指其定义时的环境(window)。

没有值radius属性,返回undefined。

6. 下面代码的输出是什么?

+true;
!"张三";

答案:1 and false

一元加号会尝试将boolean类型转换为数字类型。 true被转换为1,false被转换为0。

字符串'Lydia'是一个真值。 我们实际上要问的是“这个真值是假的吗?”。 这会返回false。

7、如何判断一个对象是否为数组

function isArray(arg) {
    if (typeof arg === 'object') {
        return Object.prototype.toString.call(arg) === '[object Array]';
    }
    return false;
}

8、冒泡排序

每次比较相邻的两个数,如果后一个比前一个小,换位置

var arr = [3, 1, 4, 6, 5, 7, 2];

function bubbleSort(arr) {
for (var i = 0; i < arr.length - 1; i++) {
    for(var j = 0; j < arr.length - 1; j++) {
        if(arr[j + 1] < arr[j]) {
            var temp;
            temp = arr[j];
            arr[j] = arr[j + 1];
            arr[j + 1] = temp;
        }
    }
}
return arr;
}

console.log(bubbleSort(arr));

9、快速排序(二分法)

采用二分法,取出中间数,数组每次和中间数比较,小的放到左边,大的放到右边

var arr = [3, 1, 4, 6, 5, 7, 2];

function quickSort(arr) {
    if(arr.length == 0) {
        return [];    // 返回空数组
    }

    var cIndex = Math.floor(arr.length / 2);
    var c = arr.splice(cIndex, 1);
    var l = [];
    var r = [];

    for (var i = 0; i < arr.length; i++) {
        if(arr[i] < c) {
            l.push(arr[i]);
        } else {
            r.push(arr[i]);
        }
    }

    return quickSort(l).concat(c, quickSort(r));
}

console.log(quickSort(arr));

10. 下面的代码输出什么?

const promise = new Promise((resolve, reject) => {
    console.log(1);
    resolve();
    console.log(2);
})

promise.then(() => {
    console.log(3);
})

console.log(4);

// 答案
1
2
4
3

解析

首先 Promise 新建后立即执行,所以会先输出 1,2,而 Promise.then() 内部的代码在 当次 事件循环的 结尾 立刻执行 ,所以会继续输出4,最后输出3。

11. 下面的代码输出什么?

const promise = new Promise((resolve, reject) => {
    resolve('success1');
    reject('error');
    resolve('success2');
});

promise.then((res) => {
    console.log('then:', res);
}).catch((err) => {
    console.log('catch:', err);
})

// 答案
then: success1

解析

resolve 函数将 Promise 对象的状态从“未完成”变为“成功”(即从 pending 变为 resolved),在异步操作成功时调用,并将异步操作的结果,作为参数传递出去;

reject 函数将 Promise 对象的状态从“未完成”变为“失败”(即从 pending 变为 rejected),在异步操作失败时调用,并将异步操作报出的错误,作为参数传递出去。

而一旦状态改变,就不会再变。 所以 代码中的reject('error'); 不会有作用。

Promise 只能 resolve 一次,剩下的调用都会被忽略。 所以 第二次的 resolve('success2'); 也不会有作用。

12. 下面代码输出什么?

Promise.resolve(1)
  .then(2)
  .then(Promise.resolve(3))
  .then(console.log)

解析

Promise.resolve 方法的参数如果是一个原始值,或者是一个不具有 then 方法的对象,则 Promise.resolve 方法返回一个新的 Promise 对象,状态为resolved,Promise.resolve 方法的参数,会同时传给回调函数。

then 方法接受的参数是函数,而如果传递的并非是一个函数,它实际上会将其解释为 then(null),这就会导致前一个 Promise 的结果会穿透下面。

Promise.resolve(1)
  .then(null) // 穿透
  .then(null) // 穿透
  .then(console.log)

// 答案
1

13. 实现需求

红灯三秒亮一次,绿灯一秒亮一次,黄灯2秒亮一次;如何让三个灯不断交替重复亮灯?(用Promse实现)三个亮灯函数已经存在:

function red() {
    console.log('red');
}
function green() {
    console.log('green');
}
function yellow() {
    console.log('yellow');
}

解析

红灯三秒亮一次,绿灯一秒亮一次,黄灯2秒亮一次,意思就是3秒,执行一次 red 函数,2秒执行一次 green 函数,1秒执行一次 yellow 函数,不断交替重复亮灯,意思就是按照这个顺序一直执行这3个函数,这步可以就利用递归来实现。

答案

function red() {
    console.log('red');
}
function green() {
    console.log('green');
}
function yellow() {
    console.log('yellow');
}

var light = function (timmer, cb) {
    return new Promise(function (resolve, reject) {
        setTimeout(function () {
            cb();
            resolve();
        }, timmer);
    });
};

var step = function () {
    Promise.resolve().then(function () {
        return light(3000, red);
    }).then(function () {
        return light(2000, green);
    }).then(function () {
        return light(1000, yellow);
    }).then(function () {
        step();
    });
}

step();

14. 以下代码最后输出什么?

const first = () => (new Promise((resolve, reject) => {
    console.log(3);
    let p = new Promise((resolve, reject) => {
        console.log(7);
        setTimeout(() => {
            console.log(5);
            resolve(6);
        }, 0)
        resolve(1);
    });
    resolve(2);
    p.then((arg) => {
        console.log(arg);
    });

}));

first().then((arg) => {
    console.log(arg);
});
console.log(4);

// 答案
3
7
4
1
2
5

解析

这道题就其实和 Promise 的关系不太大,主要是需要理解 JS执行机制,才能很好的解决这道题。

第一轮事件循环

先执行宏任务,主script ,new Promise立即执行,输出【3】,

执行 p 这个new Promise 操作,输出【7】,

发现 setTimeout,将回调放入下一轮任务队列(Event Queue),

p 的 then,姑且叫做 then1,放入微任务队列,

发现 first 的 then,叫 then2,放入微任务队列。

执行console.log(4),输出【4】,宏任务执行结束。

再执行微任务,执行 then1,输出【1】,

执行 then2,输出【2】。

到此为止,第一轮事件循环结束。开始执行第二轮。

第二轮事件循环

先执行宏任务里面的,也就是 setTimeout 的回调,输出【5】。 resolve(6) 不会生效,因为 p 这个 Promise 的状态一旦改变就不会在改变了。

15、js 统计一个字符串出现频率最高的字母/数字

let str = 'asdfghjklaqwertyuiopiaia';
const strChar = str => {
    let string = [...str],
        maxValue = '',
        obj = {},
        max = 0;
    string.forEach(value => {
        obj[value] = obj[value] == undefined ? 1 : obj[value] + 1
        if (obj[value] > max) {
            max = obj[value]
            maxValue = value
        }
    })
return maxValue;
}
console.log(strChar(str))    // a

16、不借助临时变量,进行两个整数的交换

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

下面代码输出什么结果?

function *printLog(array) {
    let nextIndex = 0;
    while(nextIndex < array.length) {
        yield array[nextIndex++]
    }
    return 5;
}

let gen = printLog([3, 4])
console.log(gen.next())
console.log(gen.next())
console.log(gen.next())
console.log(gen.next())

//答案:
{value: 3, done: false}
{value: 4, done: false}
{value: 5, done: true}
{value: undefined, done: true}

下面代码中什么情况下会打印1?

var a = ?
if (a == 1 && a == 2 && a== 3) {
    console.log(1)
}
var a = {
    i:1,
    toString() {
        return a.i++;
    }
}

下面代码运行的结果?

async function async1() {
    console.log('async1 start')
    await async2();
    console.log('async1 end')
}

async function async2() {
    console.log('async2')
}
console.log('script start')
setTimeout(function () {
    console.log('setTimeout')
}, 0)

async1();
new Promise(function (resolve) {
    console.log('promise1')
    resolve();
}).then(function () {
    console.log('promise2')
})
console.log('script end')

//答案:
script start
async1 start
async2
promise1
script end
async1 end
promise2
setTimeout

实现一个sleep函数

比如sleep(1000)意味着等待1000毫秒

const sleep = time => {
    return new Promise((resolve, reject) => {
        setTimeout(resolve, time)
    })
}
sleep(1000).then(() => {
    console.log(1)
})

如何把一个字符串的大小写取反(大写变小写,小写变大写),例如'AbC' 变成 'aBc'?

let str = 'aBc';
str = str.replace(/[a-zA-Z]/g, content => {
    return content.toUpperCase() === content ? content.toLowerCase() : content.toUpperCase();
})

请自定义一个函数,实现字符串的反转

function reverse(str) {
    return str.split('').reverse().join('');
}

请写出三种对象合并的方法

let arr1 = {
    a:1,
    b:2,
    c:3
};
let arr2 = {
    d:4,
    e:5,
    a:6
};
//方法一拓展运算符,后面相同的属性覆盖前一个
let arr3={...arr1,...arr2};

//方法二Object.assign(),后面相同的属性覆盖前一个
let arr3=Object.assign(arr1,arr2);

//方法三Object.keys(),相同的属性只取第一个
Object.keys(arr1).forEach(key => {
    arr2[key] = arr1[key]
})

数组去重的方法

1.ES6 的 Set

let arr = [1,1,2,3,4,5,5,6]
let arr2 = [...new Set(arr)] // 两种写法均可:Array.from(new Set(arr))

2.reduce()

let arr = [1,1,2,3,4,5,5,6]
let arr2 = arr.reduce(function(ar,cur) {
  if(!ar.includes(cur)) {
    ar.push(cur)
  }

  return ar
},[])

3.filter()

// 这种方法会有一个问题:[1,'1']会被当做相同元素,最终输入[1]
let arr = [1,1,2,3,4,5,5,6]
let arr2 = arr.filter(function(item,index) {
  // indexOf() 方法可返回某个指定的 字符串值 在字符串中首次出现的位置
  return arr.indexOf(item) === index
})

手写一个递归方法求斐波那契数列的第N项的值

function fn(n){
    if(n==1 || n==2){
        return 1;
    }else {
        return fn(n-1)+fn(n-2);
    }
}

将数组扁平化并去除其中重复数据,最终得到一个升序且不重复的数组

Array.from(new Set(arr.flat(Infinity))).sort((a,b)=>{ return a-b})

实现 add(1)(2)(3)

考点:函数柯里化

function add (...args) {
    //求和
    return args.reduce((a, b) => a + b)
}

function currying (fn) {
    let args = []
    return function temp (...newArgs) {
        if (newArgs.length) {
            args = [
                ...args,
                ...newArgs
            ]
            return temp
        } else {
            let val = fn.apply(this, args)
            args = [] //保证再次调用时清空
            return val
        }
    }
}

let addCurry = currying(add)
console.log(addCurry(1)(2)(3)(4, 5)())  //15
console.log(addCurry(1)(2)(3, 4, 5)())  //15
console.log(addCurry(1)(2, 3, 4, 5)())  //15