Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

Vì vậy, tôi đã viết về JavaScript được một thời gian rồi, tôi cũng giúp một vài người bạn của mình học JavaScript khi họ muốn tham gia phát triển giao diện người dùng. Đây là một trong những chủ đề yêu thích của cá nhân tôi, nhưng tôi hơi ngạc nhiên khi thấy nhiều lần mọi người hiểu sai

Vấn đề với JS là nó không tuân theo khuôn mẫu chung như các ngôn ngữ khác. cho e. g. trong C#, tham chiếu…

Trong Pass by Value, Hàm được gọi bằng cách truyền trực tiếp giá trị của biến làm đối số. Thay đổi đối số bên trong hàm không ảnh hưởng đến biến được truyền từ bên ngoài hàm

Javascript luôn chuyển theo giá trị, vì vậy việc thay đổi giá trị của biến không bao giờ thay đổi nguyên hàm cơ bản (Chuỗi hoặc số)

function callByValue(varOne, varTwo) { 
console.log("Inside Call by Value Method");
varOne = 100;
varTwo = 200;
console.log("varOne =" + varOne +"varTwo =" +varTwo);
}
let varOne = 10;
let varTwo = 20;
console.log("Before Call by Value Method");
console.log("varOne =" + varOne +"varTwo =" +varTwo);
callByValue(varOne, varTwo) console.log("After Call by Value Method");
console.log("varOne =" + varOne +"varTwo =" +varTwo);

output will be :
---------------
Before Call by Value Method
varOne =10 varTwo =20
Inside Call by Value Method
varOne =100 varTwo =200
After Call by Value Method
varOne =10 varTwo =20

Tuy nhiên, khi một biến đề cập đến một đối tượng bao gồm mảng, giá trị là tham chiếu đến đối tượng

Vượt qua tham khảo

Trong Pass by Reference, Hàm được gọi bằng cách chuyển trực tiếp tham chiếu/địa chỉ của biến làm đối số. Thay đổi đối số bên trong hàm ảnh hưởng đến biến được truyền từ bên ngoài hàm. Trong các đối tượng và mảng Javascript theo sau chuyển qua tham chiếu

function callByReference(varObj) { 
console.log("Inside Call by Reference Method");
varObj.a = 100;
console.log(varObj);
}
let varObj = {a:1};
console.log("Before Call by Reference Method");
console.log(varObj);
callByReference(varObj) console.log("After Call by Reference Method");
console.log(varObj);
output will be :
---------------
Before Call by Reference Method
{a: 1}
Inside Call by Reference Method
{a: 100}
After Call by Reference Method
{a: 100}

vì vậy nếu chúng ta chuyển đối tượng hoặc mảng làm đối số cho phương thức, thì có khả năng giá trị của đối tượng có thể thay đổi

Trong JavaScript, khi một hàm được gọi, các đối số có thể được truyền theo hai cách, Truyền theo giá trị hoặc Truyền theo tham chiếu (địa chỉ). Các loại dữ liệu nguyên thủy như chuỗi, số, null, không xác định và boolean, được truyền theo giá trị trong khi các loại dữ liệu không nguyên thủy như đối tượng, mảng và hàm được truyền theo tham chiếu trong Javascript

Phạm vi của Điều khoản

  • Bài viết này định nghĩa truyền theo giá trị và truyền theo phương thức tham chiếu trong javascript. Chúng ta cũng sẽ tìm hiểu sự khác biệt giữa hai phương pháp này và khi nào thì sử dụng phương pháp nào
  • Ghi chú. Bài viết này không giải thích pass-by-value và pass-by-reference đối với các ngôn ngữ lập trình khác

Các kiểu dữ liệu nguyên thủy và không nguyên thủy

Trước khi hiểu về pass-by-value và pass-by-reference trong JavaScript, trước tiên, hãy hiểu kiểu dữ liệu nguyên thủy và không nguyên thủy là gì. Vì vậy, trong JavaScript, các loại dữ liệu được chia thành hai loại lớn

Các kiểu dữ liệu như chuỗi, số, null, không xác định, ký hiệu và boolean thuộc danh mục kiểu dữ liệu Nguyên thủy, trong khi tất cả các đối tượng, mảng và hàm thuộc danh mục kiểu dữ liệu không nguyên thủy hoặc tham chiếu

Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

