/InterviewQuestions

some questions and answers

Primary LanguageJavaScript

InterviewQuestions

#2017更新

Q2017-8:

如何判断一个函数 是Promise?

    function isPromise(value){
        return value instanceof Promise;
    }

Q2017-7:

实现一个函数可以对不确定参数个数的情况下对参数求和 len=0;//0 len=1;//arguments[0];

    function sum(){
        var len =arguments.length;
        if(len===0) return 0;
        else{
            let result=0;
            for(let i=0;i<len;i++){
                result+=arguments[i];
            }
            return result;
        }
    }

利用sum函数实现自增函数

    var addOne=sum.bind(null,1);
    console.log(sum());//1
    console.log(sum(1));//2

bind()会继承sum的属性并生成一个新的函数,可以给新的函数添加参数

Q2017-6:

    tick(); //1;
    tick(); //2;
    tick(); //3;
    ...
    ...
    不使用全局变量实现这种自增
    function tick(){
        var num=0;
        return function(){
            return num++;
        }
    }

    var children = tick();
    tick();
    tick();

Q2017-5:

    function a(){
        for(var i=0;i<10;i++){
            setTimeout(()=>{
                console.log(i);
            },i*1000)
        }
    }

很经典的闭包问题吧,因为setTimeout延迟执行,此时的i已经循环完毕,变成了9; 所以打印出来的都是9; 如果想顺序打印出来,有两种方法:

  1. 使用let代替var 即for(let i=0;i<10;i++)因为let会绑定作用域;
  2. 使用闭包 即
    for(var i=0;i<10;i++){
        setTimeout(()=>{
            return ()=>{
                console.log(i);
            }
        },i*1000);
    }

Q2017-4:

    var x='abc';
    function abc(){
        var y=x;
        var x='def';
        console.log(y);//undefined
    }

理解函数生成的时候会在函数内第一行先声明所有的局部变量,即

    function abc(){
        var x;
        var y;
        y=x;
        x='def';
        console.log(y);//undefined
    }

Q2017-3:MySql的数据库引擎分哪几种,区别是什么?

ANSWER:

1.Innodb引擎

提供对事务的支持,提供了行级锁和外键约束,不支持全文TEXT字段的索引,没有保存表的行数

2.MyIASM引擎

没有提供对事务的支持,不支持行级锁和外键,INSERT和UPDATE操作时需要锁全表,存储了表的行数

主要区别:

  • MyIASM是非事务安全的,而InnoDB是事务安全的
  • MyIASM锁的粒度是表级的,而InnoDB支持行级锁
  • MyIASM支持全文类型索引,而InnoDB不支持全文索引
  • MyIASM相对简单,效率上要优于InnoDB,小型应用可以考虑使用MyIASM
  • MyIASM表保存成文件形式,跨平台使用更加方便

Q2017-2:Redis常用命令

ANSWER:

  1. keys * 查看所有key
  2. set key value 设置key的值,若存在则覆盖
  3. setnx key value :set if not exists,若存在则不操作
  4. MSET key1 value1 key2 value2...设置一组key-value
  5. get key 获取key对应的值
  6. incr key 自增+1;decr key 自减-1
  7. expire key seconds 设置缓存存在时间,过时间删除变量
  8. hset key "name" "kevin" 设置对象属性值
  9. hget key "name" 获取对象属性值

参考:

  1. redis命令行文档
  2. redis官网
  3. redis学习博客

Q2017-1: js实现青蛙跳台阶算法,青蛙可以跳1阶或2阶,跳到N层有多少种跳法?

ANSWER:

其实考的是斐波那契数列算法,结果如下:

    'use strict';//使用严格模式
    function Step(FloorN){
        let N = Number(FloorN);
        if(N<=0){
            console.error("INPUT UNEXPECTED!");
            return 0;
        }
        if(!N) return 0;
        if(N<=2) return N;
        return Step(N-1)+Step(N-2);
    }

上述递归的方法效率很低,时间复杂度为O(2^N),会有很多重复计算,下面使用非递归方法重构

