美文网首页
js常考算法

js常考算法

作者: 罂粟1995 | 来源:发表于2020-03-16 22:22 被阅读0次
1. 一个数组,有2n个元素,其中有个数字重复出现了n次。其他n个数字都是不相同的。 空间复杂度O(1)。时间复杂度O(n)。找出这个重复的数字。
 function findNumber(array) {
        var findArr = [];
        for(var i=0;i<array.length;i++){
           if(findArr.indexOf(array[i]) != -1){
               return array[i];
           }
           if(findArr.indexOf(array[array.length-1-i]) != -1){
               return array[array.length-1-i];
           }
           array.push(array[i]);
           array.push(array[array.length-1-i]);
        }
   }
2.如何判断一个链表是否存在环?时间复杂度为n,空间复杂度为1。

快慢指针法:

function judge(list) {
        //创建快慢指针
        var fast = list.next.next;
        var slow = list.next;
        while(list) {
            if(fast === slow){
                return true;
            }

           fast = fast.next.next;
           slow = show.next;
        }
   }
3. 如何判断一个数组符合前序遍历/后续遍历?

后续遍历:数组中最后一个节点是根节点;除根节点的那部分,左半部分是左子树,右半部分是右子树。左子树小于根节点,右子树大于根节点。
思路:找到左右子树的分割点,如果当一个数小于后一个数,则后一个数的后半部分为右子树,如果右子树有小于根节点的则不符合。
代码:

