您是否曾經遇到過這樣的情況:您正在使用JavaScript 創建功能性Web 應用程序,突然間,您的完美構建變成了嵌入粗紅線(錯誤)中的惡夢?沮喪開始出現,你會感到擔憂,質疑自己的技能並問自己,「出了什麼問題?」
有趣的是,在大多數情況下,程式碼中斷和錯誤的原因並不是一些複雜的錯誤,而是隱藏在程式碼基礎中的常見錯誤。在這種情況下,本指南將成為您編碼之旅的指南針。
在本文中,您將學習如何克服這些常見錯誤。本指南將提供引人入勝的解釋和實用的解決方案,您可以在其中看到錯誤以及如何修復它們。
話不多說,讓我們開始吧。
在 JavaScript 中,您可以使用 let、var 或 const 等關鍵字宣告變數。雖然開發人員通常將 var 和 let 視為可互換的,因為它們具有共享的可變性特性(允許使用它們聲明的變數被更改或變異),但它們的作用域存在關鍵差異。必須認識到這些關鍵字在定義變數範圍的方式上有顯著差異。
JavaScript 將作用域分為全域作用域、函數作用域和區塊作用域。在本節中,您只需要關注兩種類型的作用域:函數和區塊。
函數作用域:函數內宣告的變數只能在該函數內部訪問,而不能在函數外部存取。此技術封裝並防止程式碼其他部分的意外修改。
區塊作用域:在ES6 (ECMAScript 2015) 中引入,區塊作用域允許您在大括號定義的特定程式碼區塊內使用let 和const 關鍵字聲明變量,例如if 語句、循環和箭頭函數。這提供了對變數可訪問性的更精確的控制,並有助於防止不必要的副作用。
考慮到區塊作用域的定義,您可能已經注意到缺少 var 關鍵字。這是一個疏忽嗎?一點也不!使用 var 關鍵字定義的變數是函數作用域的,而使用 let 或 const 宣告的變數是區塊作用域的。以下程式碼片段將進一步闡明這些概念。
我們先將函數作用域的定義翻譯成程式碼:
function testVar() { var test = "new variable"; } console.log(test); // This will return an error: Uncaught ReferenceError: test is not defined.
上面的程式碼確認使用 var 關鍵字宣告的變數是函數作用域的。當您嘗試在定義該變數的函數之外使用 console.log() 語句存取測試變數時,您會收到錯誤 - “Uncaught ReferenceError:測試未定義。”
在這種情況下,變數的作用域僅限於定義它並隨後被銷毀的函數。因此,在函數外部,console.log() 語句會導致錯誤,因為變數在銷毀時會從記憶體堆疊中刪除。
但是,與先前的嘗試不同,在定義函數中使用 console.log() 語句存取測試變數可以檢索變數的值,而不會遇到引用錯誤。
function testVar() { var test = "new variable"; console.log(test); // This returns - new variable } testVar();
下面的程式碼將幫助您了解這些變數如何根據其作用域類型運作。
function variableScopes() { // testing var scope var varTest = 1; if (true) { var varTest = 2; console.log("Inside this if block, var yields this:", varTest); } console.log("Outside the if block, var yields this:", varTest); // testing let scope let letTest = 1; if (true) { let letTest = 2; console.log("Inside this if block, let yields this:", letTest); } console.log("Outside the if block, let yields this:", letTest); // testing const scope const constTest = 1; if (true) { const constTest = 2; console.log("Inside this if block, const yields this:", constTest); } console.log("Outside the if block, const yields this:", constTest); } variableScopes(); // The code result: // Inside this if block, var yields this: 2 // Outside the if block, var yields this: 2 // Inside this if block, let yields this: 2 // Outside the if block, let yields this: 1 // Inside this if block, const yields this: 2 // Outside the if block, const yields this: 1
從上面的程式碼扣除:
var:var 關鍵字是函數作用域的,這意味著用 var 宣告的變數可以在整個函數中存取。提供的程式碼在 if 語句內部和外部宣告變數 varTest 兩次。然而,由於 var 的函數作用域,該變數本質上是在同一作用域內重新聲明的,從而導致意外的行為。因此,當 if 區塊內修改 varTest 的值時,它會影響區塊外宣告的變量,導致兩個 console.log 語句都顯示值 2。
let:與 var 不同,let 關鍵字是區塊作用域的,這表示用 let 宣告的變數只能在定義的區塊內存取。在程式碼中,變數letTest是使用let關鍵字在if語句的內部和外部宣告的。正如區塊作用域所預期的那樣,修改 if 區塊內的 letTest 值只會影響該區塊內的變數。因此,if 區塊內的第一個 console.log 顯示值 2,而區塊外的第一個 console.log 保留初始值 1。
const: Similar to the let keyword, variables declared with const are also block-scoped. The code declares the variable constTest as a constant inside and outside the if statement. Despite being constants, the block scope allows re-declaration within different blocks. However, reassigning a value to a constant is not allowed. Thus, modifying the value of constTest inside the if block affects only the variable within that block, while the value outside the block remains unchanged.
Mastering the use of equality operators is crucial for JavaScript developers. Unlike other languages that rely on a single type, JavaScript offers two: the strict equality operator and the loose equality operator. Understanding the nuances between these operators is essential for writing robust and error-free JavaScript code.
The loose equality performs type coercion before comparison. This operator tends to convert the operands to the same type before evaluating the equality. While this may seem convenient, it often leads to unexpected results, especially when comparing different data types.
The loose equality operator in JavaScript follows several basic rules to determine when to perform type coercion:
If the operands have the same type, In this case, where both have the same type, no coercion is necessary, and the equality comparison gets done.
If one operand is null and the other is undefined, JavaScript treats null and undefined as equal when using loose equality. Also, null and undefined values can't undergo type coercion, so they are equal.
If one operand is a number and the other is a string, JavaScript will attempt to convert the string operand to a number before comparing the values.
If one operand is a boolean, If one operand is a boolean, JavaScript will attempt to convert that operand to the numeric value. The numeric values: the value of false is converted to 0, whereas the value of true is converted to 1.
If one operand is an object and the other is not, JavaScript will attempt to convert the object operand to a primitive value (using the valueOf and toString methods) before comparing.
If both operands are objects, JavaScript compares their references to determine whether they are the same object. Note that the references are compared, not their values. Two objects are only equal if they reference the same object in memory.
If either operand is NaN, the loose equality operator returns false. Note that even if both operands are NaN, you still get false because, by default, NaN is not equal to NaN.
Strict equality compares both values and types strictly without performing any type coercion. It demands that both operands have the same value and data type for equality to be true. This approach offers clarity and predictability, eliminating the pitfalls of implicit type conversion.
It's of utmost importance to be aware of the potential issues and unexpected outcomes that can arise when using loose equality. To steer clear of these, it is strongly recommended that you always use strict equality when comparing operands. This practice will help you maintain data type integrity throughout your codebase, ensuring a more robust and reliable application.
Here is a code snippet for testing different scenarios with their respective results:
// These snippets showcase how JavaScript's loose and //strict equality operators behave in various scenarios. // Testing loose equality const num1 = 5; const num2 = "5"; console.log(num1 == num2); // Testing strict equality: const num3 = 5; const num4 = "5"; console.log(num3 === num4); // false // More examples of loose equality const value1 = 0; const value2 = false; console.log(value1 == value2); // true // More examples of strict equality: const value3 = 0; const value4 = false; console.log(value3 === value4); // false // Array comparison with loose equality: const arr1 = [1, 2, 3]; const arr2 = "1,2,3"; console.log(arr1 == arr2); // true // Array comparison with strict equality: const arr3 = [1, 2, 3]; const arr4 = "1,2,3"; console.log(arr3 === arr4); // false
JavaScript developers are often fond of adding an else block immediately after an if block, even when the logic inside the else block is essentially the default behaviour or can be handled without the else block. This leads to unnecessary code complexity, which can be simplified by restructuring the code.
An else block is executed when the condition of the preceding if statement is evaluated as false. However, in situations where the else block contains code that can be executed without explicitly checking the condition, using the else block becomes redundant and can be avoided.
Code example:
const isEven = (num) => { if (num % 2 === 0) { return true; } else { return false; } } isEven(4); // true isEven(3); // false
In this code, the function isEven takes a number as input and checks if it's even. If the number is divisible by two without a remainder, it returns true, indicating that the number is even. Otherwise, it returns false, indicating that the number is odd. However, using the else block is unnecessary because if the condition in the if statement is not met (i.e., the number is not even), the function will naturally reach the next line after the if block without needing an else statement.
Now, without the else statement:
const isEven = (num) => { if (num % 2 === 0) { return true; } return false; } isEven(4); // true isEven(3); // false
This code is a more concise version of the first code. It checks if the number is even using the if statement; if it is, it returns true. However, if the number is not even, it will return false. There's no need for an else block because the return false statement will only execute if the condition for the if statement is false.
You can even simplify it further by using an arrow function like this:
const isEven = (num) => { return num % 2 === 0; }
This code further simplifies the logic by directly returning the result of the expression num % 2 === 0. If this expression is evaluated as true, the function returns true, indicating that the number is even. Otherwise, it returns false.
A developer must understand the difference between mutable and immutable types and how they work under the hood. This segment will take objects as the case study to deeply understand how mutable types work and the mistakes usually made when working with them.
Consider the code snippet below to understand how immutable types work in JavaScript:
let x = 2; let y = x; // y becomes the copy of the value in variable x which is 2 x += 2; y += 1; console.log(x, y); //This will return 4, 3
In this code sample, we define a variable x with a value of 2. The variable x holds an immutable type, precisely a number. Once you define an immutable type, you can't change it directly.
The line "let y = x" creates a copy of the variable x value and assigns it to y. It's crucial to understand that y is independent of x; any operations performed on x will not affect y, and vice versa. This concept is fundamental to understanding immutable types.
Mutable types, such as objects and arrays in JavaScript, behave differently from immutable types. When you assign one variable to another with mutable types, they point to the same underlying data in memory. This means that modifying one variable will directly affect the other, as they share a reference to the same data.
Let's consider a simple array code snippet to illustrate this concept:
let x = []; let y = x; x.push(1); y.push(2); console.log(x); // output [1, 2] console.log(y); // output [1, 2]
In the code above, variables x and y are arrays initially empty. When we push elements into x and y, we observe that both arrays end up with the same elements - [1, 2]. Unlike immutable types, this behaviour is because mutable types store a reference to the object rather than the actual value.
Here's a breakdown of what happens in the code snippet:
A common mistake occurs when developers mix mutable and immutable types, mainly when dealing with nested objects. Let’s consider the code snippet below:
const developer = { name: "John Doe", age: 25, core: { programmingLanguage: "JavaScript", framework: "React", role: "Frontend Developer", }, }; // creating a similar developer but with a different name and language const developer2 = developer; developer2.name = "Jane Doe"; developer2.core.programmingLanguage = "TypeScript"; console.log("developer", developer); // {name: 'Jane Doe', age: 25, core: {programmingLanguage: 'TypeScript', framework: 'React', role: 'Frontend Developer"}} console.log("developer2", developer2); // {name: 'Jane Doe', age: 25, core: {programmingLanguage: 'TypeScript', framework: 'React', role: 'Frontend Developer"}}
When you run the code above, you will notice that developer2, which copies the property of the developer object, modified the developer object, automatically making the two objects identical. Now, I believe you understand why! It is because they are both pointing to the same reference in memory. The question is, "How can you have the two objects without them modifying each other?"
When dealing with objects in JavaScript, a common approach for creating a copy is to use the spread operator (...). However, it's important to note that this method, known as shallow copy, only addresses part of the cloning process.
Consider the use of the spread operator in creating a shallow copy:
const developer2 = { ...developer };
While this method prevents direct modifications to the global object developer, it falls short when dealing with nested objects. Changes to nested properties in the copied object developer2 can still impact the original object developer. For example:
developer2.core.programmingLanguage = "TypeScript";
This modification affects the programmingLanguage value in the nested core object of both developer and developer2.
The reason behind this behaviour lies in how the spread operator operates. It copies the object's properties, but when encountering nested objects, it copies their references rather than their values. As a result, changes to nested objects in the copied version get reflected in the original object due to shared references.
To overcome this limitation and ensure a complete separation between objects, developers turn to a concept known as deep cloning. Deep cloning involves creating a new object with independent values, including nested objects.
In JavaScript, a standard method for deep cloning is using JSON.parse() and JSON.stringify(). To see how this works, replace this line of code:
const developer2 = developer;
With this:
const developer2 = JSON.parse(JSON.stringify(developer));
This new line of code creates a deep clone of the developer object, ensuring that modifications to developer2 do not affect the developer. The JSON.stringify() method converts the object into a string (an immutable type), and then JSON.parse() converts it back into an object, effectively creating a new object with its separate memory reference. Note that there are other methods for performing deep cloning.
The code using deep cloning:
const developer = { name: "John Doe", age: 25, core: { programmingLanguage: "JavaScript", framework: "React", role: "Frontend Developer", }, }; // creating a similar developer but with a different name and language const developer2 = JSON.parse(JSON.stringify(developer)); developer2.name = "Jane Doe"; developer2.core.programmingLanguage = "TypeScript"; console.log("developer", developer); // {name: 'John Doe', age: 25, core: {programmingLanguage: 'JavaScript', framework: 'React', role: 'Frontend Developer"}} console.log("developer2", developer2); // {name: 'Jane Doe', age: 25, core: {programmingLanguage: 'TypeScript', framework: 'React', role: 'Frontend Developer"}}
One common mistake JavaScript developers encounter is not fully grasping the behaviour of the this keyword within arrow functions compared to regular functions. This distinction is crucial as it affects how this is scoped and accessed within different function types.
Sample Code:
function createProgrammingLanguage(name, author) { return { name, author, useRegularFunction: function () { console.log(`${this.name} was created by ${this.author}`); }, useArrowFunction: () => { console.log(`${this.name} was created by ${this.author}`); }, }; } const javascript = createProgrammingLanguage( "JavaScript", "Brendan Eich", ); javascript.useArrowFunction(); javascript.useRegularFunction();
Run the code above in a code editor. If you use an online code editor like CodeSandbox, you will notice that the editor throws an error - "Cannot read properties of undefined (reading 'type')." It points out that the issue is from the useArrowFunction(), but if you are using an IDE like Visual Studio Code or Sublime Text then you would notice that calling the useArrowFunction() method results in this: "JavaScript was created by Brendan Eich" while calling the useRegularFunction() results in this: "undefined was created by undefined".
The error encountered when running the useArrowFunction method occurs because arrow functions do not have this context; they inherit it from the surrounding lexical scope where they are defined. In this case, since useArrowFunction is defined within createProgrammingLanguage, it refers to the global context (usually a window object in a browser environment) where name, and author are undefined.
On the other hand, useRegularFunction is a regular function defined inside an object literal. Regular functions have their own this context, which is determined by how they are called. In this context, this correctly refers to the object returned by createProgrammingLanguage(), allowing useRegularFunction() to access name and author properties without issues.
This solution demonstrates how to correctly use this within an arrow function by leveraging lexical scoping.
function createProgrammingLanguage(name, author) { return { name, author, overallFunction: function () { console.log( `Regular function: ${this.name} was created by ${this.author}`, ); // Arrow function inherits this from the parent function const arrowFunction = () => { console.log( `Arrow function: ${this.name} was created by ${this.author}`, ); }; arrowFunction(); }, }; } const javascript = createProgrammingLanguage("JavaScript", "Brendan Eich"); javascript.overallFunction(); // Result // Regular function: JavaScript was created by Brendan Eich // Arrow function: JavaScript was created by Brendan Eich
Regular Function Context: Inside the overallFunction, this correctly refers to the object returned by createProgrammingLanguage(). Therefore, the values are retrieved when accessing this.name and this.author.
Arrow Function Context: The arrowFunction is defined within overallFunction. Arrow functions inherit this from their surrounding lexical scope, the overallFunction. This inheritance ensures that this within the arrow function refers to the same object as this in the parent regular function. As a result, the arrow function correctly accesses this.name and this.author, producing the desired output without encountering the "Cannot read properties of undefined" error.
By understanding arrow functions' lexical scoping behaviour and leveraging this behaviour to inherit this from the parent function's context, you ensure that arrow functions can access object properties and methods within the appropriate context, resolving issues that result in having the this references being undefined.
Memory leaks in JavaScript can be subtle and insidious, causing problems on devices with limited memory or in frequently called functions. In JavaScript, memory leaks happen when data is held in memory even though it's no longer needed. The primary cause of memory leaks in JavaScript is often unwanted references. Let's explore some common scenarios, how they lead to memory leaks and the solutions.
function defineLanaguage() { language = "JavaScript"; // Accidental global variable }
In this code, language becomes a property of the window object ("window.language = 'JavaScript'"). To prevent this and ensure language is scoped correctly, use var, let, or const:
function defineLanaguage() { let language = "JavaScript"; // Scoped variable }
Using a scoped variable ensures that language goes out of scope at the end of the function's execution, preventing memory leaks.
let language = "JavaScript"; setInterval(() => { console.log(language); }, 100);
In this code, the interval timer keeps a reference to the language variable, preventing it from being garbage collected even when not needed. To avoid this, clear the interval timer when it's no longer required:
let language = "JavaScript"; let intervalId = setInterval(() => { console.log(language); }, 100); // Clear the interval when no longer needed clearInterval(intervalId);
Clearing the interval ensures the handler function and its references are cleaned up properly.
let outer = function () { let language = "JavaScript"; return function () { return language; }; };
This closure retains a reference to the language variable, preventing it from being collected as garbage. To avoid this, ensure that closures release unnecessary references:
let outer = function () { let language = "JavaScript"; return function () { return language; }; }; // Clear the outer function reference when no longer needed outer = null;
Clearing the reference to the outer function allows the language variable to be garbage collected when it's no longer needed.
By addressing these common scenarios of memory leaks in JavaScript, developers can improve the efficiency and stability of their code. To prevent memory leaks, always scope variables correctly, manage interval timers responsibly, and release unnecessary closures. You can research and see some other instances that lead to memory leaks, but the ones mentioned here are common.
One common mistake in JavaScript is improperly using a variable as a key when defining an object literal. This mistake can lead to unexpected behaviour or incorrect object structures. Let's delve into this issue, understand why it occurs, and explore the correct approach to defining object keys dynamically.
In order to understand the mistake, consider the following code snippet:
var type = true; var language = null; if (type) { language = "TypeScript"; } else { language = "JavaScript"; } var obj = { language: "programming language", }; console.log(obj); // Output - { language: "programming language" }
In the code above, the mistake is using the variable language directly as the object (obj) key. Despite language being dynamically assigned a value based on the type condition, it is interpreted as a literal key language rather than using its value as the key.
Expected Result vs Actual Result
JavaScript provides a syntax using square brackets ([ ]) to use a variable as a dynamic key in an object. This allows for the evaluation of the variable's value as the key.
var type = true; var language = null; if (type) { language = "TypeScript"; } else { language = "JavaScript"; } var obj = { [language]: "programming language", }; console.log(obj); // Output - { TypeScript: "programming language" }
Through this exploration of common JavaScript mistakes, you've gained valuable insights into how to write cleaner, more efficient code.
Keep growing, learning, and building!
以上是開發人員常犯的 JavaScript 錯誤的詳細內容。更多資訊請關注PHP中文網其他相關文章!