//非递归

    'use strict';
    function Step2(FloorN){
        let array = [];
        let N = Number(FloorN);
        if(N<=0){
            console.error("INPUT UNEXPECTED!");
            return 0;
        }
        if(N<=2) return N;
        array[0]=1;
        array[1]=2;
        for(let i=2;i<N;i++){
            array[i]=array[i-1]+array[i-2];
        }
        return array[N-1];
    }

    //不使用数组
    function Step3(FloorN){
        let N = Number(FloorN);
        if(N<=0){
            console.error("INPUT UNEXPECTED!");
            return 0;
        }
        if(N<=2) return N;
        let first=1;
        let second=2;
        let fn=0;
        for(let i=2;i<N;i++){
            fn=first+second;
            first=second;
            second=fn;
        }
        return fn;
    }

具体代码在src/frog.js中,并有在我机器上测试的效率对比

######----------------------------我是分割线------------------------------- #2016更新 some questions and answers


Q1:JS 继承有哪几种方式?

ANSWER:

首先有两个构造函数Animal,Dog

function Animal(){
  this.species = 'animal';
}
      
function Dog(name,color){
  this.name = name;
  this.color = color;
}

我们要让Dog继承Animal的属性。

ANSWER:

1.构造函数绑定

使用apply或者call方法

function Dog(name,color){
   Animal.apply(this,arguments);
  //Animal.call(this,name,color);
  this.name = name;
  this.color = color;
}
    