function  judge(array) {
    if(array.length <= 0){
        return false;
    }
    var len = array.length;
    var root = array[len-1];//找到根节点
    var i;
    for(i=0;i<len-1;i++){ 
        //从最开始,找到第一个 大于根节点的数时,跳出循环
        if(root < array[i]) 
             break;
    }

    var j;
    for(j=i;j<len-1;j++){
        //开始测试后半部分数据
        if(root > array[j])
               return false;// 后半部分若小于 根节点  ,则该数组不是后序遍历
    }
    // 设置左半部分的数据标志
    var left = true;
    if(i>0){
        // 采用递归 ,测试前半部分数组是否满足后序遍历
        //数组的 slice 方法 ,不会改变原数组 ,会返回一个新数组 ,会从原数组中截取 从 0 到 i的新数组 
        left = judge(array.slice(0,i))  
    }
    var right = true;
    if(i<len - 1){
        // 采用递归 ,测试后半部分数组是否满足后序遍历
       right = judge((array.slice(i,len-1));
    }
    // 只有 leftFlag 和 rightFlag 都为 true 时,说明 该数组是 后序遍历
   return left && right;
}

同理前序遍历,只需把根节点是最后一个元素改成根节点是第一个元素即可:

function  judge(array)
{
    if(array.length <= 0){
        return false;
    }
    var len = array.length;
    var root = array[0];//找到根节点
    var i;
    for(i=1;i<len;i++){ 
        //找到第一个大于根节点的数时,跳出循环
        if(root < array[i]) 
             break;
    }

    var j;
    for(j=i;j<len;j++){
        //开始测试后半部分数据
        if(root > array[j])
               return false;// 后半部分若小于 根节点  ,则该数组不是前序遍历
    }
    // 设置左半部分的数据标志
    var left = true;
    if(i>1){
        // 采用递归 ,测试前半部分数组是否满足前序遍历
        //数组的 slice 方法 ,不会改变原数组 ,会返回一个新数组 ,会从原数组中截取 从 0 到 i的新数组 
        left = judge(array.slice(1,i))  
    }
    var right = true;
    if(i<len){
        // 采用递归 ,测试后半部分数组是否满足前序遍历
       right = judge((array.slice(i,len));
    }
    // 只有 leftFlag 和 rightFlag 都为 true 时,说明 该数组是前序遍历
   return left && right;
}
4. 实现一个二分查找

二分法查找,也称折半查找,是一种在有序数组中查找特定元素的搜索算法。
思路:从数组中间元素查起,如果中间元素就是目标元素,返回;如果不是,目标元素小于中间元素,往左找,大于中间元素,往右找。
代码:
递归法:

 function  judge(array,key) {
            if(array.length <= 0){
                  return -1;
            }
            var mid = parseInt((array.length) / 2);
            if(array[mid] === key){
                return mid;
            }else if (array[mid] > key){
                var leftArray = array.slice(0,array[min-1]);
                return judge(leftArray, key);
            }else if (arr[mid] < key){
                var rightArray = array.slice(array[min+1],array.length);
                return judge(rightArray, key);
            }
}
5.二叉树前序遍历
//定义节点
function Node(element,left,right){
        this.element=element;
        this.left=left;
        this.right=right;
        this.show=function () {
            return this.element;
       }
}

//定义树结构
 function BST() {
        this.root = null;
        //插入节点(insert node)
        this.insert=function (element) {
            var n = new Node(element, null, null);
            if (this.root == null) {
                 this.root = n;
            }else {
                 var current = this.root;
                 var parent;
                 while (true) {
                     parent = current;
                    if (element < current.element) {
                         //待插入节点保存的数据小于当前节点,则设新的当前节点为原节点的左节点
                         current = current.left;

                         //如果当前节点的左节点为null,就将新的节点插入这个位置,退出循环;反之,继续执行下一次while循环。
                        if (current == null) {
                               parent.left = n;
                               break;
                        }
                    }else {
                        //待插入节点保存的数据小于当前节点,则设新的当前节点为原节点的右节点
                        current = current.right;
                        if (current == null) {
                             parent.right = n;
                             break;
                        }
                    }
             }
        }

        //前序遍历
        this.preSort = function (node) {
            if(node != null){
                console.log(node.show());
                this.preSort(node.left);
                this.preSort(node.right);
            }
        }
        //中序遍历
       this.inSort = function (node) {
            if(node != null){
                this.inSort(node.left);
                console.log(node.show());
                this.inSort(node.right);
            }
        }
       //后序遍历
       this.nextSort = function (node) {
            if(node != null){
                this.nextSort(node.left);
                this.nextSort(node.right);
                console.log(node.show());
            }
        }
6.实现大数相加求和
  • 什么是大数?
    在32位系统下最大只能表示2^31-1,也就是2147483647,想要表示更大的数可以使用字符串。
  • 思路:逐位相加并进位。
function sumStrings(a,b) {  
    // 操作数类型判断, 要求为字符串
    if (typeof a !== 'string' || typeof b !== 'string') {
          return false;
    }
    //通过补零让a和b对齐  
    //若a比b短,则对a补零  
    while(a.length < b.length){  
        a = "0" + a;  
    }  
    //若b比a短,则对b补零  
    while(b.length < a.length){  
        b = "0" + b;  
    }  
    //是否有进位  
    var addOne = 0;  
    //结果数组  
    var result = [];  
    //从个位开始相加  
    for(var i=a.length-1;i>=0;i--){  
        var c1 = a.charAt(i) - 0;  
        var c2 = b.charAt(i) - 0;  
        var sum = c1 + c2 + addOne;  
        //若数字相加大于9,则进位  
        if(sum > 9){  
            result.unshift(sum - 10);  
            addOne = 1;  
        }  
        else{  
            result.unshift(sum);  
            addOne = 0;  
        }  
    }  

    //如果还有进位,最后要把进位进完  
    if(addOne){  
        result.unshift(addOne);  
    }  

    //移除第一位的"0"  
    if(result[0] == 0){  
        result.splice(0,1);  
    }  
    return result.join("");  
}  
7. 请实现如下函数,可以批量请求数据。所有的url地址都在urls参数中,同时可以通过max控制请求的并发度,当所有请求结束后,调用callback回调函数。请求直接用fetch就可以。
var urls = [1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20];
const limit = 5;

// 主函数
function sendRequest(urls, limit , callback) {
    function _send (urls) {
        return fetch(urls.shift())
            .finally(() => {
                if(urls.length) {
                    return _send(urls)
                }
            })
    }
    let asyncList = [];
    while(limit--) {
        asyncList.push(_send(urls));
    }
    return Promise.all(asyncList).then(callback);
}

sendRequest(urls, limit, function() {
    console.log('finish')
});

相关文章

  • js常考算法

    1. 一个数组,有2n个元素,其中有个数字重复出现了n次。其他n个数字都是不相同的。 空间复杂度O(1)。时间复...

  • js开发 面试常考基础算法题

    js 开发面试常考基础算法题 1 不需要借助第三个临时变量,实现两个变量的交换 2 确保字符串的每个单词首字母都大...

  • 面试常考的算法

    有感于最近数据结构的难度越来越大,觉得先面试应付要紧,于是这边先写了几个关于面试要用的算法 第一个是判断是不是质数...

  • python面试算法

    python内置数据结构算法常考 常用的内置算法和数据结构 sorted排序函数 dict/list/set/tu...

  • 字节跳动面试必问:海量算法高频面试题精编解析(深度好文,值得收藏

    常考面试算法题类型总结 结合2019春招和秋招真题,以下几类算法题最常考,汇总了一下: 一、暴力枚举 好多鱼! D...

  • 数据结构&算法

    常见排序算法小结如何判断一个单链表是否有环?程序员必须知道的10大基础实用算法及其讲解排序算法总结 校招常考算法J...

  • iOS面试常考算法(持续更新)

    1.字符串翻转 reservString具体实现如下 2.链表原地翻转 3.合并有序数组,尽可能快 4.查找一个字...

  • 面试常问的小算法总结

    前言 本文快速回顾了面试常考的算法,用作面试复习,事半功倍。 需要说明的是,由于算法的代码实现主要注重思路的清晰,...

  • 复制js变量问题(面试常考)

    js变量中存储了两种不同数据结构的值 基本类型值 和引用类型值。 基本类型值值的是简单的数据段,而引用类型值是指那...

  • 常考的数据结构与算法之算法

    本文记录一下我学习数据结构与算法中算法的知识点,使用的语言是C/C++语言 查找 二分查找又叫折半查找,要求线性表...

网友评论

      本文标题:js常考算法

      本文链接:https://www.haomeiwen.com/subject/zuiiehtx.html