Bạn đã thực sự hiểu generator function?
Hello anh em, hình như blog chưa có bài viết nào về #ES6 cả hôm nay mình sẽ quết định viết về generator function vì thành thật mà nói mình cũng chưa thực sự hiểu thằng này lắm :3, tiện đây mình sẽ giới thiệu một số ứng dụng cũng như tiềm năng của tính năng này trong các tác vu thực tế.
cái nhìn đầu tiên
Một trong những tính năng mình thấy tú vị nhất trong ES6 - ECMAScript 2015 chính là generator function, khi lướt qua các tính năng của ES6 và bắt gặp tính năng này suy nghĩ đầu tiên trong đầu mình chính là…
cái WTF gì thế, bác Allen Wirfs-Brock đúng là rảnh rỗi, tại sao lại sáng tạo ra cái thứ này trong khi callback với promisse là làm async tốt rồi, iteritor thì for với while là được, cái máy phát điện này chả phải là đồ bỏ đi hay sao. trong khi tình hình tình hình giang hồ đang loạn tiễn phân tâm như nhế này bớt học được cái gì thi hay cái đó :’‑(
hôm nay dạo qua một vòng google thấy cái thứ bỏ đi này cũng lợi hại ra phết. để tìm hiểu tại sao lại có cái này mình sẽ đi từ nề tảng của nó, iterator.
Iterator
Iterator chính là thứ mà ai cũng dùng nhưng không phải ai cũng biết đến nó (đừng tin nó lừa đấy 😠 ), cũng giống trong các ngôn ngữ khác iterator là một đối tượng dùng để truy cập vào một dãy các phần tử, mỗi lần nó sẽ trả lại phầ tử tiếp theo trong dãy và giữ được tứ tự duyệt mỗi lần nó được gọi đến. javascript iterator là một đói tượng cung cấp phương thức next()
trả lại một đối tượng gồm hai thuộc tính là done
kiểm tra xem dãy đã duyệt hết chưa và value
gía trị phần tử tiếp theo trong dãy.
1 | function makeIterator(array){ |
Một khi đã được khởi tạo chúng ta có thể gọi next
để duyệt qua thằng tiếp theo.
1 |
|
Thấy quen quen chưa nào? chuẩn cơm mẹ nấu rối, đây chính chính là cha đẻ của function *
chứ gì nữa 😎.
Generator
Iterator là một pattern thực sự hữu dụng nhưng khá lắt léo trong cài đặt và không dễ dàng để maintain được các trạng tái của nó, đến đây mới thấy đúng là JS đang thiếu một cú pháp chuẩn cho pattrens này nể khai thác hết tiềm năng của nó, và… à, vâng xin trân trọng giới thiệu Generator function còn được biết đến với biệt danh function *
hay yield function
:
1 |
|
có thể hiểu generator là một loại function tạo ra iterator, một function được gọi là generator nếu bên trong có chứa một hay nhiểu biểu thức yield
và được khai báo bằng từ khóa function *
:
1 |
|
nếu trong
function *
có nhiều giá trịyield
thì mỗi lần gọinext()
biểu thứcyield
tiếp theo sẽ được thực hiện, giống như là hàm bị đóng băng lại chờ lần gọinext
tiếp theo vậy. sử dụng cái này mà tràn lan thì có thể dẫn đến khó debug và quản lý state, hãy thử tưởng tượng trong hàm của bạn có vài lệnh lặp, vài lệnh rẽ nhanh và vài cáiyield
thì…
Built-in iterables
Còn nhớ ở trên mình nói là bạn dùng iterator rồi không 😎, không đúng đâu nhé, bạn không chỉ dùng rồi mà còn dùng nó hăng ngày nữa. các đối tượng trong javavascript như String
, Array
, TypedArray
, Map
và Set
đều có built-in iterables để duyệt các phần tử, bởi vì prototype
của các đối tượng này đều có phương thức Symbol.iterator
.
các biểu thức với iterables
một vài biểu thức thao tác với iterator bao gồm vòng lặp for-of, spread operator, yield, và destructuring assignment.
1 |
|
yield*
cũng như function *
nói chung là thằng nào cứ có thêm cai ""* là như cầm cờ trong tay rồi cờ đến tay ai người đó phất thôi 😄. biểu thức yield
nhận vào một iterator
khác hoặc một mảng khi chạy đến nó sẽ lần lượt chạy qua các giá trị của yield*
rồi mới tới các yield
tiếp theo.
1 |
|
nói nôm na thì nó có vẻ giống với callback 😄.
Kết Luận
À thôi để thay cho lời kết mình xin đưa ra một số ứng dụng thực tế của thằng này nhé:
1Async / Await
bạn đã nghe đến async/await chưa, nếu chưa thì hãy tập làm quên đi vì tính năng này sẽ được đưa vào ES7 đấy, và tin mình đi, nó thực sự đáng đồng tiền bát gạo đấy:
code của bạn từ thế này
1 |
|
sẽ biến thành thế này:
1 |
|
một chút tương tác
một trong những điểm thú vị của generator là hàm next có thể nhận lại một đối số để quết định nên làm gì tiếp theo, bạn sẽ thích ví dụ này cho mà xem ❤️ :
1 |
|
Generator + Promise
một sự kết hợp hoàn hảo, có lẽ nào 2 thứ này sinh ra để giành co nhau:
1 |
|
Observables và còn nhiều nữa…
@minhlv with ❤️