var lucky = new Dog('lucky','brown');
console.log(lucky.species; // animal'
2.prototype模式
Dog.prototype = new Animal();
Dog.prototype.constructor = Dog;
var lucky = new Dog('lucky','Brown');
console.log(lucky.species);

先将Dog的prototype对象指向一个Animal的实例,然后将prototype.constructor指向构造函数Dog,这样就实现了继承

3.直接继承prototype
function Animal(){}
Animal.prototype.species = 'animal';

function Dog(name,color){
  this.name = name;
  this.color = color;
};

Dog.prototype = Animal.prototype;
Dog.prototype.constructor = Dog;
var lucky = new Dog('lucky','brown');
console.log(lucky.species);

这样做的好处是效率高,速度快,省内存,但是Dog.prototype和Animal.prototype现在指向了同一个对象,修改Dog.prototype会直接修改Animal.prototype。

4.利用空对象作为中介
var F = function(){};
F.prototype = Animal.prototype;
Dog.prototype = new F();
Dog.prototype.constructor = Dog;

console.log(Animal.prototype.constructor); //Animal
var lucky = new Dog('lucky','brown');
console.log(lucky.species); // animal
5.拷贝继承
function extend(Child,Parent){
  var p = Parent.prototype;
  var c = Child.prototype;
  for(var i in p){
    c[i] = p[i];
  }
  c.uber = p;//为子对象留有可以访问父对象的接口
}

function Dog(name,color){}

extend(Dog,Animal);
var lucky = new Dog('lucky','brown');
console.log(lucky.species); //animal

Q2:JS 严格模式跟普通模式的区别?

ANSWER:

1.在js文件中使用'use strict'进入严格模式

  • 消除Javascript语法的一些不合理、不严谨之处,减少一些怪异行为;
  • 消除代码运行的一些不安全之处,保证代码运行的安全;
  • 提高编译器效率,增加运行速度;
  • 为未来新版本的Javascript做好铺垫。
2.会将拼写错误转成异常
'use strict';

var maxNumber = 100;
maxNumbre = 10;//ReferenceError
console.log(maxNumber);
3.全局变量显式声明

正常模式中,一个变量未声明直接赋值,默认是全局变量,严格模式中禁止这种用法,全局变量必须显示声明

'use strict';
v1 = 1;//error
4.禁用with,eval
5.禁止删除声明变量
'use strict';

var x;
delete x; //语法错误

Q3:V8垃圾回收机制

ANSWER:

V8使用是垃圾回收器(GC)进行回收, 具体信息可以参考下列文章:

  1. 阿里云团队博客
  2. Segmentfault

Q4:web后端缓存策略有哪些?

ANSWER:

1.利用redis(node-redis)或 memcached

在服务器与数据库之间设置redis数据库作为缓存介质,在需要缓存的请求到达时,先检查redis数据库是否有,使用

var redis = require('redis').createClient();
redis.get('key',function (err,reply){
      //logic code
});

进行判断,如果不存在,则从对应数据库获取数据,然后将数据写入redis数据库,并设置过期时间。

var redis = require('redis').createClient();
redis.set('key','value');
redis.expire('key',1000);//ms
2.max-age和cache-control

主要针对静态资源的缓存

response.setHeader('Cache-Control','public,max-age=3600');
3.etag

从req.headers里取'if-none-match',如果存在且hash值一致,返回304

var hashStr = "A hash string.";
var hash = require("crypto").createHash('sha1').update(hashStr).digest('base64');
require("http").createServer(function(req, res){  
    if(req.headers['if-none-match'] == hash){ 
        res.writeHead(304);
        res.end(); 
        return; 
    } 
    res.writeHead(200, { "Etag": hash }) 
    res.write(hashStr); res.end();
}).listen(9999);

这样可以在输出没有变化的时候过滤请求,减轻服务器压力。

4.CDN缓存

将图片等数据放在CDN缓存,可以大大减轻自己服务器压力,可以参考这个模块express-cdn

参考资料:

  1. Web开发后端缓存思路
  2. Etag缓存
  3. 缓存策略

Q5:TCP与UDP的区别?

A:

  1. TCP是面向链接的,虽然说网络的不安全不稳定特性决定了多少次握手都不能保证连接的可靠性,但TCP的三次握手在最低限度上(实际上也很大程度上保证了)保证了连接的可靠性; 而UDP不是面向连接的,UDP传送数据前并不与对方建立连接,对接收到的数据也不发送确认信号,发送端不知道数据是否会正确接收,当然也不用重发,所以说UDP是无连接的、不可靠的一种数据传输协议。
  2. 也正由于1所说的特点,使得UDP的开销更小数据传输速率更高,因为不必进行收发数据的确认,所以UDP的实时性更好。

Q6:为什么选择mocha这种测试框架?

ANSWER:

  • 与node完美结合,可以在node端和浏览器端测试
  • 可以使用不同的断言库(chai,should等)
  • 异步方法测试简单
  • 支持before,after等hook
  1. mocha git
  2. mocha测试

Q7:TDD/BDD分别是什么?

ANSWER:

TDD(测试驱动开发)

即先写测试用例,再一一实现功能

BDD(行为驱动开发)

先写功能,再对功能进行测试,更贴近人类思维方式

Q8:闭包

ANSWER:

我理解的闭包就是外部函数获取内部函数私有属性的一种方法,简单的用下面的例子解释一下:

function myHouse(){
  var __myname = 'Kevin';
  var getMyname = function(){
      return __myname;
  };
  return getMyname;
}

var getName = myHouse();
console.log(getName());//Kevin

即通过在myHouse()中返回getMyname函数,getMyname函数中返回私有变量__myname,从而获取私有变量,同时保护私有变量不会直接暴露给外部,以免改变私有变量值。

Q9:node.js跨域问题

ANSWER:

var express = require('express');
var app = express();//设置跨域访问
app.all('*', function(req, res, next) { 
    res.header("Access-Control-Allow-Origin", "*"); //核心
    res.header("Access-Control-Allow-Headers", "X-Requested-With"); //核心
    res.header("Access-Control-Allow-Methods","PUT,POST,GET,DELETE,OPTIONS"); //核心
    res.header("X-Powered-By",' 3.2.1') ;
    res.header("Content-Type", "application/json;charset=utf-8"); next();
});
app.get('/auth/:id/:password', function(req, res) { 
    res.send({
        id:req.params.id, 
        name: req.params.password
    });
});
app.listen(3000);console.log('Listening on port 3000...');

还可以参考这个模块:cors 参考:推酷

Q10:ES6新特性:

ANSWER:

这个写的很不错,推荐阅读 阮一峰的日志

Q11: 使用JS写排序算法

ANSWER:

  • 冒泡排序:
function bubbleSort(arr){
    var flag = true;
    var len = arr.length;
    for(var i = 0; i < len-1;i++){
        flag = true;
        for(var j = 0; j < len -1 - i; j++){
            if(arr[j] > arr[j+1]){
                var temp = arr[j];
                arr[j] = arr[j+1];
                arr[j+1] = temp;
                flag = false;
            }
        }
        console.log(flag+':'+i+':'+j);
        if(flag){
            break;
        }
    }
    return arr;
}
  • 快速排序:
function quickSort(arr){
    
    if(arr.length <= 1){
        return arr;
    }
    var len = arr.length;

    var midLen = Math.floor(len/2);
    var midValue = arr.splice(midLen,1);

    var leftArr = [];
    var rightArr = [];

    for(var i = 0; i < arr.length;i++){
        if(arr[i] > midValue){
            rightArr.push(arr[i]);
        }else{
            leftArr.push(arr[i]);
        }
    }

    return quickSort(leftArr).concat(midValue,quickSort(rightArr));
}