Sự khác biệt chính giữa nguyên thủy và không nguyên thủy là nguyên thủy là bất biến. e. không có cách nào để thay đổi giá trị nguyên thủy sau khi nó được tạo, trong khi giá trị không nguyên thủy có thể thay đổi được. e. giá trị của một đối tượng có thể được thay đổi sau khi nó được tạo ra

Kiểu dữ liệu nguyên thủy

var string = 'Scaler Acadmey';
string[7] = 'a'
console.log(string) // 'Scaler Acadmey'

Ghi chú

let myString = "Scaler";
console.log(myString); // Scaler

myString = myString + " Academy";
console.log(myString); // Scaler Academy

Trong ví dụ nêu trên, đầu tiên, chúng tôi đã chỉ định Scaler cho 'myString', sau đó chúng tôi ghi nhật ký vào bảng điều khiển và chúng tôi có Scaler trong bảng điều khiển, sau đó chúng tôi thêm giá trị " Academy" vào giá trị hiện tại của 'myString' và trên bảng điều khiển ghi nhật ký giá trị đó . Bây giờ bạn có thể hỏi làm thế nào nó có thể xảy ra như tôi đã đề cập rằng các kiểu dữ liệu nguyên thủy là bất biến và chuỗi là một trong số đó?

Để hiểu điều này, chúng ta cần chia quá trình thành các bước

  1. Đầu tiên, giá trị hiện tại của "myString" được tìm nạp, tôi. e. , "Máy chia tỷ lệ"
  2. Sau đó, "Học viện" được thêm vào giá trị hiện có đó
  3. Sau đó, giá trị kết quả, tôi. e. "Scaler Academy" được phân bổ cho một khối bộ nhớ mới
  4. Bây giờ, đối tượng"myString" trỏ đến không gian bộ nhớ mới được tạo (không gian mà giá trị kết quả được phân bổ trong bước thứ ba)
  5. Bây giờ, không gian bộ nhớ được tạo trước đó (i. e. , không gian của myString = "String") có sẵn để thu gom rác

Do đó, chúng tôi đi đến kết luận rằng giá trị được tạo trước đó không bị sửa đổi, nó được gửi để thu gom rác trong khi biến 'myString' hiện trỏ đến không gian bộ nhớ mới được tạo, có "Scaler Academy" được lưu trữ trong đó. Nói một cách đơn giản, chúng ta chỉ tạo biến trỏ đến một không gian khác trong bộ nhớ và không sửa đổi giá trị hiện tại của nó

Các kiểu dữ liệu nguyên thủy được so sánh theo giá trị. Nếu hai giá trị giống nhau, thì chúng hoàn toàn bằng nhau

var num1 = 40;
var num2 = 40;
numb1 === num2; // true

var string1 = 'Scaler Academy';
var string2 = 'Scaler Academy';
string1 === string2; // true

Các kiểu dữ liệu không nguyên thủy

var myArray = [ 'Scaler', 'Academy' ];
myArray[1] = 'Topics';
console.log(myArray) // [ 'Scaler', 'Topics' ];

Các đối tượng và mảng không được so sánh theo giá trị. Điều đó có nghĩa là ngay cả khi hai đối tượng và mảng có cùng giá trị và thuộc tính hoặc cùng các phần tử tương ứng, chúng không hoàn toàn bằng nhau

var obj1 = { 
    'website': 'Scaler Topics',
    'topic': 'JavaScript'
    };
    
var obj2 = {    
   'website': 'Scaler Topics',
   'topic': 'JavaScript'
    };
    
obj1 === obj2;  // false

var myArray1 = [ 'Scaler', 'Topics' ];
var myArray2 = [ 'Scaler', 'Topics' ];
arr1 === arr2;  // false

Hai đối tượng hoàn toàn bằng nhau chỉ khi chúng đề cập đến cùng một đối tượng

var obj1 = { 
    'website': 'Scaler Topics',
    'topic': 'JavaScript'
    };
    
var obj2 = obj1;    
obj1 === obj2;  // true

