javascript 是一门复杂的语言。若是你是一名 javascript 开发者,理解它的一些基础概念是很重要的。本文选取了 12 个 JS 开发者应该掌握的概念,但不表明 JS 开发者须要了解的所有内容。javascript
注意:我会在 github 仓库JS Tips & Tidbits上持续更新这个列表,若是有兴趣欢迎 star。java
理解 javascript 中如何分配变量的值是写好代码的基础。若是你还不了解这些,你可能很容易写出无心中修改值的代码。git
javascript 老是按值分配变量。可是请特别注意:当分配的值是 javascript 的 5 种原始类型(boolean
null
undefined
string
number
)时,是分配的真正的值。而若是分配的值是 Array
Function
Object
时,只会分配该对象在内存中的一个引用。github
举个例子。在下面的代码中,var2 被赋值为 var1。由于 var1 是原始类型(string),var2 的值就是 var1 的 值,而且与 var1 是彻底独立的两个值,只是它们都是一样的字符串。或者说,从新给 var2 赋值对 var1 没有影响。api
let var1 = 'My string';
let var2 = var1;
var2 = 'My new string';
console.log(var1);
// 'My string'
console.log(var2);
// 'My new string'
复制代码
和赋值对象进行比较:数组
let var1 = { name: 'Jim' };
let var2 = var1;
var2.name = 'John';
console.log(var1);
// { name: 'John' }
console.log(var2);
// { name: 'John' }
复制代码
若是你指望像分配原始类型那样的结果,这里就会出现问题,修改 var2 一样会影响到 var1。若是你建立了一个无心中修改对象的函数,就可能有难以预料的错误。promise
闭包是 javascript 中很重要的特性,能够实现变量的私有访问。在下面的例子中,createGreeter 返回了一个匿名函数,函数能够访问外层函数的 greeting 参数。安全
function createGreeter(greeting) {
return function(name) {
console.log(greeting + ', ' + name);
};
}
const sayHello = createGreeter('Hello');
sayHello('Joe');
// Hello, Joe
复制代码
在实际编码中,你可能但愿有一个初始化函数 apiConnect(apiKey) 可以返回某些方法会用到的 apiKey。这种状况下,apiKey 只须要提供一次便可。闭包
function apiConnect(apiKey) {
function get(route) {
return fetch(`${route}?key=${apiKey}`);
}
function post(route, params) {
return fetch(route, {
method: 'POST',
body: JSON.stringify(params),
headers: {
Authorization: `Bearer ${apiKey}`,
},
});
}
return { get, post };
}
const api = apiConnect('my-secret-key');
// No need to include the apiKey anymore
api.get('http://www.example.com/get-endpoint');
api.post('http://www.example.com/post-endpoint', { name: 'Joe' });
复制代码
不要忽略 javascript 的参数解构!这是从对象中干净地提取属性的经常使用方法。app
const obj = {
name: 'Joe',
food: 'cake',
};
const { name, food } = obj;
console.log(name, food);
// 'Joe' 'cake'
复制代码
若是你想将属性解构成不一样的名称,参考下面的语法:
const obj = {
name: 'Joe',
food: 'cake',
};
const { name: myName, food: myFood } = obj;
console.log(myName, myFood);
// 'Joe' 'cake'
复制代码
下面的例子中,解构用来干净地将 person 对象传递给 introduce 函数。或者说,解构能够(常常)用来提取传递给函数的参数的属性。若是你熟悉 React,你可能见过下面的代码。
const person = {
name: 'Eddie',
age: 24,
};
function introduce({ name, age }) {
console.log(`I'm ${name} and I'm ${age} years old!`);
}
console.log(introduce(person));
// "I'm Eddie and I'm 24 years old!"
复制代码
一个相对简单的 javascript 概念。下面的例子中,Math.max 不能接收一个数组,而是接收单个值做为参数。展开运算符...
就是用来把数组里的元素一个一个拉出来。
const arr = [4, 6, -1, 3, 10, 4];
const max = Math.max(...arr);
console.log(max);
// 10
复制代码
说一下 javascript 的剩余运算符。你能够用它将任意数量的参数放入一个数组再传递给函数。
function myFunc(...args) {
console.log(args[0] + args[1]);
}
myFunc(1, 2, 3, 4);
// 3
复制代码
javascript 的数组方法常常能让你很优雅、便捷地转换你想要的数据。我常常看到有关如何以某种方式操纵对象数组的问题。这正是数组方法的可用之处。
我将在这里介绍一些不一样的数组方法,以相似的可能会混淆的方法来分类。这个列表并不全面,我鼓励大家在 MDN 上反复复习并练习这些方法。
有人可能对这 3 个方法有些混乱。但这些都是转换数组或返回聚合值的有用方法。
const arr = [1, 2, 3, 4, 5, 6];
const mapped = arr.map(el => el + 20);
console.log(mapped);
// [21, 22, 23, 24, 25, 26]
复制代码
const arr = [1, 2, 3, 4, 5, 6];
const filtered = arr.filter(el => el === 2 || el === 4);
console.log(filtered);
// [2, 4]
复制代码
const arr = [1, 2, 3, 4, 5, 6];
const reduced = arr.reduce((total, current) => total + current);
console.log(reduced);
// 21
复制代码
这三个方法一般会被混为一谈,下面是使用方法:
const arr = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10];
const found = arr.find(el => el > 5);
console.log(found);
// 6
复制代码
注意,虽然 5
以后的元素都知足条件,但只会返回第一个匹配元素。
const arr = ['Nick', 'Frank', 'Joe', 'Frank'];
const foundIndex = arr.findIndex(el => el === 'Frank');
console.log(foundIndex);
// 1
复制代码
const arr = ['Nick', 'Frank', 'Joe', 'Frank'];
const foundIndex = arr.indexOf('Frank');
console.log(foundIndex);
// 1
复制代码
这是一组很棒的数组方法,可让你有针对性地添加或删除数组中的元素。
let arr = [1, 2, 3, 4];
const pushed = arr.push(5);
console.log(arr);
// [1, 2, 3, 4, 5]
console.log(pushed);
// 5
复制代码
let arr = [1, 2, 3, 4];
const popped = arr.pop();
console.log(arr);
// [1, 2, 3]
console.log(popped);
// 4
复制代码
let arr = [1, 2, 3, 4];
const shifted = arr.shift();
console.log(arr);
// [2, 3, 4]
console.log(shifted);
// 1
复制代码
let arr = [1, 2, 3, 4];
const unshifted = arr.unshift(5, 6, 7);
console.log(arr);
// [5, 6, 7, 1, 2, 3, 4]
console.log(unshifted);
// 7
复制代码
这 2 个方法会修改数组或返回一个子数组。
下面的例子能够理解为:在索引为 1 的位置移除 0 个元素并插入了元素 b。
let arr = ['a', 'c', 'd', 'e'];
arr.splice(1, 0, 'b');
复制代码
let arr = ['a', 'b', 'c', 'd', 'e'];
const sliced = arr.slice(2, 4);
console.log(sliced);
// ['c', 'd']
console.log(arr);
// ['a', 'b', 'c', 'd', 'e']
复制代码
let arr = [1, 7, 3, -1, 5, 7, 2];
const sorter = (firstEl, secondEl) => firstEl - secondEl;
arr.sort(sorter);
console.log(arr);
// [-1, 1, 2, 3, 5, 7, 7]
复制代码
你是否已经所有理解了呢?我尚未。事实上,我在写这篇文章的时候也常常参考 MDN 文档——不要紧!只要知道有什么样的方法就能够了。
不要惧怕*
。生成器函数指定了下次调用next()
时会迭代什么值。能够进行有限次数的迭代,也能够在循环中进行无限次数的迭代。
function* greeter() {
yield 'Hi';
yield 'How are you?';
yield 'Bye';
}
const greet = greeter();
console.log(greet.next().value);
// 'Hi'
console.log(greet.next().value);
// 'How are you?'
console.log(greet.next().value);
// 'Bye'
console.log(greet.next().value);
// undefined
复制代码
无限迭代:
function* idCreator() {
let i = 0;
while (true) yield i++;
}
const ids = idCreator();
console.log(ids.next().value);
// 0
console.log(ids.next().value);
// 1
console.log(ids.next().value);
// 2
// etc...
复制代码
确信你已经了解了===
与==
的差别。在进行比较时,==
会进行类型转换,===
则不会。
console.log(0 == '0');
// true
console.log(0 === '0');
// false
复制代码
一个 javascript 新手常犯的错误就是直接比较对象。变量指向对象在内存中的引用,而不是对象自己。比较变量的一个方法是将它们转换为 JSON 字符串。固然这样会有缺点:不能保证对象属性的顺序。更安全的方式是使用第三方库中专用的比较方法来比较(lodash.isEqual)。
下面的对象看着是同样的,但其实它们指向不一样的引用。
const joe1 = { name: 'Joe' };
const joe2 = { name: 'Joe' };
console.log(joe1 === joe2);
// false
复制代码
反过来讲,下面的比较结果为 true,由于变量被直接赋值为相同的值,都指向同一个引用(在内存中只有一个对象)。
const joe1 = { name: 'Joe' };
const joe2 = joe1;
console.log(joe1 === joe2);
// true
复制代码
复习下以前的值 VS 引用章节,保证能彻底理解将一个引用变量赋值给其余变量时,实际上是赋值了内存中同一对象的相同引用给其余变量。
许多人都会被 javascript 的回调函数吓到!其余它们很简单,来看例子。console.log
做为回调函数传递给了myFunc
。当计时器就绪时执行。
function myFunc(text, callback) {
setTimeout(function() {
callback(text);
}, 2000);
}
myFunc('Hello world!', console.log);
// 'Hello world!'
复制代码
一旦你理解的回调函数,你就会陷入回调地狱。而后 Promises 解决了问题。在 Promise 中包裹你的异步逻辑,成功时调用 resolve,失败时调用 reject。使用 then 来处理成功状态,使用 catch 来处理异常状态。
const myPromise = new Promise(function(res, rej) {
setTimeout(function() {
if (Math.random() < 0.9) {
return res('Hooray!');
}
return rej('Oh no!');
}, 1000);
});
myPromise
.then(function(data) {
console.log('Success: ' + data);
})
.catch(function(err) {
console.log('Error: ' + err);
});
// If Math.random() returns less than 0.9 the following is logged:
// "Success: Hooray!"
// If Math.random() returns 0.9 or greater the following is logged:
// "Error: On no!"
复制代码
一旦你掌握了 Promises
,你就会喜欢async/await
,它是基于 promises 的语法糖。下面的例子中咱们使用了 async
函数,在函数里使用了 await
来处理greeter
。
const greeter = new Promise((res, rej) => {
setTimeout(() => res('Hello world!'), 2000);
});
async function myFunc() {
const greeting = await greeter;
console.log(greeting);
}
myFunc();
// 'Hello world!'
复制代码
若是你还不了解这 12 个概念,你可能已经至少增加了一些 javascript 的知识。若是你已经了解了这些,但愿这是你练习和增强知识的机会。