You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Copy file name to clipboardexpand all lines: css-tips.md
+2-1
Original file line number
Diff line number
Diff line change
@@ -100,8 +100,9 @@ button {
100
100
101
101
## Background image opacity
102
102
103
-
Apply opacity to absolute psuedo element, parent element use isolation: isolate
103
+
Apply opacity to absolute psuedo element, parent element use `isolation: isolate`, that will create a stacking context and prevent absolute children with negatives `z-index` to break out of parents
104
104
<https://www.youtube.com/watch?v=lRPguPbovro>
105
+
<https://www.youtube.com/watch?v=o1HzOJfgugE>
105
106
106
107
## Replace `position: absolute` with `display: grid`
Dependancy injection is a popular pattern used to remove hardcoded dependancy between units in an application. The aim of this technique is to make our code more flexible and scalable
6
+
7
+
So what's hardcoded dependancy?
5
8
Hardcode dependancy means class A is tightly coupled to class B, class B is called inside class A. There are some issue with this:
6
9
7
10
- It's harder for testing: Let's say we want to test class A. And because class A is using class B, it's really hard to mock class B
Copy file name to clipboardexpand all lines: interview.md
+21-2
Original file line number
Diff line number
Diff line change
@@ -70,6 +70,10 @@ If you mutate the data directly, the components might not re-render that could l
70
70
71
71
With `useEffect`, we can control how the side effects run in the component. Normally, the side effects make the state change. So, if you put them outside the `useEffec` hook, it will lead to infinite re-render of the component.
72
72
73
+
## Why we the callback in `useEffect` mustn't be async function
74
+
75
+
Because async function return promise. `useEffect` hook isn't expecting us to return a promise. It expects us to return either nothing (like we are above), or a cleanup function.
76
+
73
77
## When react does not re-render
74
78
75
79
<https://www.zhenghao.io/posts/react-rerender>
@@ -296,6 +300,7 @@ The difference happens when the `success` callback returns a **rejected promise*
- User A authen thành công ➜ server response token ➜ User A bị Man-in-the-middle attack, do User A bất cẩn làm token rơi vào tay hacker ➜ hacker chiếm quyền truy cập account của user A mà server không hề hay biết
377
382
- User A đang truy cập các private API, đột nhiên token expired, màn hình đột ngột bị rediect sang trang login vì server response 403
378
-
- server chưa cài đặt cơ chế hủy token, phía database thì account user A đã bị ban, nhưng token vẫn đang lưu rải rác tại các client (desktop app, mobile app, web, server khác, ...), client vẫn cứ gửi token, server encode payload token thấy vẫn còn hạn sử dụng, phần mã hash chữ ký vẫn hợp lệ ➜ user vẫn author thành công
383
+
- server chưa cài đặt cơ chế hủy token, phía database thì account user A đã bị ban, nhưng token vẫn đang lưu rải rác tại các client (desktop app, mobile app, web, server khác, ...), client vẫn cứ gửi token, server decode payload token thấy vẫn còn hạn sử dụng, phần mã hash chữ ký vẫn hợp lệ ➜ user vẫn author thành công
384
+
385
+
Cách logout jwt
386
+
387
+
- Khi gọi API logout thì lưu token đã logout vào blacklist.
388
+
- Trong các API được authen, khi có request đến, check token có nằm trong blacklist hay không, nếu có thì coi như unauthorize.
389
+
- Bài toán đặt ra là lưu blacklist ở đâu để việc đọc token đảm bảo được về mặt hiệu suất khi mà tần số gọi API dày đặc như vậy? Câu trả lời là lưu trong redis. Mặc khác vì jwt token nó có thời hạn sống ngắn, nên hoàn toàn có thể clean up dữ liệu, xoá các token đã thực sự hết hạn trong redis.
379
390
380
391
## Debounce và throttle
381
392
@@ -583,7 +594,7 @@ const copied = new Map(myMap);
583
594
584
595
## Else if using operation tenerry
585
596
586
-
const a = b ? c : d ? e : f
597
+
`consta= b ? c : d ? e : f`
587
598
588
599
## OOP
589
600
@@ -593,6 +604,7 @@ const a = b ? c : d ? e : f
593
604
594
605
Là cơ chế của server cho phép các origin khác (origin = scheme(protocol) + domain + port) ngoài chính nó gửi request.
- Là tất cả các outer scope cộng với inner scope của hàm đó.
644
657
- Outer scope là môi trường bên ngoài nơi định nghĩa của 1 hàm.
@@ -676,14 +689,18 @@ Primitive type là immutable. Ta chỉ có thể gán lại biến đấy sang 1
676
689
Reference type là mutable. Nếu ta copy biến đấy sang 1 biến khác, 2 biến đấy sẽ trỏ vào cùng 1 box chứ 2 biến không trỏ lẫn sang nhau. Và nếu 1 trong 2 biến mutate (ví dụ như là thay đổi giá trị của property) thì biến còn lại cũng thay đổi theo (vì vẫn chung 1 box)
677
690
678
691
Notes:
692
+
679
693
- Các giá trị dạng reference là riêng biệt:
694
+
680
695
Mỗi khi ta khai báo 1 hàm hoặc 1 object, `() => {}` hoặc `{}` thì 1 box mới được tạo ra
681
696
```js
682
697
consta= () => {}
683
698
constb= () => {}
684
699
console.log( a === b ) // return false.
685
700
```
701
+
686
702
- Truyền tham số dạng reference có thể xảy ra Mutation không mong muốn
703
+
687
704
```javascript
688
705
functionminimum(array) {
689
706
array.sort();
@@ -746,6 +763,7 @@ Có 3 loại scope: `global scope`, `local scope` (function) và `block scope` (
746
763
## for-in and for-of
747
764
748
765
`for-in`:
766
+
749
767
- iterate over enumerable properties of an object. enumerable properties are the "key" of the property the Object or "index" of the Array.
750
768
If the propery is defined with the `enumerable` false, it is not interated
Copy file name to clipboardexpand all lines: promise.md
+66-1
Original file line number
Diff line number
Diff line change
@@ -80,19 +80,23 @@ Both take an array of promises
80
80
## Async
81
81
82
82
**Syntax**:
83
+
83
84
```javascript
84
85
asyncfunction() {
85
86
};
86
87
```
88
+
87
89
Put in front of function, then **returns a promise**. This promise will be resolved with the **value** returned by the async function or rejected with an error that is thrown inside async function
88
90
89
91
If the **value** is not a promise, it will be wrapped in promise.
92
+
90
93
```javascript
91
94
asyncfunctionfoo() {
92
95
return1
93
96
}
94
97
```
95
98
...is similar to:
99
+
96
100
```javascript
97
101
functionfoo() {
98
102
returnPromise.resolve(1)
@@ -117,17 +121,19 @@ async function run() {
117
121
}
118
122
```
119
123
120
-
121
124
## Await
122
125
123
126
Can only be used inside `async` function.
124
127
**Syntax**:
128
+
125
129
```javascript
126
130
constvalue=await expression;
127
131
```
132
+
128
133
`expression` is a `Promise` or any value. If a value, it convert to `Promise.resolve(expression)`, which is a `Promise`;
129
134
130
135
`Await` pause the `async` function ultil a `Promise` is fulfilled or rejected.
136
+
131
137
- If rejected, `await` throw an error.
132
138
- If fulfilled, it resumes the `async` function execution and **return the value of the fulfilled Promise**
133
139
@@ -136,3 +142,62 @@ You can catch the error right after await, without using `try...catch`
Copy file name to clipboardexpand all lines: react-query.md
+21
Original file line number
Diff line number
Diff line change
@@ -295,6 +295,27 @@ function Todo({ todoId }) {
295
295
}
296
296
```
297
297
298
+
## Handle errors
299
+
300
+
While most utilities like `axios` automatically throw errors for unsuccessful HTTP calls, some utilities like fetch do not throw errors by default. If that's the case, you'll need to throw them on your own. Here is a simple way to do that with the popular fetch API
0 commit comments