├── .gitignore ├── README.md ├── config ├── constant.js ├── react-dom.development.js ├── runDemo.js └── runTest.js ├── jest.config.js ├── package.json ├── src ├── helper │ ├── homeworkHook.ts │ ├── renderTip │ │ ├── index.ts │ │ └── style.css │ ├── types.ts │ └── utils.ts ├── lib.dom.d.ts ├── vite-env.d.ts ├── 课程_1_工作原理概览 │ ├── demo │ │ └── App.tsx │ ├── homework │ │ ├── homework1.test.ts │ │ └── index.ts │ ├── index.html │ └── main.tsx ├── 课程_2_render阶段概览 │ ├── demo │ │ └── App.tsx │ ├── homework │ │ ├── homework2.test.tsx │ │ └── index.ts │ ├── index.html │ └── main.tsx └── 课程_3_状态更新原理 │ ├── demo │ └── App.tsx │ ├── homework │ ├── homework3.test.ts │ └── index.ts │ ├── index.html │ └── main.tsx ├── test.ts ├── tsconfig.json └── vite.config.ts /.gitignore: -------------------------------------------------------------------------------- 1 | node_modules/ 2 | .history -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | ## 安装 2 | 3 | ```sh 4 | npx react53 5 | ``` 6 | 7 | ## 使用方式 8 | 9 | 1. 执行命令,选择课程 10 | 11 | ```sh 12 | npm start 13 | # 或 14 | yarn start 15 | ``` 16 | 17 | 2. 在课程页面点击**顶部Tab**查看本节示例对应课程地址 18 | 19 | 3. 打开控制台查看日志信息 20 | 21 | 4. 学完后完成课程对应课后作业 22 | 23 | 5. 进群讨论,共同进步 24 | 25 | ## 课程更新 26 | 27 | 如果发现课程内容不是最新的,执行`git pull`拉取最新代码。 28 | 29 | ## 课程大纲 30 | 31 | ### 基础篇 32 | 33 | 1. 工作原理概览 34 | 2. render阶段概览 35 | 3. 状态更新原理 36 | 37 | ### 性能优化 38 | 39 | 4. 性能优化原理 40 | 5. 深入性能优化API 41 | 6. 性能优化实践 42 | 43 | ### Hooks 44 | 45 | 还在奋笔疾书... -------------------------------------------------------------------------------- /config/constant.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | LESSION_PREFFIX: '课程', 3 | LESSION_ENV: 'lession_env', 4 | LESSION_TEST_ARGV: 'lession', 5 | LESSION_TEST_ARGV_SHORT: 'l' 6 | } -------------------------------------------------------------------------------- /config/runDemo.js: -------------------------------------------------------------------------------- 1 | const path = require('path'); 2 | const fs = require('fs'); 3 | const {spawn} = require('child_process'); 4 | const util = require('util'); 5 | const rimraf = util.promisify(require('rimraf')); 6 | const inquirer = require('inquirer'); 7 | 8 | const constant = require('./constant'); 9 | 10 | async function hackReactDOM() { 11 | const pkgName = 'react-dom.development.js'; 12 | const pkgOriginPath = path.resolve(__dirname, `../node_modules/react-dom/cjs/${pkgName}`); 13 | const pkgTargetPath = path.resolve(__dirname, `./${pkgName}`); 14 | const viteCachePath = path.resolve(__dirname, '../node_modules/.vite'); 15 | // 清除vite缓存 16 | await rimraf(viteCachePath); 17 | // 替换ReactDOM 18 | const writeStream = await fs.createReadStream(pkgTargetPath).pipe(fs.createWriteStream(pkgOriginPath)); 19 | return new Promise(resolve => { 20 | writeStream.on('finish', resolve) 21 | }) 22 | } 23 | 24 | function parseLessionList() { 25 | const pkgOriginPath = path.resolve(__dirname, `../src`); 26 | return fs.readdirSync(pkgOriginPath).filter(name => name.startsWith(constant.LESSION_PREFFIX)); 27 | } 28 | 29 | inquirer.prompt([ 30 | { 31 | type: 'list', 32 | name: 'lession', 33 | message: '你想看哪个课程的Demo?', 34 | choices: parseLessionList() 35 | }, 36 | ]).then(async ({lession}) => { 37 | process.env[constant.LESSION_ENV] = lession; 38 | await hackReactDOM(); 39 | spawn('vite', [], {stdio: 'inherit', shell: true}); 40 | }) -------------------------------------------------------------------------------- /config/runTest.js: -------------------------------------------------------------------------------- 1 | const chalk = require('chalk'); 2 | const path = require('path'); 3 | const fs = require('fs'); 4 | const {spawn} = require('child_process'); 5 | const constant = require('./constant'); 6 | 7 | function parseLessionList() { 8 | const pkgOriginPath = path.resolve(__dirname, `../src`); 9 | return fs.readdirSync(pkgOriginPath) 10 | .filter(name => name.startsWith(constant.LESSION_PREFFIX)) 11 | .map(name => +name.split('_')[1]); 12 | } 13 | 14 | function checkLessionArg(num) { 15 | const lessionList = parseLessionList(); 16 | if (!num) { 17 | console.log(chalk.red('请输入课程序号,如:npm test 2 代表运行作业2的测试用例'), `所有课程序号包括:`, chalk.green(lessionList)); 18 | return false; 19 | } 20 | 21 | if (!lessionList.includes(num)) { 22 | console.log(chalk.red(`没有序号为${num}的课程`), `所有课程序号包括:`, chalk.green(lessionList)); 23 | return false; 24 | } 25 | return true; 26 | } 27 | 28 | const lessionNum = +process.argv.slice(2)[0]; 29 | const isValidNum = checkLessionArg(lessionNum); 30 | isValidNum && spawn(`jest`, [`${constant.LESSION_PREFFIX}_${lessionNum}`], {stdio: "inherit", shell: true}); 31 | -------------------------------------------------------------------------------- /jest.config.js: -------------------------------------------------------------------------------- 1 | const path = require('path'); 2 | const constant = require('./config/constant'); 3 | 4 | module.exports = { 5 | rootDir: path.resolve(__dirname), 6 | clearMocks: true, 7 | coverageDirectory: 'coverage', 8 | coverageProvider: 'v8', 9 | moduleFileExtensions: ['js', 'json', 'jsx', 'ts', 'tsx', 'node'], 10 | preset: 'ts-jest', 11 | testEnvironment: 'jsdom', 12 | // 测试文件 13 | testMatch: [`/src/${constant.LESSION_PREFFIX}*/**/*.test.ts?(x)`], 14 | 15 | transform: { 16 | '^.+\\js$': 'babel-jest', 17 | '^.+\\.(t|j)sx?$': 'ts-jest' 18 | } 19 | } -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "version": "0.0.0", 3 | "name": "react53-template", 4 | "scripts": { 5 | "start": "node ./config/runDemo.js", 6 | "test": "node ./config/runTest.js" 7 | }, 8 | "repository": { 9 | "type": "git", 10 | "url": "git+https://github.com/BetaSu/react53.git" 11 | }, 12 | "author": "KaSong", 13 | "license": "ISC", 14 | "bugs": { 15 | "url": "https://github.com/BetaSu/react53/issues" 16 | }, 17 | "homepage": "https://github.com/BetaSu/react53#readme", 18 | "dependencies": { 19 | "react": "^17.0.0", 20 | "react-dom": "17.0.2" 21 | }, 22 | "devDependencies": { 23 | "@types/jest": "^26.0.23", 24 | "@types/react": "^17.0.0", 25 | "@types/react-dom": "17.0.2", 26 | "@vitejs/plugin-react-refresh": "^1.3.1", 27 | "arg": "^5.0.0", 28 | "babel-jest": "27.0.6", 29 | "chalk": "^4.1.1", 30 | "fast-glob": "^3.2.6", 31 | "inquirer": "^8.1.1", 32 | "jest": "27.0.6", 33 | "jest-cli": "27.0.6", 34 | "rimraf": "^3.0.2", 35 | "ts-jest": "^27.0.3", 36 | "typescript": "^4.3.2", 37 | "vite": "2.3.8" 38 | } 39 | } 40 | -------------------------------------------------------------------------------- /src/helper/homeworkHook.ts: -------------------------------------------------------------------------------- 1 | import {log, getType2Use, exist, createCounter} from './utils'; 2 | let hackErrorEmit = false; 3 | 4 | type THomeworkCB = (...args: any) => void; 5 | 6 | function checkHack() { 7 | if (!window.hackReactDOM && !hackErrorEmit) { 8 | hackErrorEmit = true; 9 | console.warn('[错误]请重新执行 yarn start(或 npm start)'); 10 | } 11 | } 12 | 13 | function bindHook(lession: number, hookType: string, callback: THomeworkCB) { 14 | type2Hook[lession] = type2Hook[lession] || {}; 15 | type2Hook[lession][hookType] = callback; 16 | } 17 | 18 | const type2Hook: {[lession: string]: {[lessionType: string]: THomeworkCB}} = {}; 19 | 20 | window.homeworkHook = function(type, ...args) { 21 | checkHack(); 22 | const lessionMap = type2Hook[window.lession]; 23 | const cb = lessionMap?.[type]; 24 | cb?.(...args); 25 | } 26 | 27 | bindHook(1, 'workLoopSync', () => { 28 | log('#727205', '开始render阶段,计算本次更新带来的副作用'); 29 | }) 30 | bindHook(1, 'commitRoot', () => { 31 | log('#04048a', '开始commit阶段,执行副作用'); 32 | }) 33 | bindHook(1, 'commitPlacement', (fiber) => { 34 | log('brown', `执行副作用:${getType2Use(fiber)}插入`, fiber); 35 | }) 36 | bindHook(1, 'commitWork', (fiber) => { 37 | log('brown', `执行副作用:${getType2Use(fiber)}更新属性`, fiber); 38 | }) 39 | bindHook(1, 'commitDeletion', (fiber) => { 40 | log('brown', `执行副作用:${getType2Use(fiber)}删除DOM`, fiber); 41 | }) 42 | 43 | bindHook(2, 'workLoopSync', () => { 44 | log('#727205', '开始深度优先遍历(render阶段开始)'); 45 | }) 46 | bindHook(2, 'commitRoot', () => { 47 | log('#04048a', 'commit阶段,执行副作用...'); 48 | }) 49 | // bindHook(2, 'beginWork', (_, wip) => { 50 | // log('green', `递 ${getType2Use(wip)}`); 51 | // }) 52 | // bindHook(2, 'completeWork', (_, wip) => { 53 | // log('red', `归 ${getType2Use(wip)}`); 54 | // }) 55 | // bindHook(2, 'reconcileChildren', (wip) => { 56 | // log('green', `创建${getType2Use(wip)}对应fiberNode:`, wip); 57 | // }) 58 | bindHook(2, 'beginWork', (_, wip) => { 59 | log('green', `创建${getType2Use(wip)}对应fiberNode:`, wip); 60 | }) 61 | // bindHook(2, 'appendAllChildren', (wip, ins) => { 62 | // log('purple', '生成DOM节点', ins); 63 | // }) 64 | 65 | bindHook(3, 'workLoopSync', () => { 66 | log('#727205', '开始深度优先遍历(render阶段开始)'); 67 | }) 68 | bindHook(3, 'commitRoot', () => { 69 | log('#04048a', 'commit阶段,执行副作用...'); 70 | }) 71 | bindHook(3, 'dispatchAction', (wip) => { 72 | const who = getType2Use(wip); 73 | log('blue', `${who}触发了更新,创建Update:`, wip.memoizedState.queue.pending); 74 | }) 75 | bindHook(3, 'mountState', (hook, wip) => { 76 | const who = getType2Use(wip); 77 | log('blue', `${who}的useState根据初始state计算出新state:`, hook.memoizedState); 78 | }) 79 | bindHook(3, 'updateReducer', (hook, wip) => { 80 | const who = getType2Use(wip); 81 | log('blue', `${who}的useState根据Update计算出新state:`, hook.memoizedState); 82 | }) 83 | bindHook(3, 'beginWork', (_, wip) => { 84 | log('green', `创建${getType2Use(wip)}对应fiberNode`); 85 | 86 | const update = wip.memoizedState?.queue?.pending; 87 | exist(update) && log('#04048a', `在${getType2Use(wip)}中发现Update:`, update); 88 | 89 | const pendingProps = wip.pendingProps; 90 | exist(pendingProps) && Object.keys(pendingProps).length && log('brown', 'props', pendingProps); 91 | }) 92 | bindHook(3, 'commitPlacement', (fiber) => { 93 | log('brown', `执行副作用:${getType2Use(fiber)}插入`, fiber); 94 | }) 95 | bindHook(3, 'commitWork', (fiber) => { 96 | log('brown', `执行副作用:${getType2Use(fiber)}更新属性`, fiber); 97 | }) 98 | bindHook(3, 'commitDeletion', (fiber) => { 99 | log('brown', `执行副作用:${getType2Use(fiber)}删除DOM`, fiber); 100 | }) 101 | 102 | 103 | 104 | // bindHook(4, 'renderWithHooks', (wip) => { 105 | // const cpn = getType2Use(wip); 106 | // log('red', `${cpn}开始render`, wip); 107 | // }) 108 | // bindHook(4, 'mountState', (hook) => { 109 | // log('green', `执行useState,数据为:`, hook); 110 | // }) 111 | // bindHook(4, 'updateState', (hook) => { 112 | // log('green', `执行useState,数据为:`, hook); 113 | // }) 114 | // bindHook(4, 'updateReducer', (hook) => { 115 | // log('green', `执行useReducer,数据为:`, hook); 116 | // }) 117 | // bindHook(4, 'mountReducer', (hook) => { 118 | // log('green', `执行useReducer,数据为:`, hook); 119 | // }) 120 | // bindHook(4, 'mountEffect', (hook) => { 121 | // log('green', `执行useEffect,数据为:`, hook); 122 | // }) 123 | // bindHook(4, 'updateEffect', (hook) => { 124 | // log('green', `执行useEffect,数据为:`, hook); 125 | // }) 126 | // bindHook(4, 'mountRef', (hook) => { 127 | // log('green', `执行useRef,数据为:`, hook); 128 | // }) 129 | // bindHook(4, 'updateRef', (hook) => { 130 | // log('green', `执行useRef,数据为:`, hook); 131 | // }) 132 | // bindHook(4, 'mountCallback', (hook) => { 133 | // log('green', `执行useCallback,数据为:`, hook); 134 | // }) 135 | // bindHook(4, 'updateCallback', (hook) => { 136 | // log('green', `执行useCallback,数据为:`, hook); 137 | // }) 138 | // bindHook(4, 'mountMemo', (hook) => { 139 | // log('green', `执行useMemo,数据为:`, hook); 140 | // }) 141 | // bindHook(4, 'updateMemo', (hook) => { 142 | // log('green', `执行useMemo,数据为:`, hook); 143 | // }) 144 | 145 | export {}; -------------------------------------------------------------------------------- /src/helper/renderTip/index.ts: -------------------------------------------------------------------------------- 1 | 2 | import './style.css'; 3 | 4 | interface TTipData { 5 | lession: number; 6 | title: string; 7 | mainUrl: string; 8 | } 9 | 10 | export default ({title, lession, mainUrl}: TTipData): void => { 11 | document.title = title; 12 | window.lession = lession; 13 | const ele = document.createElement('section'); 14 | ele.className = 'tip-zone'; 15 | ele.innerHTML = ` 16 | 33 | `; 34 | const target = document.querySelector('#root'); 35 | target?.parentNode?.insertBefore(ele, target); 36 | } -------------------------------------------------------------------------------- /src/helper/renderTip/style.css: -------------------------------------------------------------------------------- 1 | #root { 2 | background: rgb(0, 0, 225, .2); 3 | padding: 20px; 4 | border-radius: 0 0 20px 20px; 5 | } 6 | #root::before { 7 | display: block; 8 | content: '示例区域'; 9 | color: white; 10 | font-weight: bolder; 11 | } 12 | 13 | .tip-zone { 14 | padding:10px 10px 10px 20px; 15 | border-radius: 20px 20px 0 0; 16 | background: skyblue; 17 | } 18 | 19 | #nav-drawer { 20 | position: relative; 21 | } 22 | 23 | .nav-unshown { 24 | display:none; 25 | } 26 | 27 | #nav-open { 28 | display: inline-block; 29 | cursor: pointer; 30 | width: 30px; 31 | height: 22px; 32 | vertical-align: middle; 33 | } 34 | 35 | #nav-open span, #nav-open span:before, #nav-open span:after { 36 | position: absolute; 37 | height: 3px; 38 | width: 25px; 39 | border-radius: 3px; 40 | background: #fff; 41 | display: block; 42 | content: ''; 43 | cursor: pointer; 44 | } 45 | #nav-open span:before { 46 | bottom: -8px; 47 | } 48 | #nav-open span:after { 49 | bottom: -16px; 50 | } 51 | 52 | #nav-close { 53 | display: none; 54 | position: fixed; 55 | z-index: 99; 56 | top: 0; 57 | left: 0; 58 | width: 100%; 59 | height: 100%; 60 | background: black; 61 | opacity: 0; 62 | transition: .3s ease-in-out; 63 | } 64 | 65 | #nav-content { 66 | overflow: auto; 67 | position: fixed; 68 | top: 0; 69 | left: 0; 70 | z-index: 9999; 71 | width: 90%; 72 | max-width: 330px; 73 | height: 100%; 74 | background: #fff; 75 | transition: .3s ease-in-out; 76 | -webkit-transform: translateX(-105%); 77 | transform: translateX(-105%); 78 | } 79 | 80 | #nav-title { 81 | color: #fff; 82 | font-weight: bolder; 83 | cursor: pointer; 84 | } 85 | 86 | #nav-input:checked ~ #nav-close { 87 | display: block; 88 | opacity: .5; 89 | } 90 | 91 | #nav-input:checked ~ #nav-content { 92 | -webkit-transform: translateX(0%); 93 | transform: translateX(0%); 94 | box-shadow: 6px 0 25px rgba(0,0,0,.15); 95 | } 96 | 97 | #nav-content ol { 98 | padding-right: 20px; 99 | } 100 | 101 | .highlight { 102 | color: #ea4335; 103 | } 104 | 105 | .qrcode-magician { 106 | width: 160px; 107 | height: 160px; 108 | background-size: contain; 109 | background-image: url(); 110 | } -------------------------------------------------------------------------------- /src/helper/types.ts: -------------------------------------------------------------------------------- 1 | export type TChoice = () => number; 2 | -------------------------------------------------------------------------------- /src/helper/utils.ts: -------------------------------------------------------------------------------- 1 | interface TWIP { 2 | type: any; 3 | tag: number; 4 | memoizedProps: any; 5 | } 6 | 7 | export const getType2Use = (wip: TWIP): string => { 8 | const {type, tag, memoizedProps} = wip; 9 | switch (tag) { 10 | case 3: 11 | return '根组件'; 12 | case 2: 13 | return `${type.name}(函数组件或类组件)`; 14 | case 1: 15 | return `${type.name}(类组件)`; 16 | case 0: 17 | return `${type.name}(函数组件)`; 18 | case 6: 19 | return `${memoizedProps}(文本组件)` 20 | default: 21 | return type; 22 | } 23 | } 24 | 25 | export const log = (color: string, label: string, message?: any) => { 26 | console.log( 27 | `%c ${label} %c`, 28 | `background-color: ${color}; color: #FFFFFF`, 29 | `background-color: inherit; color: inherit` 30 | , exist(message) ? message : ''); 31 | } 32 | 33 | export const exist = (data: any) => data !== undefined && data !== null; 34 | 35 | export const getReadableAnswer = (count: number, answer: number) => { 36 | return `第${count + 1}题,答案:${answer}`; 37 | } 38 | 39 | export const createCounter = (initialCount = 0) => { 40 | let count = initialCount; 41 | return { 42 | add() { 43 | count++; 44 | }, 45 | reset() { 46 | count = initialCount; 47 | }, 48 | get() { 49 | return count; 50 | } 51 | } 52 | } -------------------------------------------------------------------------------- /src/lib.dom.d.ts: -------------------------------------------------------------------------------- 1 | declare interface Window { 2 | hackReactDOM: boolean; 3 | homeworkHook: (type: string, ...args: any) => void; 4 | lession: number; 5 | } -------------------------------------------------------------------------------- /src/vite-env.d.ts: -------------------------------------------------------------------------------- 1 | /// 2 | -------------------------------------------------------------------------------- /src/课程_1_工作原理概览/demo/App.tsx: -------------------------------------------------------------------------------- 1 | import React, {useState} from 'react' 2 | 3 | function App() { 4 | const [count, update] = useState(0); 5 | const add = () => { 6 | if (count > 0) { 7 | update(count - 1); 8 | } 9 | }; 10 | 11 | const del = () => update(count + 1); 12 | 13 | return ( 14 |
15 |

