这是一篇记录学习 underscore v0.0.5 的fragment,觉得有点意思,和大家分享一下。

先看_.sortedIndex的源码,它用来确定 obj 在 array中的位置(array升序):

  _.sortedIndex = function(array, obj, iterator) {
iterator = iterator || _.identity;
var low = 0, high = array.length;
while (low < high) {
var mid = (low + high) >> 1;
iterator(array[mid]) < iterator(obj) ? low = mid + 1 : high = mid;
}
return low;
}; _.identity = function(value) {return value};

代码虽短却两个我们不常用的姿势:>> 运算符和二分查找。


右移运算符 >>

右移运算符的计算规则如下

int a = 8; int b = a >> 1; // b ?

先将 8 转化为二进制数,(在 C 中,32位操作系统下int 类型占2个字节,即16bit)

0000 0000 0000 1000

整体右移 1 位,高位不足添 0

0000 0000 0000 0100

再将新的二进制数转化为十进制得到 b = 4;

那么,代码 var mid = (low + high) >> 1 mid 变量取值是多少呢?因为好(dou)奇(bi),我作了一点尝试:

0 >> 1  // 0
1 >> 1 // 0
2 >> 1 // 1
3 >> 1 // 1
4 >> 1 // 2
5 >> 1 // 2
...
10 >> 1 // 5
11 >> 1 // 5
12 >> 1 // 6
...

由归纳法可得 a >> 1 === parseInt(a / 2)。这点相信眼尖的同学早就看出来了。

二分查找

既然我们已经知晓mid >> 1 === parseInt((low + high) / 2),我们再来回顾下代码片段:

while (low < high) {
var mid = (low + high) >> 1; // (1)
iterator(array[mid]) < iterator(obj) ? low = mid + 1 : high = mid; // (2)
}

(1)我们已经知晓,mid 取得了一个中间数值,在(2)中是一个三元运算符,解释起来,就是用一个有序数组 array 的中间数和目标数进行比较,则如果中间数比目标数小,就只需要比较中间数右侧的值,反之亦然。然后继续循环...

等价于下面的代码,由此差距立显:

while (low < high) {
var mid = (low + high) >> 1;
var a = iterator(array[mid]);
var b = iterator(obj)
if(a == b) {
return mid;
} else if(a < b) {
low = mid + 1;
} else {
high = mid
}
}

题外话

内容照理说已经讲完了,但对于 >>运算符我还是有些疑虑。既然有a >> 1 === parseInt(a / 2) 那么可否假设 a >> 2 === parseInt(a / (2*2))呢?

0 >> 2  // 0
3 >> 2 // 0 4 >> 2 // 1
7 >> 2 // 1 8 >> 2 // 2
11 >> 2 // 2 12 >> 2 // 3

继而a >> 3 === parseInt(a / (2*2*2)),可得a >> n === parseInt(a / (X)) X 为 2 的 n 次方。

左移运算符 <<

与右移运算符相对的叫左移运算符,记为 <<,相信它也不是吃干饭的。

有了前面的铺垫,我这里一步到位,假设有a << n === parseInt(a * (X)) X 为 2 的 n 次方。简单验证下:

1 << 1 // 1*2 = 2
2 << 1 // 2*2 = 4
3 << 2 // 3 * (2*2) = 12
5 << 4 // 5 * (2*2*2*2) = 80
...

最后的最后,由于水平有限(大家看出来了,忍住不要笑:-D),文章内容连蒙带猜,没有经过特别严谨的证明,欢迎拍砖!!!