Giá trị không nguyên thủy đôi khi còn được gọi là loại tham chiếu vì thay vì giá trị, chúng được so sánh bằng tham chiếu

Ghi chú. Trong JavaScript, các giá trị nguyên thủy được lưu trữ trên ngăn xếp, trong khi các giá trị không nguyên thủy được lưu trữ trong một đống

Truyền theo giá trị trong JavaScript

Truyền giá trị trong JavaScript có nghĩa là một bản sao giá trị của tham số thực tế được tạo trong bộ nhớ i. e. , cấp phát bộ nhớ mới được thực hiện và tất cả các thay đổi được thực hiện trong giá trị mới đó (i. e. , giá trị được sao chép). Giá trị ban đầu và giá trị được sao chép độc lập với nhau vì cả hai đều có một không gian khác nhau trong bộ nhớ i. e. , khi thay đổi giá trị bên trong hàm thì biến bên ngoài hàm không bị ảnh hưởng

Nói một cách đơn giản, chúng ta có thể hiểu là, trong một giá trị truyền vào, hàm nhận một bản sao của biến, bản sao này độc lập với biến ban đầu được truyền vào

Truyền giá trị trong JavaScript yêu cầu nhiều không gian hơn vì các hàm nhận được một bản sao của nội dung thực, do đó, một biến mới được tạo trong bộ nhớ

Trong khái niệm này, toán tử bằng đóng một vai trò lớn. Khi chúng ta tạo một biến, toán tử bằng sẽ thông báo xem bạn đang gán cho biến đó một giá trị nguyên thủy hay không nguyên thủy và sau đó hoạt động tương ứng

Ghi chú. Khi chúng ta sử dụng toán tử =, sẽ có một lệnh gọi hàm (đằng sau) trong đó việc chuyển giá trị (hoặc tham chiếu) trong JavaScript được thực hiện

Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

Khi chúng ta gán cho một biến một giá trị nguyên thủy, toán tử bằng sẽ thiết lập một khoảng trống (vị trí/địa chỉ) trong bộ nhớ (giả sử tại địa chỉ 2001) để lưu trữ dữ liệu của biến đó (num1) vào địa chỉ đó

Bây giờ, khi chúng ta tạo một biến mới num2 (ví dụ như ở địa chỉ 2002) và gán cho nó giá trị của biến trước đó num1, toán tử bằng sẽ tạo KHÔNG GIAN MỚI trong bộ nhớ độc lập với biến trước đó num1 có địa chỉ 2001 và đặt bản sao của nó . Do đó, điều này sao chép giá trị của biến ban đầu, num1, vào hai vị trí riêng biệt trong bộ nhớ (với địa chỉ 2001 và 2002)

________số 8_______

Ở đây, chúng tôi đã gán cho num1 một giá trị là 70. Điều này tạo ra một không gian trong bộ nhớ theo tên num1 và địa chỉ 2001 (giả định). Khi chúng ta tạo một biến num2 và gán cho nó giá trị của num1, thì toán tử equals thông báo rằng chúng ta đang xử lý một giá trị nguyên thủy, do đó nó tạo một KHÔNG GIAN MỚI trong bộ nhớ có địa chỉ 2002 và gán cho nó một bản sao của giá trị của num1, i. e. 70. Bây giờ chúng ta có thể thấy rằng cả hai biến đều có các khoảng trống khác nhau trong bộ nhớ và cả hai đều có giá trị là 70

Bây giờ, nếu chúng ta thay đổi giá trị của num1, thì num2 sẽ không có tác dụng vì nó có không gian riêng trong bộ nhớ và bây giờ nó không liên quan gì đến giá trị của num2 vì cả hai đều có không gian (địa chỉ) khác nhau trong bộ nhớ

Hãy hiểu điều này tốt hơn bằng một ví dụ khác

function multiplication(tmp) {
    tmp = tmp * 50;
    return tmp;
}
var num = 30;
var result = multiplication(num);
console.log(num); // 30
console.log(result); // 1500

Từ đoạn mã trên, chúng ta có thể thấy rằng hàm nhân nhận một đối số và thay đổi giá trị của nó

