前端面试核心技术栈 - 编程题(八)

八、编程题

1 写⼀个通用的事件侦听器函数

// event(事件)工具集,来源:github.com/markyun
markyun.Event = {
    // 视能力分别使用dom0| |dom2 | |IE方式 来绑定事件
    // 参数: 操作的元素 ,事件名称 ,事件处理程序
    addEvent : function(element, type, handler) {
        if (element.addEventListener) {
            //事件类型 、需要执行的函数 、是否捕捉
            element.addEventListener(type, handler, false);
        } else if (element.attachEvent) {
            element.attachEvent( 'on' + type, function() {
                handler.call(element);
            });
        } else {
        element [ 'on' + type] = handler;
    }
},

// 移除事件
removeEvent : function(element, type, handler) {
    if (element.removeEventListener) {
        element.removeEventListener(type, handler, false);
    } else if (element.datachEvent) {
        element.detachEvent( 'on' + type, handler);
    } else {
        element [ 'on' + type] = null;
    }
},

// 阻止事件 (主要是事件冒泡, 因为IE不支持事件捕获)
stopPropagation : function(ev) {
    if (ev.stopPropagation) {
        ev.stopPropagation();
    } else {
        ev.cancelBubble = true;
    }
},

// 取消事件的默认行为
preventDefault : function(event) {
    if (event.preventDefault) {
        event.preventDefault();
    } else {
        event.returnValue = false;
    }
},

// 获取事件目标
getTarget : function(event) {
    return event.target || event.srcElement;
}

2 如何判断⼀个对象是否为数组

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

3 冒泡排序

每次比较相邻的两个数, 如果后⼀个比前⼀个小,换位置
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 - i - 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));

4 快速排序

  • 采用二分法, 取出中间数,数组每次和中间数比较,小的放到左边,大的放到右边
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));

5 编写⼀个方法 求⼀个字符串的字节长度

假设:⼀个英文字符占用⼀个字节,⼀个中文字符占用两个字节
function GetBytes( str) {
    var len = str.length;
    var bytes = len;
    for(var i=0; i<len; i++){
        if (str.charCodeAt(i) > 255) bytes++;
    }
    return bytes;
}
alert(GetBytes("你好,as"));

6 bind的用法, 以及如何实现bind的函数和需要注意的点

  • bind 的作用与 call apply 相同, 区别是 call apply 是立即调用函数, 而 bind 是返回了⼀个函数, 需要调用的时候再执行 。 ⼀个简单的 bind 函数实现如下
Function.prototype.bind = function(ctx) {
    var fn = this;
    return function() {
        fn.apply(ctx, arguments);
    };
};

7 实现⼀个函数clone

可以对 JavaScript 中的5种主要的数据类型,包括 Number String Object Array Boolean ) 进行值复
  • 考察点1:对于基本数据类型和引用数据类型在内存中存放的是值还是指针这⼀区别是否清 楚
  • 考察点2:是否知道如何判断⼀个变量是什么类型的
  • 考察点3:递归算法的设计
// 方法⼀:
Object.prototype.clone = function(){
    var o = this.constructor === Array ? [] : {};
    for(var e in this){
        o[e] = typeof this [e] === "object" ? this [e].clone() : th
    }
    return o;
}

//方法二:
/**
* 克隆⼀个对象
* @param Obj
* @returns
*/
function clone(Obj) {
    var buf;
    if (Obj instanceof Array) {
        buf = [];
        //创建⼀个空的数组
        var i = Obj.length;
        while (i--) {
            buf [i] = clone(Obj [i]);
        }
        return buf;
    }else if (Obj instanceof Object){
        buf = {};
        //创建⼀个空对象
        for (var k in Obj) {
            //为这个对象添加新的属性
            buf [k] = clone(Obj [k]);
        }
        return buf;
    }else{
        //普通变量直接赋值
        return Obj;
    }
}

8 下面这个ul, 如何点击每⼀列的时候alert其index

考察闭包
<ul id=”test”>
    <li>这是第⼀条</li>
    <li>这是第二条</li>
    <li>这是第三条</li>
</ul>

