每天一道算法题(第一期)

如果每天做一道算法题,那是不是每天都在进步?

前言

这个活动是从2019年7月中旬开始的,人数不算多,也就几个好友和好友的好友,过程中也会有人因为工作的缘故或其他原因放弃,或许未来还会有人离开。

活动的主要形式就是在leetcode刷题,每个工作日一道题,每周做总结,目前已经是第十二期,接下来我会把每期的题做一个总结,由于我是侧重javascript,所以活动中的每道题都是以js语言来实现解题方法。

活动的规则比较严谨,群里每天早上10点之前发题,晚上10点审核,审核有管理员专门审核,称为打卡,没有打卡的人,需要发红包给管理员作为每天统计费用。

活动的目的就是培养算法思维,了解常见的算法,比如分治算法、贪心算法、动态优化等等。

微信公众号惊天码盗同步第一期。

两个数组的交集 II

给定两个数组,编写一个函数来计算它们的交集。

  • 示例 1:
    输入: nums1 = [1,2,2,1], nums2 = [2,2]
    输出: [2,2]
    
  • 示例 2:
    输入: nums1 = [4,9,5], nums2 = [9,4,9,8,4]
    输出: [4,9]
    
题解:
  • 替换法
    循环nums1,在循环中判断nums2中是否存在当前元素,如果存在就push到新的数组,然后在通过splice替换到当前元素,最后排序;
    执行用时:88ms;内存消耗:35.5MB;
    var intersect = function(nums1, nums2) {
        let arr = [];
        nums1.forEach((cur, i) => {
          if (nums2.indexOf(cur) > -1) {
            arr.push(cur)
            nums2.splice(nums2.indexOf(cur), 1, null)
          }
        })
        return arr.sort()
    }; 
    
  • 双指针法
    两个数组排序,设定两个为0的指针,比较两个指针的元素是否相等,如果相等,元素push到返回值里,两个指针同时往前,如果不相等,元素小的指针往前。
    执行用时:84ms;内存消耗:35MB;
    var intersect = function(nums1, nums2) {
        let p1 = 0
        let p2 = 0
        let res = []
        nums1 = nums1.sort((a, b) => a - b)
        nums2 = nums2.sort((a, b) => a - b)
        while(p1 < nums1.length && p2 < nums2.length) {
            if(nums1[p1] == nums2[p2]) {
                res.push(nums1[p1])
                p1++
                p2++
            } else if(nums1[p1] < nums2[p2]) {
                p1++
            } else {
                p2++
            }
        }
        return res
    }
    

1比特与2比特字符

有两种特殊字符。第一种字符可以用一比特0来表示。第二种字符可以用两比特(10 或 11)来表示。

现给一个由若干比特组成的字符串。问最后一个字符是否必定为一个一比特字符。给定的字符串总是由0结束。

  • 示例 1:
    输入: bits = [1, 0, 0]
    输出: True
    解释: 唯一的编码方式是一个两比特字符和一个一比特字符。所以最后一个字符是一比特字符。
    
  • 示例 2:
    输入: bits = [1, 1, 1, 0]
    输出: False
    解释: 唯一的编码方式是两比特字符和两比特字符。所以最后一个字符不是一比特字符。
    
题解:
  • 正则
    这是我见过最暴力直接的方式,由群内小伙伴提出的解题方式。
    执行用时:84ms;内存消耗:34.4MB;(我感觉这个测试不准,每次提交的用时和消耗均不同)

     var isOneBitCharacter = function(bits) {
      return /^(10||11||0)*0$/.test(bits.join(""))
     };
    
  • 对象存储
    通过对象的方式;在循环中判断,如果遇到0、10或者11等情况添加到对象的属性上,这种方式可以有效的记录一共出现1比特和2比特出现的次数和顺序;最后判断最后一个比特是1比特还是2比特。
    执行用时:88ms;内存消耗:37.6MB;

    var isOneBitCharacter = function(bits) {
        let obj = {};
        let count = 1;
        bits.forEach((item, i) => {
          if (!obj[count]) obj[count] = [];
          if (obj[count].length >= 2) {count++; obj[count] = []};
          obj[count].push(item)
          if (obj[count][0] == 0 && i!=bits.length-1) count++;
        })
        return obj[count]&&obj[count][0] === 0 ? true : false
    };
      //对象结构如下
      {
        1:[0],
        2:[1,1],
        3:[1,0],
        ...
      } 
    
  • 循环跳跃法
    循环,遇1就++,如果++后跳出循环,就返回false,如果还在循环内,当循环到最后一个的时候返回true;
    执行用时:72ms;内存消耗:34.2MB;

    var isOneBitCharacter = function(bits) {
      let bitsSize = bits.length
      for (let i = 0; i < bitsSize; ++i) {
        if (i == bitsSize - 1) return true
        if (bits[i])++i;
      }
      return false;
    };
    

二进制求和

给定两个二进制字符串,返回他们的和(用二进制表示)。
输入为非空字符串且只包含数字 1 和 0。

  • 示例 1:
    输入: a = "11", b = "1"
    输出: "100"
    
  • 示例 2:
    输入: a = "1010", b = "1011"
    输出: "10101"
    
题析:

面对这道题,很多人立马想到的是2转10,然后转2,但是这会面临精度丢失的问题,如果解决了精度丢失的问题,那么也不失为一个好的解题方法。那么就只有一种方法那就是逐位相加法。