Sau đó, chúng tôi đã khai báo một biến num, với giá trị là 30

Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

Sau đó, chúng ta chuyển biến num cho hàm nhân. Javascript tự động sao chép giá trị của biến num sang biến tmp. Vì vậy, ở đây tmp là một biến mới được phân bổ một không gian mới trong bộ nhớ và độc lập với num

Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

Bây giờ tất cả các thay đổi do phép nhân hàm thực hiện được thực hiện trực tiếp với biến tmp;

Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

Điều này xảy ra vì một bản sao riêng biệt của biến num được tạo trong bộ nhớ có tên tmp với giá trị ban đầu là 30, giá trị này sau khi tính toán sẽ trở thành 1500

Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

tmp và num không có liên kết với nhau tôi. e. , chúng độc lập với nhau

Chuyển qua tham chiếu trong JavaScript

Không giống như truyền theo giá trị trong JavaScript, truyền theo tham chiếu trong JavaScript không tạo khoảng trống mới trong bộ nhớ, thay vào đó, chúng ta truyền tham chiếu/địa chỉ của tham số thực, nghĩa là hàm có thể truy cập giá trị ban đầu của biến. Như vậy, nếu chúng ta thay đổi giá trị của biến bên trong hàm thì giá trị ban đầu cũng bị thay đổi theo.

Nó không tạo bản sao, thay vào đó, nó hoạt động trên biến ban đầu, vì vậy tất cả các thay đổi được thực hiện bên trong hàm cũng ảnh hưởng đến biến ban đầu

Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

Không giống như giá trị truyền qua trong JavaScript, ở đây, khi toán tử bằng xác định rằng biến obj1 được đặt bằng một đối tượng, nó sẽ tạo một không gian bộ nhớ mới và trỏ obj1 đến 3005(giả định địa chỉ). Bây giờ, khi chúng ta tạo một biến mới, obj2 và gán nó với giá trị của obj1, toán tử bằng xác định rằng chúng ta đang xử lý các kiểu dữ liệu không nguyên thủy; . Do đó, chúng ta có thể thấy rằng không có không gian bộ nhớ mới nào được tạo ra thay vào đó, cả hai biến đều trỏ đến cùng một địa chỉ mà obj1 đã trỏ đến

let obj1 = {website: "Scaler Academy"}
let obj2 = obj1;

console.log(obj1)     // {website: "Scaler Academy"}
console.log(obj2)     // {website: "Scaler Academy"}

obj1.website = "Scaler Topics"

console.log(obj1)     // {website: "Scaler Topics"}
console.log(obj2)     // {website: "Scaler Topics"}

Trong ví dụ trên, chúng ta đã tạo một biến obj1 và đặt nó bằng một đối tượng, sau đó chúng ta đặt giá trị của biến obj2 khác bằng obj1

Vì toán tử bằng xác định rằng chúng ta đang xử lý các kiểu dữ liệu không nguyên thủy, nên thay vì tạo một không gian bộ nhớ mới, nó trỏ obj2 đến cùng một không gian bộ nhớ mà obj1 được trỏ đến. Do đó, khi chúng ta thay đổi (biến đổi) giá trị của obj1, thì giá trị của obj2 cũng bị thay đổi do obj2 cũng trỏ đến cùng một không gian bộ nhớ như obj1.

Chuyển qua Tham chiếu trong Đối tượng (có Chức năng)

let originalObj = {
name: "Scaler Academy",
rating: 4.5,
topic: "JavaScript"
};

function demo(tmpObj) { 
  tmpObj.rating = 5; 
  console.log(tmpObj.rating); 
} 

console.log(originalObj.rating);    // 4.5
demo(originalObj);             // 5
console.log(originalObj.rating);    //5

Từ ví dụ trên, chúng ta có thể thấy rằng khi thay đổi giá trị tmpObj thì giá trị của originalObj cũng bị thay đổi. Lý do cho điều này là khi chúng ta gọi demo và truyền đối tượng, thì originalObj được truyền theo tham chiếu của nó, vì vậy tham số cục bộ tempObj sẽ trỏ đến cùng một đối tượng mà chúng ta đã xác định, i. e. , bản gốcObj

Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

