This article is from minegithub
The foreword 0.
When you learn, you start off as a beginner, and then when you get to the more advanced things, you find that everything is easy, and some people may start to say that they are proficient. Suddenly one day, found some SAO operation based on very basic things, began to doubt life: WTF JavaScript? If you don’t get hit by something or see a new world, maybe forever, you will sigh: jQuery is so easy to use, I am proficient in jQuery, PROFICIENT in JS. Or vue? Presents? The react? I know them all. I know them all.
However, I now can only say that I am just familiar with, the more learning found more pit. For data type conversions and regex pits, we saw earlier: data type regular expressions
An array of 1.
Similar to some traversal types of API: forEach, map, maybe some people say: Arr. Map ((x,index)=>x+ 1); arr. Map ((x,index)=>x+ 1); [‘1′,’2′,’3’]. Map (parseInt) First, map can pass two arguments: map (the function called on each element, the function’s this value) and the function called on each element, which passes the arguments (current, index, arr) to the current element (must), index, and array.
1.1 Have you mastered parseInt
In this case, these are the three arguments passed to parseInt. And parseInt, the native function, takes the argument (num, radix) : radix denotes the radix of the number to be parsed. The value is between 2 and 36. If this parameter is omitted or its value is 0, the number is parsed on the basis of 10. If it starts with “0x” or “0x”, it will be based on 16. If this parameter is less than 2 or greater than 36, parseInt() returns NaN
parseInt(10.10)// Round the base 10, 10
parseInt(10.2)// Round the base 2 number 10, 2
parseInt(111.2)// Round the base 2 number 111, 7
parseInt(111.4)// Round the base 4 111, 21
parseInt('1'.2[1])//1, passing a third array makes no sense
parseInt('1'.2['1'.'2'.'3'.'4'])/ / 1
parseInt('2'.2['1'.'2'.'3'.'4'])//NaN, base 2 has no 2
parseInt('2'.3['1'.'2'.'3'.'4'])/ / 3
Copy the code
Then it’s obvious why [1,NaN,NaN] is:
parseInt('1'.0['1'.'2'.'3'])// the radix is 0, then the numbers will be resolved on the basis of 10
parseInt('2'.1['1'.'2'.'3'])// What is base 1? It's NaN
parseInt('3'.2['1'.'2'.'3'])// Base 2 has no 3
Copy the code
In addition, parseInt, when it sees a string number, it tries to interpret it until it can’t.
parseInt('123sfd')/ / 123
parseInt('123sfd123')/ / 123
parseInt('sfd213')//NaN
Copy the code
1.2 Array traversal API second parameter
The second parameter, this, is the parameter inside the previous function. Normally this parameter is not filled in, strictly this mode is undefined. ForEach (function(x){console.log(this)})//window: [1,2,3] [1, 2, 3]. The forEach (function (x) {the console. The log (this)}, Math) / / Math
1.3 Make Array fine
Array(5); Array(5);
Array.apply(null, { length:5 })// [undefined, undefined, undefined, undefined, undefined]
Array.apply(null, { '0':1.length:5 })// [1, undefined, undefined, undefined, undefined]
Array.apply(null, { '2':1.length:5 })// [undefined, undefined, 1, undefined, undefined]
Copy the code
This is equivalent to directly changing the properties of the newly created array
1.4 Do you loop to generate a sequence?
Generates an array of ordinals:
var arr = [];
for(var i = 0; i<10; i++){ arr.push(i) }Copy the code
Normal operation, no problem, but as a jQuery expert will you use any other methods? Such as:
Array.apply(null, {length:5 }).map(f.call,Number)//[0, 1, 2, 3, 4] f can be any function
Array.apply(null, { '0':1.length:5 }).map(f.call,Number)//[0, 1, 2, 3, 4], no matter what the element is
Array.apply(null, {length:5 }).map(f.call,Boolean)//[false, true, true, true, true]
Array.apply(null, {length:5 }).map(f.call,String)//["0", "1", "2", "3", "4"]
Array.apply(null, {length:5 }).map(eval.call,Object)//[Number, Number, Number, Number, Number]
Copy the code
For the last result, click on the second one
The map function takes three arguments, the first of which is the current element. But for call, the first argument is that call’s this points to. The map parameter, which is this, overwrites current and is the dominant parameter. The first parameter of map fn, and the second parameter of MAP FN, is index, which is the current index. For f.call, the first argument is this, and the second argument is all the arguments used by f of f. Call. Number (index), Boolean (index), String (index), Object (index)
2. Bitwise operators
Leave out the basic usage and concepts and read the documentation.
2.1 Converting a String to a Number
ParseInt is known all over the world, but we can do it a little simpler:
// Let him do the math without changing his own value
+"1"
"1"*1
"1"|0
"1" >> 0
"1" << 0
Copy the code
2.2 More operations
If you want to get a arbitrarily large number, it is usually 9999*9999. If you want to get a arbitrarily large number, it is 9999*9999. If you want to get a arbitrarily large number, it is 9999*9999. Generating random strings (numbers + letters) I know a lot of people who don’t have to think about it and just pick up the keyboard and churn out, say, a random 6-digit string
var n = 6
var str = 'abcdefghijklmnopqrstuvwxyz0123456789'
var result = ' '
for (var i = 0; i<n; i++){ result += str[parseInt(Math.random()*(str.length+1))]}Copy the code
For bitwise operations, this is a short code solution:
(~ ~ (Math.random()*(1<<24))).toString(16) (~ ~ (Math.random()*(1<<30))).toString(36)
Copy the code
First generate a large number, then switch to base. Hexadecimal is 0-9 plus the first few letters, and hexadecimal is 0-9 plus 26 letters, so we get a stable version of the random string that generates n bits:
function f(n){
if(n==1) return(~ ~ (Math.random()*36)).toString(36)
return(~ ~ (Math.random()*36)).toString(36) + f(n- 1)}// Tail recursive optimization
function k(n){
return function f(n,s){
if(n==1) return s
return f(n- 1, (~ ~ (Math.random()*36)).toString(36)+s)
}(n,(~~(Math.random()*36)).toString(36))}Copy the code
Another way: (also based on high base) we can get a string with 0. XXX followed by 11 decimal places from math.random ().tostring (36), so we can just get the string after 2 decimal places. Math.random().tostring (36).slice(2)
Let’s have a little interlude
For obsessive-compulsive disorder in pursuit of code is very short, see above the if – else, suddenly want to wrote a lot of the if – else at ordinary times is not pleasing to the eye, yes, but no brain the if three expression and short, we have a short circuit expressions: | | && a && b: a is true, ran to a | b | b: If a is false, run B, if a is true, take A
// Let's take a slightly retarded example
function f(a){
if(a==1) console.log(1)
if(a==2) console.log(2)
if(a==3) console.log(3)}// Be sure to write a semicolon
function f(a){
(a==1) &&console.log(1);
(a==2) &&console.log(2);
(a==3) &&console.log(3);
}
Copy the code
In practice, the code will be much cleaner, but may be hard to understand the first time around
The bit operation swaps two integers
A = a+b; b = a-b; A ^= b; a ^= b; b ^= a; a ^= b;
The specific process can be proved like this:
We shilling a0 = a, b0 = b. A = b ^ a = b0 ^ a = b0 ^ (a0 ^ B0) = A0 ^ (b0 ^ B0) = A0 ^ 0 = a0 ^ 0 A = a ^ b = a ^ (b0 ^ a)= b0 ^ (a ^ a)= b0 ^ 0 = b0, successfully swapped with the original b0Copy the code
3. The class construction
Going back to the previous example:
Array.apply(null, {length:5 }).map(f.call,Number)//[0, 1, 2, 3, 4] f can be any function
Array.apply(null, { '0':1.length:5 }).map(f.call,Number)//[0, 1, 2, 3, 4], no matter what the element is
Array.apply(null, {length:5 }).map(f.call,Boolean)//[false, true, true, true, true]
Array.apply(null, {length:5 }).map(f.call,String)//["0", "1", "2", "3", "4"]
Array.apply(null, {length:5 }).map(eval.call,Object)//[Number, Number, Number, Number, Number]
Copy the code
The map second argument CTX refers to this, and the first argument is a function f (any function). The first argument of F is deprecated because the first argument is the context of call this, but this is replaced by CTX, so the valid argument of F starts with the second argument. This is the equivalent of CTX (index), which is: construct class (index)
So we can look at another interesting aspect of the constructor class
var toFixed = 1;
var obj = {
toFixed:"I'm just a guest star.".show:function(){
return this. toFixed;
}
}
obj.show.call( toFixed); ƒ toFixed() {[native code]}
Copy the code
Call may look like 1 at first glance, but the first argument of call is not null, undefined. The context of our call is toFixed. Number (1); this refers to Number (1); this refers to Number
This is also true for strings
var big = '1sdasdsadsdasd';// If it is not a string, the other constructor classes have no big method and return undefined
var obj = {
big:"I'm a guest star.".show:function(){
return this.big;
}
}
obj.show.call(big); ƒ big() {[native code]}
// Alternatively, print a length and see
var l = '1sdasdsadsdasd';// Change the variable to l
var obj = {
length:"I'm a guest star.".show:function(){
return this.length;// It doesn't matter what the variable is
}
}
obj.show.call(l); / / 14
Copy the code
There are too many properties. Go to the console and see what String.prototype has.
Or, see what happens when the function executes:
var l = true;// Try Boolean this time
var obj = {
length:"I'm a guest star.".show:function(){
return this.valueOf();// We will not print this function this time, let it execute
}
}
obj.show.call(l,this); //true to call valueOf directly during the conversion
Copy the code
It’s all the same. You can go back and play