题解:
  • 补位法
    字符串转数组,然后翻转数组,取数组最长的length循环,在循环中数组短的自动补0,在循环中相加,判断相加为0或1或2的情况。然后push到新数组中,最后翻转,转成字符串。
    执行用时:108ms;内存消耗:35.6MB;(最笨的办法)
    var addBinary = function(a, b) {
      const alist=a.split('').reverse();
      const blist=b.split('').reverse();
      const arr=[];
      const len=alist.length>blist.length?alist.length:blist.length;
      for (let i = 0; i <= len-1; i++) {
          if (alist[i] != 0 && alist[i] != 1) {
            alist[i] = 0
          }
          if (blist[i] != 0 && blist[i] != 1) {
            blist[i] = 0
          }
          alist[i] = alist[i] - 0
          blist[i] = blist[i] - 0
    
          if (alist[i] + blist[i] > 1) {
            if (arr[i] == 0) {
              arr[i] = 1
            } else if (arr[i] == 1) {
              arr[i] = 1
              arr.push(1)
           } else {
              arr[i] = 0
              arr.push(1)
            }
          }
          if (alist[i] + blist[i] == 1) {
            if (arr[i]) {
              arr[i] = 0
              arr.push(1)
            } else {
              arr[i] = 1
            }
          }
          if (alist[i] + blist[i] == 0) {
            if (arr[i]) {
              arr[i] = 1
            } else {
              arr[i] = 0
            }
          }
        }
        return arr.reverse().join('')
    };
    
  • 进位拼接法
    与补位法类似,区别就是在进行计算时直接拼接字符串,会得到一个反向字符,需要最后再进行翻转;按照位置给结果字符赋值,最后如果有进位,则在前方进行字符串拼接添加进位。
    执行用时:112ms;内存消耗:35.9MB;(性能不好)
    var addBinary = function(a, b) {
      let ans = "";
      let ca = 0;
      for(let i = a.length - 1, j = b.length - 1;i >= 0 || j >= 0; i--, j--) {
          let sum = ca;
          sum += i >= 0 ? parseInt(a[i]) : 0;
          sum += j >= 0 ? parseInt(b[j]) : 0;
          ans += sum % 2;
          ca = Math.floor(sum / 2);
      }
      ans += ca == 1 ? ca : "";
      return ans.split('').reverse().join('');
    }
    

两数之和

给定一个整数数组 nums和一个目标值 target,请你在该数组中找出和为目标值的那两个整数,并返回他们的数组下标。

你可以假设每种输入只会对应一个答案。但是,你不能重复利用这个数组中同样的元素。

  • 示例:
    给定 nums = [2, 7, 11, 15], target = 9
    因为nums[0] + nums[1] = 2 + 7 = 9
    所以返回 [0, 1]
    
题解:
  • 双循环法
    两层循环,外层取第一元素,内层取相加元素;
    执行用时:164ms;内存消耗:34.6MB;
    var twoSum = function(nums, target) {
     for (let i = 0; i < nums.length; i++) {
      for (let j = i + 1; j < nums.length; j++) {
        if (nums[i] + nums[j] === target) {
          return [i, j]
         }
       }
     }
    };
    
  • 替换法
    复制原数组,循环原数组,防止重复,通过splice替换复制的数组,如果复制数组中存在相加元素,找到索引,即可得到结果数组。
    执行用时:304ms;内存消耗:36MB;
    var twoSum = function(nums, target) {
      let list = [...nums];
      let arr = []
      nums.forEach((item, i) => {
        list.splice(i, 1, null)
        const sum = target - item;
        if (list.includes(sum)) {
          arr = [i, list.indexOf(sum)].sort()
        }
      })
      return arr
    }
    
  • map
    通过新的数据类型Map来实现;
    执行用时:72ms;内存消耗:35.1MB;
    var twoSum = function(nums, target) {
      let map = new Map()
      for (let i = 0; i < nums.length; i++) {
         if (map.has(target - nums[i])) {
            return [map.get(target - nums[i]), i]
          }
          map.set(nums[i], i)
       }
    }
    

加一

给定一个由整数组成的非空数组所表示的非负整数,在该数的基础上加一。
最高位数字存放在数组的首位, 数组中每个元素只存储一个数字。
你可以假设除了整数 0 之外,这个整数不会以零开头。

  • 示例 1:
    输入: [1,2,3]
    输出: [1,2,4]
    解释: 输入数组表示数字 123。
    
  • 示例 2:
    输入: [4,3,2,1]
    输出: [4,3,2,2]
    解释: 输入数组表示数字 4321。
    
题解:
  • 递减相加法
    循环递减,在循环中判断相加是否为10,如果为10,就把当前值改为0,下一值加一;
    执行用时:72ms;内存消耗:33.6MB;(性能不好)
    var plusOne = function(digits) {
      for(let i=digits.length-1;i>=0;i--){
          if(i==digits.length-1){
              digits[i]++
          }
          if(digits[i]==10){
              digits[i]=0
              if(i-1<0){
                  digits.unshift(1)
              }else{    
                  digits[i-1]++
              }
          }
      }
      return digits
    };
    
  • 迭代
    for循环可以实现的,while也可以实现;
    执行用时:72ms;内存消耗:34.2MB;
    var plusOne = function(digits) {
      const values = digits.reverse();
      let i = 0;
      values[i] += 1;
      while (true) {
        if (values[i] >= 10) {
          values[i] = 0;
          if (values[i + 1] === undefined) {
            values[i + 1] = 1;
            break;
          } else {
            values[i + 1] += 1
          }
          i++
        } else {
          break;
        }
      }
      return values.reverse()
    };
    

第一期结束,下期见。如果有一起学习的朋友,可以加微信群。


图片发自简书App
镇楼
最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351