Vì vậy, trong trường hợp này, chúng tôi không xử lý hai bản sao độc lập thay vào đó, chúng tôi có các biến trỏ đến cùng một đối tượng, vì vậy, bất kỳ thay đổi nào được thực hiện đối với đối tượng này sẽ hiển thị với biến kia

Truyền tham chiếu trong một mảng (có chức năng)

let myString = "Scaler";
console.log(myString); // Scaler

myString = myString + " Academy";
console.log(myString); // Scaler Academy
0

Ở đây, khi chúng ta cố gắng thêm một mục mới vào mảng được lưu trữ trong tempArr, nó cũng ảnh hưởng đến mảng originalArr. Điều này xảy ra vì không có hai bản sao riêng biệt của một mảng, chúng tôi chỉ xử lý một mảng. Biến tempArr tham chiếu đến cùng một mảng đã được khởi tạo trong biến originalArr

Ví dụ này nói rằng, giống như đối tượng, trong mảng cũng vậy, khi thay đổi giá trị của tempArr, giá trị của originalArr sẽ tự động thay đổi

Do đó, chúng ta có thể kết luận bằng cách nói rằng tất cả các loại dữ liệu không nguyên thủy đều tương tác với nhau theo tham chiếu, vì vậy khi chúng ta đặt các giá trị của chúng bằng nhau hoặc chuyển chúng cho một hàm, thì tất cả chúng đều trỏ đến cùng một không gian bộ nhớ (địa chỉ) bất cứ khi nào chúng ta thay đổi

Cuộc gọi JavaScript theo giá trị hoặc tham chiếu

Khi nào nên sử dụng Pass by Value?

Như trong giá trị truyền qua trong JavaScript, một bản sao mới của biến được tạo và mọi thay đổi được thực hiện trong biến mới đều độc lập với biến ban đầu, vì vậy sẽ hữu ích khi chúng ta muốn theo dõi biến ban đầu và không'

Khi nào nên sử dụng Pass by Reference?

Khi chúng ta chuyển các đối số có kích thước lớn, tốt hơn là sử dụng tham chiếu chuyển tiếp trong JavaScript vì không có bản sao riêng biệt nào được tạo trong hàm được gọi, vì vậy bộ nhớ không bị lãng phí và do đó chương trình hiệu quả hơn

Là cuộc gọi JavaScript theo tham chiếu hoặc giá trị?

JavaScript luôn là giá trị truyền qua . Điều này có nghĩa là mọi thứ trong JavaScript là một loại giá trị và các đối số của hàm luôn được truyền theo giá trị. Điều đó đang được nói, các loại đối tượng khó hiểu hơn một chút. Sự nhầm lẫn nằm ở chỗ các loại đối tượng là các loại tham chiếu được truyền theo giá trị.

Có thể gọi theo tham chiếu trong JavaScript không?

Trong Truyền theo tham chiếu, một hàm được gọi bằng cách truyền trực tiếp tham chiếu/địa chỉ của biến làm đối số . Thay đổi đối số bên trong hàm ảnh hưởng đến biến được truyền từ bên ngoài hàm. Trong các đối tượng và mảng Javascript được truyền theo tham chiếu.

Các đối tượng JavaScript được truyền theo giá trị hay được truyền theo tham chiếu?

Trong mảng và đối tượng JavaScript theo sau chuyển theo thuộc tính tham chiếu . Trong Truyền theo tham chiếu, các tham số được truyền dưới dạng đối số không tạo bản sao của chính nó, nó đề cập đến giá trị ban đầu nên những thay đổi được thực hiện bên trong hàm sẽ ảnh hưởng đến giá trị ban đầu.

Cách gọi nào tốt hơn theo giá trị hoặc cách gọi theo tham chiếu?

Ngoài ra, trong hầu hết các trường hợp, bạn muốn dữ liệu ở chế độ riêng tư và ai đó đang gọi hàm chỉ có thể thay đổi nếu bạn muốn. Vì vậy, tốt hơn là sử dụng gọi theo giá trị theo mặc định và chỉ sử dụng gọi theo tham chiếu nếu dự kiến ​​có thay đổi dữ liệu .