最近看zepto源碼,發現有用到了位運算符-not (~),之前也見過相似「~~value」的用法,因此研究了下爲何這樣用。測試
先看看w3c的定義:code
位運算 NOT 由否認號(~)表示,它是 ECMAScript 中爲數很少的與二進制算術有關的運算符之一。ip
位運算 NOT 是三步的處理過程:zepto
把運算數轉換成 32 位數字源碼
把二進制數轉換成它的二進制反碼(0->1, 1->0)console
把二進制數轉換成浮點數效率
簡單的理解,對任一數值 x 進行按位非操做的結果爲 -(x + 1)二進制
console.log('~null: ', ~null); // => -1 console.log('~undefined: ', ~undefined); // => -1 console.log('~0: ', ~0); // => -1 console.log('~{}: ', ~{}); // => -1 console.log('~[]: ', ~[]); // => -1 console.log('~(1/0): ', ~(1/0)); // => -1 console.log('~false: ', ~false); // => -1 console.log('~true: ', ~true); // => -2 console.log('~1.2543: ', ~1.2543); // => -2 console.log('~4.9: ', ~4.9); // => -5 console.log('~(-2.999): ', ~(-2.999)); // => 1
那麼, ~~x就爲 -(-(x+1) + 1)im
console.log('~~null: ', ~~null); // => 0 console.log('~~undefined: ', ~~undefined); // => 0 console.log('~~0: ', ~~0); // => 0 console.log('~~{}: ', ~~{}); // => 0 console.log('~~[]: ', ~~[]); // => 0 console.log('~~(1/0): ', ~~(1/0)); // => 0 console.log('~~false: ', ~~false); // => 0 console.log('~~true: ', ~~true); // => 1 console.log('~~1.2543: ', ~~1.2543); // => 1 console.log('~~4.9: ', ~~4.9); // => 4 console.log('~~(-2.999): ', ~~(-2.999)); // => -2
判斷數值中是否有某元素時,之前這樣判斷:co
if(arr.indexOf(ele) > -1){...} //易讀
如今能夠這樣判斷,二者效率:
if(~arr.indexOf(ele)){...} //簡潔
對於浮點數,~~value能夠代替parseInt(value),並且前者效率更高些
parseInt(-2.99) //-2 ~~(-2.99) //-2
var time1 = +new Date(); var count = 5000000; var ele = 1; var arr = [1,2,4,5,2]; var h = 1.01; console.time('parseInt'); for (var i = count; i > 0; i--) { parseInt(h); } console.timeEnd('parseInt'); //84.385ms console.time('~~'); for (var i = count; i>0; i--) { ~~h; } console.timeEnd('~~'); //13.386ms console.time('arr.indexOf(ele) > -1'); for (var j = count; j>0; j--) { arr.indexOf(ele) > -1; } console.timeEnd('arr.indexOf(ele) > -1'); //16.263ms console.time('~arr.indexOf(ele)'); for (var i = count; i>0; i--) { ~arr.indexOf(ele); }