提莫队长正在送命

16 | 17 | 18 |
    19 | {Array(count).fill('🍄').map((what, i) =>
  • {what}
  • )} 20 |
21 |
22 | ) 23 | } 24 | 25 | export default App 26 | -------------------------------------------------------------------------------- /src/课程_1_工作原理概览/homework/homework1.test.ts: -------------------------------------------------------------------------------- 1 | import homework from '.'; 2 | import {getReadableAnswer} from '../../helper/utils'; 3 | 4 | 5 | test('课程一选择题批改作业', () => { 6 | const answer = [1, 1, 1, 1, 1, 2].map((rightAnswer, i) => getReadableAnswer(i, rightAnswer)); 7 | homework.forEach((cur, i) => { 8 | const curAnswer = getReadableAnswer(i, cur()); 9 | const rightAnswer = answer[i]; 10 | expect(curAnswer).toBe(rightAnswer); 11 | }) 12 | }); -------------------------------------------------------------------------------- /src/课程_1_工作原理概览/homework/index.ts: -------------------------------------------------------------------------------- 1 | import {TChoice} from '../../helper/types'; 2 | 3 | /** 4 | * 今天的课后作业是选择题 5 | * 做完了执行 npm test 1 批改作业 6 | */ 7 | 8 | export default [ 9 | // 这是个选择题使用例子 10 | () => { 11 | // 问题:卡颂帅么? 12 | // 1. 帅 13 | // 2. 不帅 14 | return 1; /** return你的答案 */ 15 | }, 16 | () => { 17 | // 问题:React的更新流程分为render阶段和commit阶段 18 | // 1. 对 19 | // 2. 错 20 | return /** 你的答案 */; 21 | }, 22 | () => { 23 | // 问题:更新流程是如何触发的? 24 | // 1. ReactDOM.render、this.setState、useState的setter... 25 | // 2. 通过定时器定时触发 26 | return /** 你的答案 */; 27 | }, 28 | () => { 29 | // 问题:什么是副作用? 30 | // 1. DOM的增/删/改(属性更新)等... 31 | // 2. 吃了感冒药下午瞌睡 32 | return /** 你的答案 */; 33 | }, 34 | () => { 35 | // 问题:render阶段的作用? 36 | // 1. 找出本次更新产生的副作用 37 | // 2. 执行本次更新的副作用 38 | return /** 你的答案 */; 39 | }, 40 | () => { 41 | // 问题:commit阶段的作用? 42 | // 1. 找出本次更新产生的副作用 43 | // 2. 执行本次更新的副作用 44 | return /** 你的答案 */; 45 | } 46 | ] as TChoice[]; 47 | -------------------------------------------------------------------------------- /src/课程_1_工作原理概览/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | React53 8 | 9 | 10 |
11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /src/课程_1_工作原理概览/main.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import ReactDOM from 'react-dom' 3 | import App from './demo/App' 4 | import '../helper/homeworkHook'; 5 | import renderTip from '../helper/renderTip'; 6 | 7 | renderTip({ 8 | lession: 1, 9 | title: 'React53', 10 | mainUrl: 'https://react53.iamkasong.com/chart1.html' 11 | }); 12 | 13 | ReactDOM.render( 14 | , 15 | document.getElementById('root') 16 | ) -------------------------------------------------------------------------------- /src/课程_2_render阶段概览/demo/App.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | 3 | 4 | function App() { 5 | 6 | return ( 7 |
8 |

卡颂喜欢的水果:

9 |
    10 |
  • 🍎
  • 11 |
  • 🍌
  • 12 |
  • 🍉
  • 13 |
14 |
15 | ) 16 | } 17 | 18 | export default App 19 | -------------------------------------------------------------------------------- /src/课程_2_render阶段概览/homework/homework2.test.tsx: -------------------------------------------------------------------------------- 1 | import * as React from "react"; 2 | import {reconciler, choiceList} from '.'; 3 | import {getReadableAnswer} from '../../helper/utils'; 4 | 5 | test('课程二选择题批改作业', () => { 6 | const answer = [2, 1, 2].map((rightAnswer, i) => getReadableAnswer(i, rightAnswer)); 7 | choiceList.forEach((cur, i) => { 8 | const curAnswer = getReadableAnswer(i, cur()); 9 | const rightAnswer = answer[i]; 10 | expect(curAnswer).toBe(rightAnswer); 11 | }) 12 | }); 13 | 14 | test('深度优先遍历', () => { 15 | 16 | const jsx = ( 17 |
18 |

19 | 20 |
    21 |
  • 22 |
  • 23 |
24 |
25 | ) 26 | 27 | const result = reconciler(jsx); 28 | const answer = ['div', 'p', 'span', 'ul', 'li', 'p', 'li', 'p']; 29 | expect(result).toEqual(answer); 30 | }); 31 | 32 | -------------------------------------------------------------------------------- /src/课程_2_render阶段概览/homework/index.ts: -------------------------------------------------------------------------------- 1 | import React, { FC, ReactNode } from 'react'; 2 | import {TChoice} from '../../helper/types'; 3 | 4 | 5 | /** 6 | * 今天的课后作业是选择题和应用题 7 | * 做完了执行 npm test 2 批改作业 8 | */ 9 | 10 | export const choiceList = [ 11 | () => { 12 | // 问题:render阶段的遍历顺序是? 13 | // 1. 广度优先遍历 14 | // 2. 深度优先遍历 15 | return /** 你的答案 */; 16 | }, 17 | () => { 18 | // 问题:对于如下JSX结构: 19 | /** 20 | *
21 | *

hello

22 | *
    23 | *
  • 24 | *
25 | *
26 | */ 27 | // p生成对应fiberNode后,下一个要生成fiberNode的节点是? 28 | // 1. span 29 | // 2. ul 30 | return /** 你的答案 */; 31 | }, 32 | () => { 33 | // 问题:对于如下JSX结构: 34 | /** 35 | *
36 | *

hello

37 | *
    38 | *
  • 39 | *
40 | *
41 | */ 42 | // hello生成对应fiberNode后,下一个要生成fiberNode的节点是? 43 | // 1. div 44 | // 2. ul 45 | return /** 你的答案 */; 46 | } 47 | ] as TChoice[]; 48 | 49 | 50 | /** 51 | * 应用题 使用深度优先遍历实现一个遍历器 52 | * 接收一段JSX结构作为参数,按与React同样的“递”与“归”流程,将遍历到的element.type push到数组中,并返回数组 53 | * 如: 传入: 54 |
55 | 56 |
    57 |
  • 58 |
  • 59 |
60 |
61 | 返回: ['div', 'span', 'ul', 'li', 'li'] 62 | 注:你可以在这个链接看到JSX编译执行后的数据结构:https://codesandbox.io/s/jsx-playground-forked-tzpqj 63 | * 编写完成后执行 npm test 2 跑用例 64 | */ 65 | export const reconciler = (jsx: React.ReactElement): string[] => { 66 | // 请开始你的表演 67 | return []; 68 | } 69 | 70 | 71 | 72 | 73 | // 参考答案 74 | // export const reconciler = (jsx: React.ReactElement) => { 75 | // const result: string[] = []; 76 | 77 | // if (!jsx) { 78 | // return result; 79 | // } 80 | 81 | // const stack = [jsx]; 82 | 83 | // while (stack.length) { 84 | // const node = stack.pop(); 85 | // if (typeof node?.type === 'string') { 86 | // result.push(node.type); 87 | // } 88 | 89 | // const children = node?.props.children; 90 | // if (children) { 91 | // if (Array.isArray(children)) { 92 | // for (let i = children.length - 1; i >= 0; i--) { 93 | // stack.push(children[i]); 94 | // } 95 | // } else { 96 | // stack.push(children); 97 | // } 98 | // } 99 | // } 100 | 101 | // return result; 102 | // } 103 | -------------------------------------------------------------------------------- /src/课程_2_render阶段概览/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | React53 8 | 9 | 10 |
11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /src/课程_2_render阶段概览/main.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import ReactDOM from 'react-dom' 3 | import App from './demo/App' 4 | import '../helper/homeworkHook'; 5 | import renderTip from '../helper/renderTip'; 6 | 7 | renderTip({ 8 | lession: 2, 9 | title: 'React53', 10 | mainUrl: 'https://react53.iamkasong.com/chart2.html' 11 | }); 12 | 13 | ReactDOM.render( 14 | , 15 | document.getElementById('root') 16 | ) 17 | -------------------------------------------------------------------------------- /src/课程_3_状态更新原理/demo/App.tsx: -------------------------------------------------------------------------------- 1 | import React, { useState } from 'react' 2 | 3 | function App() { 4 | return ( 5 | 6 | ) 7 | } 8 | 9 | function GrandParent() { 10 | return ( 11 | 12 | ) 13 | } 14 | 15 | function Parent() { 16 | const [num, updateNum] = useState(0); 17 | return ; 18 | } 19 | 20 | function Child({num, onChange}: {num: number; onChange: (num: number) => void;}) { 21 | return
onChange(num + 1)}>{num}
; 22 | } 23 | 24 | export default App 25 | -------------------------------------------------------------------------------- /src/课程_3_状态更新原理/homework/homework3.test.ts: -------------------------------------------------------------------------------- 1 | import {homework, insertUpdate, TUpdate, TFiberNode} from '.'; 2 | import {getReadableAnswer} from '../../helper/utils'; 3 | 4 | test('课程3选择题批改作业', () => { 5 | const answer = [2, 2, 2, 1, 2].map((rightAnswer, i) => getReadableAnswer(i, rightAnswer)); 6 | homework.forEach((cur, i) => { 7 | const curAnswer = getReadableAnswer(i, cur()); 8 | const rightAnswer = answer[i]; 9 | expect(curAnswer).toBe(rightAnswer); 10 | }) 11 | }); 12 | 13 | test('插入update成功且数据结构正确且形成环状链表', () => { 14 | const fiberNode: TFiberNode = { 15 | updateQueue: undefined 16 | } 17 | const firstAction = 0; 18 | insertUpdate(fiberNode, firstAction); 19 | const {updateQueue} = fiberNode; 20 | let curUpdate = updateQueue as TUpdate; 21 | expect(curUpdate).toBeDefined(); 22 | expect(curUpdate.action).toBe(firstAction); 23 | expect(curUpdate.next).toBe(curUpdate); 24 | }) 25 | 26 | test('插入2个update形成正确的环状链表', () => { 27 | const fiberNode: TFiberNode = { 28 | updateQueue: undefined 29 | } 30 | for (let i = 0; i < 2; i++) { 31 | insertUpdate(fiberNode, i); 32 | } 33 | const {updateQueue} = fiberNode; 34 | let curUpdate = (updateQueue as TUpdate).next as TUpdate; 35 | let curRightAction = 0; 36 | const firstAction = 0; 37 | do { 38 | expect(curUpdate.action).toBe(curRightAction++); 39 | curUpdate = curUpdate.next as TUpdate; 40 | } while (curUpdate.action !== firstAction) 41 | }) 42 | 43 | test('插入多个update形成正确的环状链表', () => { 44 | const fiberNode: TFiberNode = { 45 | updateQueue: undefined 46 | } 47 | for (let i = 0; i < 10; i++) { 48 | insertUpdate(fiberNode, i); 49 | } 50 | const {updateQueue} = fiberNode; 51 | let curUpdate = (updateQueue as TUpdate).next as TUpdate; 52 | let curRightAction = 0; 53 | const firstAction = 0; 54 | do { 55 | expect(curUpdate.action).toBe(curRightAction++); 56 | curUpdate = curUpdate.next as TUpdate; 57 | } while (curUpdate.action !== firstAction) 58 | }) -------------------------------------------------------------------------------- /src/课程_3_状态更新原理/homework/index.ts: -------------------------------------------------------------------------------- 1 | import {TChoice} from '../../helper/types'; 2 | /** 3 | * 今天的课后作业是选择题和应用题 4 | * 做完了执行 npm test 3 批改作业 5 | */ 6 | 7 | export const homework = [ 8 | () => { 9 | // 问题:调用this.setState触发更新后,只有被更新组件及其子孙组件会执行遍历流程? 10 | // 1. 对 11 | // 2. 错 12 | return /** 你的答案 */; 13 | }, 14 | () => { 15 | // 问题:state的最新值需要计算么? 16 | // 1. 不需要,this.setState(x)中x就是最新值,直接使用就行 17 | // 2. 需要,state需要根据Update计算得出 18 | return /** 你的答案 */; 19 | }, 20 | () => { 21 | // 问题:被更新的state是在什么时候被计算的? 22 | // 1. 触发更新那一刻 23 | // 2. 遍历流程进行到state所在组件对应fiber时 24 | return /** 你的答案 */; 25 | }, 26 | () => { 27 | // 问题:为什么在触发更新后,有些组件没有render? 28 | // 1. 虽然更新是全局概念,但是React内置了遍历时的优化策略,以及提供了性能优化API(如:PureComponent),命中这些策略的组件不会render 29 | // 2. 因为更新是局部概念,只有触发更新的组件及其子孙组件才会render 30 | return /** 你的答案 */; 31 | }, 32 | () => { 33 | // 问题:一次更新,只有触发更新的组件对应DOM会变化? 34 | // 1. 是,因为这次更新只在该组件对应fiber上创建了Update 35 | // 2. 不一定,如果其他组件对应fiber上也存在Update并计算出新的state,可能也会产生对应DOM变化 36 | return /** 你的答案 */; 37 | } 38 | ] as TChoice[]; 39 | 40 | 41 | /** 42 | * 应用题,请实现向fiberNode插入update的方法insertUpdate 43 | * 其中update是一条环状链表,满足: 44 | 新插入的update是链表中最后一个update 45 | fiberNode.updateQueue指向最后一个update(即新插入这个) 46 | 最后一个update下一个(即.next)指向第一个update 47 | 插入update后函数返回fiberNode 48 | 如: 插入第一个update:u1 49 | 此时: fiberNode.updateQueue = u1 50 | u1.next = u1 51 | 插入第二个update:u2 52 | fiberNode.updateQueue = u2 53 | u2.next = u1; u1.next = u2; 54 | 插入第三个update:u3 55 | fiberNode.updateQueue = u3 56 | u3.next = u1; u1.next = u2; u2.next = u3; 57 | */ 58 | export interface TUpdate { 59 | action: any; 60 | next?: TUpdate; 61 | } 62 | export interface TFiberNode { 63 | updateQueue?: TUpdate; 64 | } 65 | 66 | export function insertUpdate(fiberNode: TFiberNode, action: any): TFiberNode { 67 | const update: TUpdate = { 68 | action, 69 | next: undefined 70 | }; 71 | 72 | // 请开始你的表演 73 | 74 | return fiberNode; 75 | } 76 | 77 | 78 | 79 | 80 | 81 | // 参考答案 82 | // export function insertUpdate(fiberNode: TFiberNode, action: any): TFiberNode { 83 | // const update: TUpdate = { 84 | // action, 85 | // next: undefined 86 | // }; 87 | 88 | // const lastUpdate = fiberNode.updateQueue; 89 | 90 | // if (!lastUpdate) { 91 | // // fiberNode上还不存在update,插入的update与自己形成环状链表 92 | // update.next = update; 93 | // } else { 94 | // // fiberNode上存在update 95 | // // 新插入的作为最后一个,指向第一个 96 | // update.next = lastUpdate.next; 97 | // // 之前的最后一个指向新插入这个 98 | // lastUpdate.next = update; 99 | // } 100 | // // updateQueue指向新插入这个 101 | // fiberNode.updateQueue = update; 102 | 103 | // return fiberNode; 104 | // } -------------------------------------------------------------------------------- /src/课程_3_状态更新原理/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | React53 8 | 9 | 10 |
11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /src/课程_3_状态更新原理/main.tsx: -------------------------------------------------------------------------------- 1 | import React from 'react' 2 | import ReactDOM from 'react-dom' 3 | import App from './demo/App' 4 | import '../helper/homeworkHook'; 5 | import renderTip from '../helper/renderTip'; 6 | 7 | renderTip({ 8 | lession: 3, 9 | title: 'React53', 10 | mainUrl: 'https://react53.iamkasong.com/chart3.html' 11 | }); 12 | 13 | ReactDOM.render( 14 | , 15 | document.getElementById('root') 16 | ) 17 | -------------------------------------------------------------------------------- /test.ts: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/BetaSu/react53/9f4256294171c1c740847fe02810ffc183b53cf0/test.ts -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "target": "ESNext", 4 | "lib": ["DOM", "DOM.Iterable", "ESNext"], 5 | "allowJs": false, 6 | "skipLibCheck": false, 7 | "esModuleInterop": false, 8 | "allowSyntheticDefaultImports": true, 9 | "strict": true, 10 | "forceConsistentCasingInFileNames": true, 11 | "module": "ESNext", 12 | "moduleResolution": "Node", 13 | "resolveJsonModule": true, 14 | "isolatedModules": true, 15 | "noEmit": true, 16 | "jsx": "react" 17 | }, 18 | "include": ["./src"] 19 | } 20 | -------------------------------------------------------------------------------- /vite.config.ts: -------------------------------------------------------------------------------- 1 | import { defineConfig } from 'vite' 2 | import reactRefresh from '@vitejs/plugin-react-refresh' 3 | import constant from './config/constant'; 4 | 5 | const curLession = process.env[constant.LESSION_ENV]; 6 | if (!curLession) { 7 | console.error('错误的课程目录:', curLession); 8 | } 9 | 10 | // https://vitejs.dev/config/ 11 | export default defineConfig({ 12 | root: `./src/${curLession}`, 13 | plugins: [reactRefresh()] 14 | }) 15 | --------------------------------------------------------------------------------