// 方法⼀:
var lis=document.getElementById( '2223').getElementsByTagName( 'li');
for(var i=0;i<3;i++) {
    lis [i].index=i;
    lis [i].onclick=function(){
        alert(this.index);
    }
}

//方法二:
var lis=document.getElementById( '2223').getElementsByTagName( 'li');
for(var i=0;i<3;i++){
    lis [i].index=i;
    lis [i].onclick=(function(a){
        return function() {
            alert(a);
        }
    })(i);
}

9 定义⼀个log方法,让它可以代理console.log的方法

可行的方法⼀:
function log(msg) {
    console.log(msg);
}
log("hello world!") // hello world!
如果要传入多个参数呢?显然上面的方法不能满足要求,所以更好的方法是
function log( ) {
    console.log.apply(console, arguments);
};

10 输出今天的日期

YYYY-MM-DD 的方式, 比如今天是2014年9月26日,则输出2014-09-26
var d = new Date();
// 获取年, getFullYear()返回4位的数字
var year = d.getFullYear();
// 获取月, 月份比较特殊, 0是1月, 11是12月
var month = d.getMonth() + 1;
// 变成两位
month = month < 10 ? '0' + month : month;
// 获取日
var day = d.getDate();
day = day < 10 ? '0' + day : day;
alert(year + '- ' + month + '- ' + day);

11 用js实现随机选取10– 100之间的10个数字,存入⼀个数组, 并排序

var iArray = [];
funtion getRandom(istart, iend){
    var iChoice = istart - iend +1;
    return Math.floor(Math.random() * iChoice + istart;
}
for(var i=0; i<10; i++){
    iArray.push(getRandom(10,100));
}
iArray.sort();

12 写⼀段JS程序提取URL中的各个GET参数

有这样⼀个 URL http://item.taobao.com/item.htm? a=1&b=2&c=&d=xxx&e ,请写⼀段JS程序提取URL中的各个GET参数(参数名参数个数不确定),将其按 key-value 形式返回到⼀个 json 结构中, 如 {a:'1', b:'2', c:'', d:'xxx', e:undefined}
function serilizeUrl(url) {
    var result = {};
    url = url.split("?") [1];
    var map = url.split("&");
    for(var i = 0, len = map.length; i < len; i++) {
        result[map[i].split("=") [0]] = map[i].split("=") [1];
    }
    return result;
}

13 写⼀个 function , 清除字符串前后的空格

使用自带接口 trim() ,考虑兼容性:
if ( !String.prototype.trim) {
    String.prototype.trim = function() {
        return this.replace(/^\s+/, "").replace(/\s+$/,"");
    }
}
// test the function
var str = " \t\n test string ".trim();
alert(str == "test string"); // alerts "true"

14 实现每隔⼀秒钟输出1,2,3...数字

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

15 实现⼀个函数, 判断输入是不是回文字符串

function run( input) {
    if (typeof input !== 'string') return false;
    return input.split( '').reverse().join( '') === input;
}

16 、数组扁平化处理

实现⼀个 flatten 方法,使得输入⼀个数组,该数组里面的元素也可以是数
组,该方法会输出⼀个扁平化的数组
function flatten(arr){
    return arr.reduce(function(prev,item){
        return prev.concat(Array.isArray(item)?flatten(item):item);
    }, []);
}

17、Tree转Array处理

方法一:递归遍历

function treeToArray(tree) {
  let result = [];
  function traverse(node) {
    if (!node) return;
    result.push(node.value);
    if (node.children) {
      for (let child of node.children) {
        traverse(child);
      }
    }
  }
  traverse(tree);
  return result;
}

方法二:使用队列进行广度优先遍历

function treeToArray(tree) {
  let result = [];
  let queue = [tree];
  while (queue.length) {
    let node = queue.shift();
    if (!node) continue;
    result.push(node.value);
    if (node.children) {
      for (let child of node.children) {
        queue.push(child);
      }
    }
  }
  return result;
}

方法三:使用栈进行深度优先遍历

function treeToArray(tree) {
  let result = [];
  let stack = [tree];
  while (stack.length) {
    let node = stack.pop();
    if (!node) continue;
    result.push(node.value);
    if (node.children) {
      for (let child of node.children) {
        stack.push(child);
      }
    }
  }
  return result;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

LiuPing_Xie

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值