Reactivity là gì? Học nhanh Proxy và Reflect qua ví dụ thực tế
🎉 Nếu anh em thấy Front-End cạnh tranh quá thì chúng ta học thêm Back-End thôi.
Mình đang có khóa học🏆 Node.js Super, với các kiến thức như: Express.js, TypeScript, MongoDB, Socket.io, Docker, AWS, Swagger, ...
Vậy nên anh em có thể mua ngay từ bây giờ để tăng khả năng pass phỏng vấn nhé 😉
Bây giờ là lúc để đi tìm hiểu sâu hơn vào bên trong các framework JS. Một trong những tính năng đặc biệt nhất của Vue hay Angular là hệ thống reactivity. Khi bạn thay đổi state thì view sẽ tự động được cập nhật. Điều này làm cho việc quản lý state trở nên dễ dàng và trực quan.
React không phải là thư viện reactive nhé. Bạn thay đổi state và gọi
setState
thì view mới cập nhật được, tức là bạn phải cập nhật bằng tay.
🥇Reactivity là gì?
Reactivity là một mô hình lập trình cho phép chúng ta phản ứng lại những sự thay đổi biến. Excel sheet dưới đây là một ví dụ kinh điển về nó.
Nếu bạn nhập số 2 ở ô thứ nhất, số 3 ở ô thứ 2 và thực hiện SUM
, sheet sẽ cho bạn giá trị tổng. Không có bất ngờ nào ở đây. Nhưng nếu bạn update giá trị thứ nhất, SUM
cũng sẽ tự động update lại.
Javascript thì thường không hoạt động giống như vậy. Cùng viết thứ một đoạn bằng JS nhé.
let val1 = 2let val2 = 3let sum = val1 + val2console.log(sum) // 5val1 = 3console.log(sum) // Still 5
Nếu chúng ta update giá trị đầu tiên, sum
sẽ không bị thay đổi.
Vậy chúng ta làm điều này trong Javascript như thế nào?
Tạo một function handle
và gọi lại bằng tay mỗi khi val1
hoặc val2
thay đổi ư, quá mệt mỏi!
let val1 = 2let val2 = 3let sumconst handle = () => {sụm = val1 + val2}handle()console.log(sum) // 5val1 = 3sum()console.log(sum) // 6
Có một vài thứ chúng ta có thể làm là:
- Giám sát các giá trị. Ví dụ
val1
+val2
thì giám sát cả 2val1
vàval2
- Phát hiện khi một giá trị thay đổi. Ví dụ chúng ta gán
val1 = 3
. - Chạy lại code. Ví dụ run
sum = val1 + val2
lại để cập nhật giá trịsum
.
Và đây là cách giải quyết đơn giản.
let data = {val1: 2,val2: 3}let sumconst handle = () => {sum = data.val1 + data.val2}handle()Object.keys(data).forEach((key) => {let internalValue = data[key]Object.defineProperty(data, key, {get() {return internalValue},set(newValue) {internalValue = newValuehandle()}})})console.log(sum) // 5data.val1 = 4console.log(sum) // 7
Đầu tiên mình sẽ đưa những giá trị mình cần giám sát vào trong object data
. Mình sẽ dùng Object.defineProperty để chuyển các thuộc tính dữ liệu của data
thành thuộc tính bộ truy cập (getter / setter đó).
Các bạn có thể đọc thêm Property getters and setters để hiểu rõ hơn.
Và mỗi khi mình thực hiện thay đổi thuộc tính thì setter sẽ chạy và function handle()
được thực thi.
Thế thôi, đơn giản mà đúng không
Nhưng ES6 cung cấp cho bạn một tính năng hay hơn chuyên để xử lý những vấn đề này, đó là Proxy.
🥇Proxy là gì?
Proxy là một object thường được sử dụng để chỉnh sửa các hành vi của các toán tử cơ bản cho object.
Proxy thường được coi như là một trung gian giữa object và các hành vi gán, sửa, xóa… trên object.
Proxy được tạo với 2 tham số:
target
: object gốc mà bạn muốn proxyhandler
: một object chứa cáctrap
.Trap
là các phương thức để đáp ứng lại các thao tác dữ liệu trên target.
Quay trở lại bài toán bên trên, ta có thể giải quyết bằng proxy như sau
let data = {val1: 2,val2: 3}let sumconst handle = () => {sum = data.val1 + data.val2}handle()console.log(sum) // 5const proxy = new Proxy(data, {get(target, prop) {return target[prop]},set(obj, prop, value) {obj[prop] = valuehandle()// Phải return true để cho Engine biết là gán hoàn thành// Nếu return false thì trong strict-mode sẽ bị lỗi TypeErrorreturn true}})proxy.val1 = 3console.log(sum) // 6
Chúng ta cũng có thể xóa get()
đi cũng được.
const proxy = new Proxy(data, {set(obj, prop, value) {obj[prop] = valuehandle()return true}})
get()
và set()
bên trên được gọi là các trap
.
Dưới đây là danh sách các trap theo trang mozilla
handler.apply()
: A trap for a function call.handler.construct()
: A trap for the new operator.handler.defineProperty()
: A trap for Object.defineProperty.handler.deleteProperty()
: A trap for the delete operator.handler.get()
: A trap for getting property values.handler.getOwnPropertyDescriptor()
: A trap for Object.getOwnPropertyDescriptor.handler.getPrototypeOf()
: A trap for Object.getPrototypeOf.handler.has()
: A trap for the in operator.handler.isExtensible()
: A trap for Object.isExtensible.handler.ownKeys()
: A trap for Object.getOwnPropertyNames and Object.getOwnPropertySymbols.handler.preventExtensions()
: A trap for Object.preventExtensions.handler.set()
: A trap for setting property values.handler.setPrototypeOf()
: A trap for Object.setPrototypeOf.
🥇Tạo thử Reactive DOM như Vue 3
Nếu bạn đã code Vue thì bạn sẽ thấy hệ thống reactivity của Vue khá xịn, thay đổi data là nó sẽ giúp chúng ta cập nhật DOM luôn. Bây giờ mình sẽ mô phỏng lại tính năng đó thử xem sao nhé.
Khá là hay ho :mrgreen: .
Cho bạn nào chưa biết thì Vue 2 dùng Object.defineProperty
, còn Vue 3 đã nâng cấp lên dùng Proxy
rồi.
Nhắc đến Proxy
thì không thể nào quên Reflect
được, cùng mình tìm hiểu thử Reflect
là gì nhé.
🥇Reflect là gì?
Reflect là một object được cung cấp sẵn trong Javascript thường được dùng để phối hợp với proxy handler
. Không như hầu hết các global object thì Reflect không phải là một constructor function, vì thế không thể tạo mới object với toán tử new
.
Reflect cung cấp cho chúng ta các phương thức tương tự như bên Object.
const target = {name: 'Dư Thanh Được'}// Tương tự Object.definePropertyconst result = Reflect.defineProperty(target, 'age', { value: 25 })if (result) {// Thành công} else {// Thất bại}
Các phương thức của Reflect cũng tương ứng với các trap của Proxy, quay trở lại bài toán tính sum
ban đầu, nếu áp dụng Reflect vào thì code sẽ như thế này.
const proxy = new Proxy(data, {get(...props) {Reflect.get(...props)},set(...props) {const result = Reflect.set(...props)handle()return result}})
🥇Tóm lại
Hy vọng bài viết ngắn này của mình giúp ích cho các bạn hiểu rõ hơn về Reactivity cũng như Proxy và Reflect.
Dạo này mình cũng cố gắng siêng hơn một chút, ra bài viết nhiều hơn một chút nên mọi người đón đọc nhé. Hẹn mọi người ở các bài viết tiếp theo 😀
🥇Tham khảo
Khóa học ReactJs giúp bạn chinh phục mức lương 25 - 30 triệu/tháng
Phew! Cuối cùng bạn cũng đã đọc xong. Bài viết này có hơi dài một tí vì mình muốn nó đầy đủ nhất có thể 😅
Chúng ta đều hiểu rằng Javascript và React không hề dễ, chúng có quá nhiều concept cần phải học. Mình cũng cảm thấy nó khó! Nay lại có thể Typescript nữa 🥲, thật sự khó nuốt.
Nhưng đừng lo: Bạn có thể nắm vững các kiến thức trên chỉ trong một khóa học ReactJs Super - Shopee Clone Typescript
Mình đã bắt đầu code React vào năm 2019, và nó đã trở thành thư viện ưa thích của mình để xây dựng UI và web app. Mình cũng đã làm việc với nhiều framework khác như Angular, Vue nhưng thực sự chỉ có React là đem lại cho mình cảm xúc và sự hiệu quả. 💓
Nếu bạn đang gặp khó khăn với React, mình ở đây để giúp bạn!
Mình đã dành hơn 6 tháng để phát triển khóa học ReactJs Super - Shopee Clone Typescript. Trong khóa này bạn sẽ được học mọi thứ về thư viện ReactJs, các kiến thức từ cơ bản cho đến nâng cao nhất, mục đích của mình là giúp bạn chinh phục mức lương 25 - 30 triêu/tháng
Nếu bạn cảm thấy bài viết này của mình hữu ích, mình nghĩ bạn sẽ thích hợp với phong cách dạy của mình. Không như bài viết này, khóa học là sự kết hợp giữa các bài viết, video, quizz, bài tập nhỏ và dự án lớn có thể xin việc được ngay. Học xong mình đảm bảo bạn sẽ lên tay ngay. 💪🏻