[underscore源码学习]——`>>` 运算符和二分查找的更多相关文章

  1. Underscore.js 源码学习笔记(上)

    版本 Underscore.js 1.9.1 一共 1693 行.注释我就删了,太长了… 整体是一个 (function() {...}());  这样的东西,我们应该知道这是一个 IIFE(立即执行 ...

  2. [算法2-数组与字符串的查找与匹配] (.NET源码学习)

    [算法2-数组与字符串的查找与匹配] (.NET源码学习) 关键词:1. 数组查找(算法)   2. 字符串查找(算法)   3. C#中的String(源码)   4. 特性Attribute 与内 ...

  3. Underscore.js 源码学习笔记(下)

    上接 Underscore.js 源码学习笔记(上) === 756 行开始 函数部分. var executeBound = function(sourceFunc, boundFunc, cont ...

  4. 【 js 基础 】【 源码学习 】源码设计 (持续更新)

    学习源码,除了学习对一些方法的更加聪明的代码实现,同时也要学习源码的设计,把握整体的架构.(推荐对源码有一定熟悉了之后,再看这篇文章) 目录结构:第一部分:zepto 设计分析第二部分:undersc ...

  5. 【 js 基础 】【 源码学习 】源码设计 (更新了backbone分析)

    学习源码,除了学习对一些方法的更加聪明的代码实现,同时也要学习源码的设计,把握整体的架构.(推荐对源码有一定熟悉了之后,再看这篇文章) 目录结构:第一部分:zepto 设计分析 第二部分:unders ...

  6. [算法1-排序](.NET源码学习)& LINQ & Lambda

    [算法1-排序](.NET源码学习)& LINQ & Lambda 说起排序算法,在日常实际开发中我们基本不在意这些事情,有API不用不是没事找事嘛.但必要的基础还是需要了解掌握. 排 ...

  7. [数据结构1.2-线性表] 动态数组ArrayList(.NET源码学习)

    [数据结构1.2-线性表] 动态数组ArrayList(.NET源码学习) 在C#中,存在常见的九种集合类型:动态数组ArrayList.列表List.排序列表SortedList.哈希表HashTa ...

  8. JDK源码学习笔记——Integer

    一.类定义 public final class Integer extends Number implements Comparable<Integer> 二.属性 private fi ...

  9. [数据结构-线性表1.2] 链表与 LinkedList<T>(.NET 源码学习)

    [数据结构-线性表1.2] 链表与 LinkedList<T> [注:本篇文章源码内容较少,分析度较浅,请酌情选择阅读] 关键词:链表(数据结构)    C#中的链表(源码)    可空类 ...

随机推荐

  1. laravel的foreach

    1.控制器 2.模板

  2. ubuntu16.04 LTS Server 安装mysql phpmyadmin apache2 php5.6环境

    1.安装apache sudo apt-get install apache2 为了测试apache2是否正常,访问http://localhost/或http://127.0.0.1/,出现It W ...

  3. java日期正则表达式精准校验

      function checkDate(obj) {       var date=obj.value;     var re = new RegExp("(([0-9]{3}[1-9]| ...

  4. MYSQL 问题小总结

    mysql 问题小总结 1.MySQL远程连接ERROR 2003(HY000):Can't connect to MySQL server on ‘ip’(111)的问题 通常是mysql配置文件中 ...

  5. hdu-1107(模拟题)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1107 注意:1.路线是要反向的,走不通就反向: 2.输入输出全部是整形 3.攻击力不断变化 #incl ...

  6. IoC的基本概念

    一.什么是IOC ioc是一个英文缩写,英文全称是 Inversion of Control,翻译过来是“控制反转”.理解好Ioc的关键是要明确“谁控制谁,控制了什么,为何是反转,哪些方面反转了” 谁 ...

  7. Shell 基本语法

    一. Linux基本命令 1.1.  cp命令 该命令的功能是将给出的文件或目录拷贝到另一文件或目录中,功能十分强大. 语法: cp [选项] 源文件或目录 目标文件或目录 1.2. mv命令 用户可 ...

  8. C/C++的Name Mangling

    C语言 函数 1.void __CALLTYPE f();2.int __CALLTYPE f();3.int __CALLTYPE f(int);4.double __CALLTYPE f(int, ...

  9. struts2从浅至深(三)拦截器

    一:拦截器概述 Struts2中的很多功能都是由拦截器完成的. 是AOP编程思想的一种应用形式. 二:拦截器执行时机                             interceptor表示 ...

  10. (匹配)Antenna Placement --POJ --3020

    链接: http://poj.org/problem?id=3020 http://acm.hust.edu.cn/vjudge/contest/view.action?cid=82834#probl ...