花花外贸 發表於 2022-6-24 19:51:00

快速入门React(基础)

<h1 id="react基础">React基础</h1>
<h2 id="react介绍">React介绍</h2>
<p><code>目标任务:</code>了解什么是React以及它的特点</p>
<p><strong>React是什么</strong></p>
<p>​   一个专注于构建用户界面的 JavaScript 库,和vue和angular并称前端三大框架,不夸张的说,react引领了很多新思想,世界范围内是最流行的js前端框架,最近发布了18版本,加入了很多新特性</p>
<p>​        React英文文档(https://reactjs.org/)</p>
<p>​        React中文文档 (https://zh-hans.reactjs.org/)</p>
<p>​        React新文档        (https://beta.reactjs.org/)(开发中....)</p>
<p><strong>React有什么特点</strong></p>
<ol>
<li>
<p>声明式UI(JSX)</p>
<p>写UI就和写普通的HTML一样,抛弃命令式的繁琐实现</p>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194157966-731544014.png" alt="" loading="lazy"></p>
</li>
<li>
<p>组件化</p>
<p>组件是react中最重要的内容,组件可以通过搭积木的方式拼成一个完整的页面,通过组件的抽象可以增加复用能力和提高可维护性</p>
</li>
<li>
<p>一次学习,跨平台编写</p>
<p>react既可以开发web应用也可以使用同样的语法开发原生应用(react-native),比如安卓和ios应用,甚至可以使用react开发VR应用,想象力空间十足,react更像是一个 <code>元框架</code>为各种领域赋能</p>
</li>
</ol>
<h2 id="环境初始化">环境初始化</h2>
<p><code>目标任务:</code>   能够独立使用React脚手架创建一个react项目</p>
<h3 id="1-使用脚手架创建项目">1. 使用脚手架创建项目</h3>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194250755-731039753.png" alt="" loading="lazy"></p>
<ul>
<li>
<p>打开命令行窗口</p>
</li>
<li>
<p>执行命令</p>
<pre><code class="language-bash">npx create-react-app react-basic
</code></pre>
<p>说明:</p>
<ol>
<li>npx create-react-app 是固定命令,<code>create-react-app</code>是React脚手架的名称</li>
<li>react-basic表示项目名称,可以自定义,保持语义化</li>
<li>npx 命令会帮助我们临时安装create-react-app包,然后初始化项目完成之后会自自动删掉,所以不需要全局安装create-react-app</li>
</ol>
</li>
<li>
<p>启动项目</p>
<pre><code class="language-bash">yarn start
or
npm start
</code></pre>
</li>
</ul>
<h3 id="2-项目目录说明调整">2. 项目目录说明调整</h3>
<ul>
<li>
<p>目录说明</p>
<ol>
<li>
<p><code>src</code> 目录是我们写代码进行项目开发的目录</p>
</li>
<li>
<p><code>package.json</code>中俩个核心库:react 、react-dom</p>
</li>
</ol>
</li>
<li>
<p>目录调整</p>
<ol>
<li>删除src目录下自带的所有文件,只保留app.js根组件和index.js</li>
<li>创建index.js文件作为项目的入口文件,在这个文件中书写react代码即可</li>
</ol>
</li>
<li>
<p>入口文件说明</p>
<pre><code class="language-jsx">import React from 'react'
import ReactDOM from 'react-dom'
import './index.css'
// 引入根组件App
import App from './App'
// 通过调用ReactDOM的render方法渲染App根组件到id为root的dom节点上
ReactDOM.render(
&lt;React.StrictMode&gt;
    &lt;App /&gt;
&lt;/React.StrictMode&gt;,
document.getElementById('root')
)
</code></pre>
</li>
</ul>
<h2 id="jsx基础">JSX基础</h2>
<h3 id="1-jsx介绍">1. JSX介绍</h3>
<p><code>目标任务:</code>   能够理解什么是JSX,JSX的底层是什么</p>
<p>概念:JSX是 JavaScript XML(HTML)的缩写,表示在 JS 代码中书写 HTML 结构</p>
<p>作用:在React中创建HTML结构(页面UI结构)</p>
<p>优势:</p>
<ol>
<li>采用类似于HTML的语法,降低学习成本,会HTML就会JSX</li>
<li>充分利用JS自身的可编程能力创建HTML结构</li>
</ol>
<p>注意:JSX 并不是标准的 JS 语法,是 JS 的语法扩展,浏览器默认是不识别的,脚手架中内置的 @babel/plugin-transform-react-jsx 包,用来解析该语法</p>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194317343-2044047554.png" alt="" loading="lazy"></p>
<h3 id="2-jsx中使用js表达式">2. JSX中使用js表达式</h3>
<p><code>目标任务:</code>   能够在JSX中使用表达式</p>
<p><strong>语法</strong></p>
<p><code>{ JS 表达式 }</code></p>
<pre><code class="language-jsx">const name = '码农权'

&lt;h1&gt;你好,我叫{name}&lt;/h1&gt;   //    &lt;h1&gt;你好,我叫柴柴&lt;/h1&gt;
</code></pre>
<p><strong>可以使用的表达式</strong></p>
<ol>
<li>字符串、数值、布尔值、null、undefined、object( [] / {} )</li>
<li>1 + 2、'abc'.split('')、['a', 'b'].join('-')</li>
<li>fn()</li>
</ol>
<p><strong>特别注意</strong></p>
<p>​        if 语句/ switch-case 语句/ 变量声明语句,这些叫做语句,不是表达式,不能出现在 <code>{}</code> 中!!</p>
<h3 id="3-jsx列表渲染">3. JSX列表渲染</h3>
<p><code>目标任务:</code>   能够在JSX中实现列表渲染</p>
<blockquote>
<p>页面的构建离不开重复的列表结构,比如歌曲列表,商品列表等,我们知道vue中用的是v-for,react这边如何实现呢?</p>
</blockquote>
<p>实现:使用数组的<code>map</code> 方法</p>
<p>案例:</p>
<pre><code class="language-jsx">// 来个列表
const songs = [
{ id: 1, name: '痴心绝对' },
{ id: 2, name: '像我这样的人' },
{ id: 3, name: '南山南' }
]

function App() {
return (
    &lt;div className="App"&gt;
      &lt;ul&gt;
      {
          songs.map(item =&gt; &lt;li&gt;{item.name}&lt;/li&gt;)
      }
      &lt;/ul&gt;
    &lt;/div&gt;
)
}

export default App
</code></pre>
<p>注意点:需要为遍历项添加 <code>key</code> 属性</p>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194353863-1844781770.png" alt="" loading="lazy"></p>
<ol>
<li>key 在 HTML 结构中是看不到的,是 React 内部用来进行性能优化时使用</li>
<li>key 在当前列表中要唯一的字符串或者数值(String/Number)</li>
<li>如果列表中有像 id 这种的唯一值,就用 id 来作为 key 值</li>
<li>如果列表中没有像 id 这种的唯一值,就可以使用 index(下标)来作为 key 值</li>
</ol>
<h3 id="4-jsx条件渲染">4. JSX条件渲染</h3>
<p><code>目标任务:</code>   能够在JSX中实现条件渲染</p>
<p>作用:根据是否满足条件生成HTML结构,比如Loading效果</p>
<p>实现:可以使用 <code>三元运算符</code> 或   <code>逻辑与(&amp;&amp;)运算符</code></p>
<p>案例:</p>
<pre><code class="language-jsx">// 来个布尔值
const flag = true
function App() {
return (
    &lt;div className="App"&gt;
      {/* 条件渲染字符串 */}
      {flag ? 'react真有趣' : 'vue真有趣'}
      {/* 条件渲染标签/组件 */}
      {flag ? &lt;span&gt;this is span&lt;/span&gt; : null}
    &lt;/div&gt;
)
}
export default App
</code></pre>
<h3 id="5-jsx样式处理">5. JSX样式处理</h3>
<p><code>目标任务:</code>   能够在JSX中实现css样式处理</p>
<ul>
<li>
<p>行内样式 - style</p>
<pre><code class="language-jsx">function App() {
return (
    &lt;div className="App"&gt;
      &lt;div style={{ color: 'red' }}&gt;this is a div&lt;/div&gt;
    &lt;/div&gt;
)
}

export default App
</code></pre>
</li>
<li>
<p>行内样式 - style - 更优写法</p>
<pre><code class="language-jsx">const styleObj = {
    color:red
}

function App() {
return (
    &lt;div className="App"&gt;
      &lt;div style={ styleObj }&gt;this is a div&lt;/div&gt;
    &lt;/div&gt;
)
}

export default App
</code></pre>
</li>
<li>
<p>类名 - className(推荐)</p>
<p><code>app.css</code></p>
<pre><code class="language-css">.title {
font-size: 30px;
color: blue;
}
</code></pre>
<p><code>app.js</code></p>
<pre><code class="language-jsx">import './app.css'

function App() {
return (
    &lt;div className="App"&gt;
      &lt;div className='title'&gt;this is a div&lt;/div&gt;
    &lt;/div&gt;
)
}
export default App
</code></pre>
</li>
<li>
<p>类名 - className - 动态类名控制</p>
<pre><code class="language-jsx">import './app.css'
const showTitle = true
function App() {
return (
    &lt;div className="App"&gt;
      &lt;div className={ showTitle ? 'title' : ''}&gt;this is a div&lt;/div&gt;
    &lt;/div&gt;
)
}
export default App
</code></pre>
</li>
</ul>
<h3 id="6-jsx注意事项">6. JSX注意事项</h3>
<p><code>目标任务:</code>   掌握JSX在实际应用时的注意事项</p>
<ol>
<li>JSX必须有一个根节点,如果没有根节点,可以使用<code>&lt;&gt;&lt;/&gt;</code>(幽灵节点)替代</li>
<li>所有标签必须形成闭合,成对闭合或者自闭合都可以</li>
<li>JSX中的语法更加贴近JS语法,属性名采用驼峰命名法<code>class -&gt; className</code><code>for -&gt; htmlFor</code></li>
<li>JSX支持多行(换行),如果需要换行,需使用<code>()</code> 包裹,防止bug出现</li>
</ol>
<h2 id="格式化配置">格式化配置</h2>
<p><code>目标任务:</code>   基于vscode配置格式化工具,提高开发效率</p>
<ol>
<li>
<p>安装vsCode prettier插件</p>
</li>
<li>
<p>修改配置文件 <code>setting.json</code></p>
<pre><code class="language-json">{
"git.enableSmartCommit": true,
// 修改注释颜色
"editor.tokenColorCustomizations": {
    "comments": {
      "fontStyle": "bold",
      "foreground": "#82e0aa"
    }
},
// 配置文件类型识别
"files.associations": {
    "*.js": "javascript",
    "*.json": "jsonc",
    "*.cjson": "jsonc",
    "*.wxss": "css",
    "*.wxs": "javascript"
},
"extensions.ignoreRecommendations": false,
"files.exclude": {
    "**/.DS_Store": true,
    "**/.git": true,
    "**/.hg": true,
    "**/.svn": true,
    "**/CVS": true,
    "**/node_modules": false,
    "**/tmp": true
},
// "javascript.implicitProjectConfig.experimentalDecorators": true,
"explorer.confirmDragAndDrop": false,
"typescript.updateImportsOnFileMove.enabled": "prompt",
"git.confirmSync": false,
"editor.tabSize": 2,
"editor.fontWeight": "500",
"": {},
"editor.tabCompletion": "on",
"vsicons.projectDetection.autoReload": true,
"editor.fontFamily": "Monaco, 'Courier New', monospace, Meslo LG M for Powerline",
"": {
    "editor.defaultFormatter": "vscode.html-language-features"
},
"editor.fontSize": 16,
"debug.console.fontSize": 14,
"vsicons.dontShowNewVersionMessage": true,
"editor.minimap.enabled": true,
"emmet.extensionsPath": [
    ""
],
// vue eslint start 保存时自动格式化代码
"editor.formatOnSave": true,
// eslint配置项,保存时自动修复错误
"editor.codeActionsOnSave": {
    "source.fixAll": true
},
"vetur.ignoreProjectWarning": true,
// 让vetur使用vs自带的js格式化工具
// uni-app和vue 项目使用
"vetur.format.defaultFormatter.js": "vscode-typescript",
"javascript.format.semicolons": "remove",
// // 指定 *.vue 文件的格式化工具为vetur
"": {
    "editor.defaultFormatter": "octref.vetur"
},
// // 指定 *.js 文件的格式化工具为vscode自带
"": {
    "editor.defaultFormatter": "vscode.typescript-language-features"
},
// // 默认使用prettier格式化支持的文件
"editor.defaultFormatter": "esbenp.prettier-vscode",
"prettier.jsxBracketSameLine": true,
// 函数前面加个空格
"javascript.format.insertSpaceBeforeFunctionParenthesis": true,
"prettier.singleQuote": true,
"prettier.semi": false,
// eslint end
// react
// 当按tab键的时候,会自动提示
"emmet.triggerExpansionOnTab": true,
"emmet.showAbbreviationSuggestions": true,
"emmet.includeLanguages": {
    // jsx的提示
    "javascript": "javascriptreact",
    "vue-html": "html",
    "vue": "html",
    "wxml": "html"
},
// end
"": {
    "editor.defaultFormatter": "vscode.json-language-features"
},
// @路径提示
"path-intellisense.mappings": {
    "@": "${workspaceRoot}/src"
},
"security.workspace.trust.untrustedFiles": "open",
"git.ignoreMissingGitWarning": true,
"window.zoomLevel": 1
}
</code></pre>
</li>
</ol>
<h2 id="阶段小练习">阶段小练习</h2>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194426297-102969837.png" alt="" loading="lazy"></p>
<p>练习说明</p>
<ol>
<li>
<p>拉取准备好的项目模块到本地 ,安装依赖,run起来项目</p>
<p>https://gitee.com/react-course-series/react-jsx-demo</p>
</li>
<li>
<p>按照图示,完成 <code>评论数据渲染</code><code>tab内容渲染</code><code>评论列表点赞和点踩</code>三个视图渲染</p>
</li>
</ol>
<h1 id="react组件基础">React组件基础</h1>
<h2 id="组件概念">组件概念</h2>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194444450-354906220.png" alt="" loading="lazy"></p>
<h2 id="函数组件">函数组件</h2>
<p><code>目标任务:</code>   能够独立使用函数完成react组件的创建和渲染</p>
<p><strong>概念</strong></p>
<blockquote>
<p>使用 JS 的函数(或箭头函数)创建的组件,就叫做<code>函数组件</code></p>
</blockquote>
<p><strong>组件定义与渲染</strong></p>
<pre><code class="language-jsx">// 定义函数组件
function HelloFn () {
return &lt;div&gt;这是我的第一个函数组件!&lt;/div&gt;
}

// 定义类组件
function App () {
return (
    &lt;div className="App"&gt;
      {/* 渲染函数组件 */}
      &lt;HelloFn /&gt;
      &lt;HelloFn&gt;&lt;/HelloFn&gt;
    &lt;/div&gt;
)
}
export default App
</code></pre>
<p><strong>约定说明</strong></p>
<ol>
<li>
<p>组件的名称<strong>必须首字母大写</strong>,react内部会根据这个来判断是组件还是普通的HTML标签</p>
</li>
<li>
<p>函数组件<strong>必须有返回值</strong>,表示该组件的 UI 结构;如果不需要渲染任何内容,则返回 null</p>
</li>
<li>
<p>组件就像 HTML 标签一样可以被渲染到页面中。组件表示的是一段结构内容,对于函数组件来说,渲染的内容是函数的<strong>返回值</strong>就是对应的内容</p>
</li>
<li>
<p>使用函数名称作为组件标签名称,可以成对出现也可以自闭合</p>
</li>
</ol>
<h2 id="类组件">类组件</h2>
<p><code>目标任务:</code>   能够独立完成类组件的创建和渲染</p>
<blockquote>
<p>使用 ES6 的 class 创建的组件,叫做类(class)组件</p>
</blockquote>
<p><strong>组件定义与渲染</strong></p>
<pre><code class="language-jsx">// 引入React
import React from 'react'

// 定义类组件
class HelloC extends React.Component {
render () {
    return &lt;div&gt;这是我的第一个类组件!&lt;/div&gt;
}
}

function App () {
return (
    &lt;div className="App"&gt;
      {/* 渲染类组件 */}
      &lt;HelloC /&gt;
      &lt;HelloC&gt;&lt;/HelloC&gt;
    &lt;/div&gt;
)
}
export default App
</code></pre>
<p><strong>约定说明</strong></p>
<ol>
<li>
<p><strong>类名称也必须以大写字母开头</strong></p>
</li>
<li>
<p>类组件应该继承 React.Component 父类,从而使用父类中提供的方法或属性</p>
</li>
<li>
<p>类组件必须提供 render 方法<strong>render 方法必须有返回值,表示该组件的 UI 结构</strong></p>
</li>
</ol>
<h2 id="事件绑定">事件绑定</h2>
<p><code>目标任务:</code>   能够独立绑定任何事件并能获取到事件对象e</p>
<h3 id="1-如何绑定事件">1. 如何绑定事件</h3>
<ul>
<li>
<p>语法</p>
<p>on + 事件名称 = { 事件处理程序 } ,比如:<code>&lt;div onClick={()=&gt;{}}&gt;&lt;/div&gt;</code></p>
</li>
<li>
<p>注意点</p>
<p>react事件采用驼峰命名法,比如:onMouseEnter、onFocus</p>
</li>
<li>
<p>样例</p>
<pre><code class="language-jsx">// 函数组件
function HelloFn () {
// 定义事件回调函数
const clickHandler = () =&gt; {
    console.log('事件被触发了')
}
return (
    // 绑定事件
    &lt;button onClick={clickHandler}&gt;click me!&lt;/button&gt;
)
}

// 类组件
class HelloC extends React.Component {
// 定义事件回调函数
clickHandler = () =&gt; {
    console.log('事件被触发了')
}
render () {
    return (
      // 绑定事件
      &lt;button onClick={this.clickHandler}&gt;click me!&lt;/button&gt;
    )
}
}
</code></pre>
</li>
</ul>
<h3 id="2-获取事件对象">2. 获取事件对象</h3>
<ul>
<li>
<p>通过事件处理程序的参数获取事件对象e</p>
<pre><code class="language-jsx">// 函数组件
function HelloFn () {
// 定义事件回调函数
const clickHandler = (e) =&gt; {
    e.preventDefault()
    console.log('事件被触发了', e)
}
return (
    // 绑定事件
    &lt;a href="http://www.baidu.com/" onClick={clickHandler}&gt;百度&lt;/a&gt;
)
}
</code></pre>
</li>
</ul>
<h2 id="组件状态">组件状态</h2>
<p><code>目标任务:</code>   能够为组件添加状态和修改状态的值</p>
<blockquote>
<p>一个前提:在react hook出来之前,函数式组件是没有自己的状态的,所以我们统一通过类组件来讲解</p>
</blockquote>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194539872-1684213540.png" alt="" loading="lazy"></p>
<h3 id="1-初始化状态">1. 初始化状态</h3>
<ul>
<li>
<p>通过class的实例属性state来初始化</p>
</li>
<li>
<p>state的值是一个对象结构,表示一个组件可以有多个数据状态</p>
<pre><code class="language-jsx">class Counter extends React.Component {
// 初始化状态
state = {
    count: 0
}
render() {
    return &lt;button&gt;计数器&lt;/button&gt;
}
}
</code></pre>
</li>
</ul>
<h3 id="2-读取状态">2. 读取状态</h3>
<ul>
<li>
<p>通过this.state来获取状态</p>
<pre><code class="language-jsx">class Counter extends React.Component {
// 初始化状态
state = {
    count: 0
}
render() {
    // 读取状态
    return &lt;button&gt;计数器{this.state.count}&lt;/button&gt;
}
}
</code></pre>
</li>
</ul>
<h3 id="3-修改状态">3. 修改状态</h3>
<ul>
<li>
<p>语法</p>
<p><code>this.setState({ 要修改的部分数据 })</code></p>
</li>
<li>
<p>setState方法作用</p>
<ol>
<li>修改state中的数据状态</li>
<li>更新UI</li>
</ol>
</li>
<li>
<p>思想</p>
<p>​        数据驱动视图,也就是只要修改数据状态,那么页面就会自动刷新,无需手动操作dom</p>
</li>
<li>
<p>注意事项</p>
<p>​        <strong>不要直接修改state中的值,必须通过setState方法进行修改</strong></p>
</li>
</ul>
<pre><code class="language-jsx">class Counter extends React.Component {
// 定义数据
state = {
    count: 0
}
// 定义修改数据的方法
setCount = () =&gt; {
    this.setState({
      count: this.state.count + 1
    })
}
// 使用数据 并绑定事件
render () {
    return &lt;button onClick={this.setCount}&gt;{this.state.count}&lt;/button&gt;
}
}
</code></pre>
<h2 id="this问题说明">this问题说明</h2>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194606999-1022139003.png" alt="" loading="lazy"></p>
<p>这里我们作为了解内容,随着js标准的发展,主流的写法已经变成了class fields,无需考虑太多this问题</p>
<h2 id="react的状态不可变">React的状态不可变</h2>
<p><code>目标任务:</code>能够理解不可变的意义并且知道在实际开发中如何修改状态</p>
<p><strong>概念</strong>:不要直接修改状态的值,而是基于当前状态创建新的状态值</p>
<p><strong>1. 错误的直接修改</strong></p>
<pre><code class="language-js">state = {
count : 0,
list: ,
person: {
   name:'jack',
   age:18
}
}
// 直接修改简单类型Number
this.state.count++
++this.state.count
this.state.count += 1
this.state.count = 1

// 直接修改数组
this.state.list.push(123)
this.state.list.spice(1,1)

// 直接修改对象
this.state.person.name = 'rose'
</code></pre>
<p><strong>2. 基于当前状态创建新值</strong></p>
<pre><code class="language-js">this.setState({
    count: this.state.count + 1
    list: [...this.state.list, 4],
    person: {
       ...this.state.person,
       // 覆盖原来的属性 就可以达到修改对象中属性的目的
       name: 'rose'
    }
})
</code></pre>
<h2 id="表单处理">表单处理</h2>
<p><code>目标任务:</code>能够使用受控组件的方式获取文本框的值</p>
<p>使用React处理表单元素,一般有俩种方式:</p>
<ol>
<li>
<p>受控组件 (推荐使用)</p>
</li>
<li>
<p>非受控组件 (了解)</p>
</li>
</ol>
<h3 id="1-受控表单组件">1. 受控表单组件</h3>
<blockquote>
<p>什么是受控组件?<code>input框自己的状态被React组件状态控制</code></p>
<p>React组件的状态的地方是在state中,input表单元素也有自己的状态是在value中,React将state与表单元素的值(value)绑定到一起,由state的值来控制表单元素的值,从而保证单一数据源特性</p>
</blockquote>
<p><strong>实现步骤</strong></p>
<p>以获取文本框的值为例,受控组件的使用步骤如下:</p>
<ol>
<li>
<p>在组件的state中声明一个组件的状态数据</p>
</li>
<li>
<p>将状态数据设置为input标签元素的value属性的值</p>
</li>
<li>
<p>为input添加change事件,在事件处理程序中,通过事件对象e获取到当前文本框的值(<code>即用户当前输入的值</code>)</p>
</li>
<li>
<p>调用setState方法,将文本框的值作为state状态的最新值</p>
</li>
</ol>
<p><strong>代码落地</strong></p>
<pre><code class="language-jsx">import React from 'react'

class InputComponent extends React.Component {
// 声明组件状态
state = {
    message: 'this is message',
}
// 声明事件回调函数
changeHandler = (e) =&gt; {
    this.setState({ message: e.target.value })
}
render () {
    return (
      &lt;div&gt;
      {/* 绑定value 绑定事件*/}
      &lt;input value={this.state.message} onChange={this.changeHandler} /&gt;
      &lt;/div&gt;
    )
}
}


function App () {
return (
    &lt;div className="App"&gt;
      &lt;InputComponent /&gt;
    &lt;/div&gt;
)
}
export default App
</code></pre>
<h3 id="2-非受控表单组件">2. 非受控表单组件</h3>
<blockquote>
<p>什么是非受控组件?</p>
<p>非受控组件就是通过手动操作dom的方式获取文本框的值,文本框的状态不受react组件的state中的状态控制,直接通过原生dom获取输入框的值</p>
</blockquote>
<p><strong>实现步骤</strong></p>
<ol>
<li>导入<code>createRef</code> 函数</li>
<li>调用createRef函数,创建一个ref对象,存储到名为<code>msgRef</code>的实例属性中</li>
<li>为input添加ref属性,值为<code>msgRef</code></li>
<li>在按钮的事件处理程序中,通过<code>msgRef.current</code>即可拿到input对应的dom元素,而其中<code>msgRef.current.value</code>拿到的就是文本框的值</li>
</ol>
<p><strong>代码落地</strong></p>
<pre><code class="language-jsx">import React, { createRef } from 'react'

class InputComponent extends React.Component {
// 使用createRef产生一个存放dom的对象容器
msgRef = createRef()

changeHandler = () =&gt; {
    console.log(this.msgRef.current.value)
}

render() {
    return (
      &lt;div&gt;
      {/* ref绑定 获取真实dom */}
      &lt;input ref={this.msgRef} /&gt;
      &lt;button onClick={this.changeHandler}&gt;click&lt;/button&gt;
      &lt;/div&gt;
    )
}
}

function App () {
return (
    &lt;div className="App"&gt;
      &lt;InputComponent /&gt;
    &lt;/div&gt;
)
}
export default App
</code></pre>
<h2 id="阶段小练习-1">阶段小练习</h2>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194637802-346771642.png" alt="" loading="lazy"></p>
<p>练习说明</p>
<ol>
<li>
<p>拉取项目模板到本地,安装依赖,run起来项目</p>
<p>https://gitee.com/react-course-series/react-component-demo</p>
</li>
<li>
<p>完成tab点击切换激活状态交互</p>
</li>
<li>
<p>完成发表评论功能</p>
<p>注意:生成独立无二的id 可以使用uuid 包<code>yarn add uuid</code></p>
<pre><code class="language-js">import {v4 as uuid} from 'uuid'
uuid() // 得到一个独一无二的id
</code></pre>
</li>
<li>
<p>完成删除评论功能</p>
</li>
</ol>
<h1 id="react组件通信">React组件通信</h1>
<h2 id="组件通信的意义">组件通信的意义</h2>
<p><code>目标任务:</code>   了解为什么需要组件通信</p>
<p>组件是独立且封闭的单元,默认情况下组件<strong>只能使用自己的数据(state)</strong></p>
<p>组件化开发的过程中,完整的功能会拆分多个组件,在这个过程中不可避免的需要互相传递一些数据</p>
<p>为了能让各组件之间可以进行互相沟通,数据传递,这个过程就是组件通信</p>
<ol>
<li>父子关系 -<strong>最重要的</strong></li>
<li>兄弟关系 -自定义事件模式产生技术方法 eventBus/通过共同的父组件通信</li>
<li>其它关系 -<strong>mobx / redux / 基于hook的方案</strong></li>
</ol>
<h2 id="父传子实现">父传子实现</h2>
<p><code>目标任务:</code>   实现父子通信中的父传子,把父组件中的数据传给子组件</p>
<p><strong>实现步骤</strong></p>
<ol>
<li>
<p>父组件提供要传递的数据-<code>state</code></p>
</li>
<li>
<p>给子组件标签<code>添加属性</code>值为 state中的数据</p>
</li>
<li>
<p>子组件中通过 <code>props</code> 接收父组件中传过来的数据</p>
<ol>
<li>类组件使用this.props获取props对象</li>
<li>函数式组件直接通过参数获取props对象</li>
</ol>
</li>
</ol>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194651945-106424730.png" alt="" loading="lazy"></p>
<p><strong>代码实现</strong></p>
<pre><code class="language-jsx">import React from 'react'

// 函数式子组件
function FSon(props) {
console.log(props)
return (
    &lt;div&gt;
      子组件1
      {props.msg}
    &lt;/div&gt;
)
}

// 类子组件
class CSon extends React.Component {
render() {
    return (
      &lt;div&gt;
      子组件2
      {this.props.msg}
      &lt;/div&gt;
    )
}
}
// 父组件
class App extends React.Component {
state = {
    message: 'this is message'
}
render() {
    return (
      &lt;div&gt;
      &lt;div&gt;父组件&lt;/div&gt;
      &lt;FSon msg={this.state.message} /&gt;
      &lt;CSon msg={this.state.message} /&gt;
      &lt;/div&gt;
    )
}
}

export default App
</code></pre>
<h2 id="props说明">props说明</h2>
<p><code>目标任务:</code>   知道props传递时的一些注意事项</p>
<p><strong>1.props是只读对象(readonly)</strong></p>
<p>根据单项数据流的要求,子组件只能读取props中的数据,不能进行修改</p>
<p><strong>2. props可以传递任意数据</strong></p>
<p>数字、字符串、布尔值、数组、对象、<code>函数、JSX</code></p>
<pre><code class="language-jsx">class App extends React.Component {
state = {
    message: 'this is message'
}
render() {
    return (
      &lt;div&gt;
      &lt;div&gt;父组件&lt;/div&gt;
      &lt;FSon
          msg={this.state.message}
          age={20}
          isMan={true}
          cb={() =&gt; { console.log(1) }}
          child={&lt;span&gt;this is child&lt;/span&gt;}
      /&gt;
      &lt;CSon msg={this.state.message} /&gt;
      &lt;/div&gt;
    )
}
}
</code></pre>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194705979-652150939.png" alt="" loading="lazy"></p>
<h2 id="子传父实现">子传父实现</h2>
<p><code>目标任务:</code>   实现父子通信中的子传父</p>
<p><strong>口诀:</strong> 父组件给子组件传递回调函数,子组件调用</p>
<p><strong>实现步骤</strong></p>
<ol>
<li>父组件提供一个回调函数 - 用于接收数据</li>
<li>将函数作为属性的值,传给子组件</li>
<li>子组件通过props调用 回调函数</li>
<li>将子组件中的数据作为参数传递给回调函数</li>
</ol>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194723939-1738779127.png" alt="" loading="lazy"></p>
<p><strong>代码实现</strong></p>
<pre><code class="language-jsx">import React from 'react'

// 子组件
function Son(props) {
function handleClick() {
    // 调用父组件传递过来的回调函数 并注入参数
    props.changeMsg('this is newMessage')
}
return (
    &lt;div&gt;
      {props.msg}
      &lt;button onClick={handleClick}&gt;change&lt;/button&gt;
    &lt;/div&gt;
)
}


class App extends React.Component {
state = {
    message: 'this is message'
}
// 提供回调函数
changeMessage = (newMsg) =&gt; {
    console.log('子组件传过来的数据:',newMsg)
    this.setState({
      message: newMsg
    })
}
render() {
    return (
      &lt;div&gt;
      &lt;div&gt;父组件&lt;/div&gt;
      &lt;Son
          msg={this.state.message}
          // 传递给子组件
          changeMsg={this.changeMessage}
      /&gt;
      &lt;/div&gt;
    )
}
}

export default App
</code></pre>
<h2 id="兄弟组件通信">兄弟组件通信</h2>
<p><code>目标任务:</code>   实现兄弟组件之间的通信</p>
<p><strong>核心思路:</strong> 通过状态提升机制,利用共同的父组件实现兄弟通信</p>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194746833-1307724945.png" alt="" loading="lazy"></p>
<p><strong>实现步骤</strong></p>
<ol>
<li>
<p>将共享状态提升到最近的公共父组件中,由公共父组件管理这个状态</p>
<ul>
<li>
<p>提供共享状态</p>
</li>
<li>
<p>提供操作共享状态的方法</p>
</li>
</ul>
</li>
<li>
<p>要接收数据状态的子组件通过 props 接收数据</p>
</li>
<li>
<p>要传递数据状态的子组件通过props接收方法,调用方法传递数据</p>
</li>
</ol>
<p><strong>代码实现</strong></p>
<pre><code class="language-jsx">import React from 'react'

// 子组件A
function SonA(props) {
return (
    &lt;div&gt;
      SonA
      {props.msg}
    &lt;/div&gt;
)
}
// 子组件B
function SonB(props) {
return (
    &lt;div&gt;
      SonB
      &lt;button onClick={() =&gt; props.changeMsg('new message')}&gt;changeMsg&lt;/button&gt;
    &lt;/div&gt;
)
}

// 父组件
class App extends React.Component {
// 父组件提供状态数据
state = {
    message: 'this is message'
}
// 父组件提供修改数据的方法
changeMsg = (newMsg) =&gt; {
    this.setState({
      message: newMsg
    })
}

render() {
    return (
      &lt;&gt;
      {/* 接收数据的组件 */}
      &lt;SonA msg={this.state.message} /&gt;
      {/* 修改数据的组件 */}
      &lt;SonB changeMsg={this.changeMsg} /&gt;
      &lt;/&gt;
    )
}
}

export default App
</code></pre>
<h2 id="跨组件通信context">跨组件通信Context</h2>
<p><code>目标任务:</code>   了解Context机制解决的问题和使用步骤</p>
<p><strong>问题场景</strong></p>
<img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220625160731281-528582364.png">
<blockquote>
<p>上图是一个react形成的嵌套组件树,如果我们想从App组件向任意一个下层组件传递数据,该怎么办呢?目前我们能采取的方式就是一层一层的props往下传,显然很繁琐</p>
<p>那么,Context 提供了一个<strong>无需为每层组件手动添加 props,就能在组件树间进行数据传递的方法</strong></p>
</blockquote>
<p><strong>实现步骤</strong></p>
<ol>
<li>
<p>创建Context对象 导出 Provider 和 Consumer对象</p>
<pre><code class="language-js">const { Provider, Consumer } = createContext()
</code></pre>
</li>
<li>
<p>使用Provider包裹根组件提供数据</p>
<pre><code class="language-jsx">&lt;Provider value={this.state.message}&gt;
    {/* 根组件 */}
&lt;/Provider&gt;
</code></pre>
</li>
<li>
<p>需要用到数据的组件使用Consumer包裹获取数据</p>
<pre><code class="language-jsx">&lt;Consumer &gt;
    {value =&gt; /* 基于 context 值进行渲染*/}
&lt;/Consumer&gt;
</code></pre>
</li>
</ol>
<p><strong>代码实现</strong></p>
<pre><code class="language-jsx">import React, { createContext }from 'react'

// 1. 创建Context对象
const { Provider, Consumer } = createContext()


// 3. 消费数据
function ComC() {
return (
    &lt;Consumer &gt;
      {value =&gt; &lt;div&gt;{value}&lt;/div&gt;}
    &lt;/Consumer&gt;
)
}

function ComA() {
return (
    &lt;ComC/&gt;
)
}

// 2. 提供数据
class App extends React.Component {
state = {
    message: 'this is message'
}
render() {
    return (
      &lt;Provider value={this.state.message}&gt;
      &lt;div className="app"&gt;
          &lt;ComA /&gt;
      &lt;/div&gt;
      &lt;/Provider&gt;
    )
}
}

export default App
</code></pre>
<h2 id="阶段小练习-2">阶段小练习</h2>
<blockquote>
<p>要求:App为父组件用来提供列表数据 ,ListItem为子组件用来渲染列表数据</p>
</blockquote>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194811013-486472842.png" alt="" loading="lazy"></p>
<pre><code class="language-js">// 列表数据
[
{ id: 1, name: '超级好吃的棒棒糖', price: 18.8, info: '开业大酬宾,全场8折' },
{ id: 2, name: '超级好吃的大鸡腿', price: 34.2, info: '开业大酬宾,全场8折' },
{ id: 3, name: '超级无敌的冰激凌', price: 14.2, info: '开业大酬宾,全场8折' }
]
</code></pre>
<p>完整代码</p>
<pre><code class="language-jsx">import React from 'react'

// 子组件
function ListItem(props) {
const { name, price, info, id, delHandler } = props
return (
    &lt;div&gt;
      &lt;h3&gt;{name}&lt;/h3&gt;
      &lt;p&gt;{price}&lt;/p&gt;
      &lt;p&gt;{info}&lt;/p&gt;
      &lt;button onClick={() =&gt; delHandler(id)}&gt;删除&lt;/button&gt;
    &lt;/div&gt;
)
}

// 父组件
class App extends React.Component {
state = {
    list: [
      { id: 1, name: '超级好吃的棒棒糖', price: 18.8, info: '开业大酬宾,全场8折' },
      { id: 2, name: '超级好吃的大鸡腿', price: 34.2, info: '开业大酬宾,全场8折' },
      { id: 3, name: '超级无敌的冰激凌', price: 14.2, info: '开业大酬宾,全场8折' }
    ]
}

delHandler = (id) =&gt; {
    this.setState({
      list: this.state.list.filter(item =&gt; item.id !== id)
    })
}

render() {
    return (
      &lt;&gt;
      {
          this.state.list.map(item =&gt;
            &lt;ListItem
            key={item.id}
            {...item}
            delHandler={this.delHandler}
            /&gt;
          )
      }
      &lt;/&gt;
    )
}
}

export default App
</code></pre>
<h1 id="react组件进阶">React组件进阶</h1>
<h2 id="children属性">children属性</h2>
<p><code>目标任务:</code>掌握props中children属性的用法</p>
<p><strong>children属性是什么</strong></p>
<p>表示该组件的子节点,只要组件内部有子节点,props中就有该属性</p>
<p><strong>children可以是什么</strong></p>
<ol>
<li>普通文本</li>
<li>普通标签元素</li>
<li>函数</li>
<li>JSX</li>
</ol>
<h2 id="props校验-场景和使用">props校验-场景和使用</h2>
<p><code>目标任务:</code>掌握组件props的校验写法,增加组件的健壮性</p>
<blockquote>
<p>对于组件来说,props是由外部传入的,我们其实无法保证组件使用者传入了什么格式的数据,如果传入的数据格式不对,就有可能会导致组件内部错误,有一个点很关键 - <strong>组件的使用者可能报错了也不知道为什么</strong>,看下面的例子</p>
</blockquote>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194828236-211824655.png" alt="" loading="lazy"></p>
<p>面对这样的问题,如何解决? <strong>props校验</strong></p>
<p><strong>实现步骤</strong></p>
<ol>
<li>安装属性校验包:<code>yarn add prop-types</code></li>
<li>导入<code>prop-types</code> 包</li>
<li>使用 <code>组件名.propTypes = {}</code> 给组件添加校验规则</li>
</ol>
<p><strong>核心代码</strong></p>
<pre><code class="language-jsx">import PropTypes from 'prop-types'

const List = props =&gt; {
const arr = props.colors
const lis = arr.map((item, index) =&gt; &lt;li key={index}&gt;{item.name}&lt;/li&gt;)
return &lt;ul&gt;{lis}&lt;/ul&gt;
}

List.propTypes = {
colors: PropTypes.array
}
</code></pre>
<h2 id="props校验-规则说明">props校验-规则说明</h2>
<p><code>目标任务:</code>掌握props常见的规则</p>
<p><strong>四种常见结构</strong></p>
<ol>
<li>常见类型:array、bool、func、number、object、string</li>
<li>React元素类型:element</li>
<li>必填项:isRequired</li>
<li>特定的结构对象:shape({})</li>
</ol>
<p><strong>核心代码</strong></p>
<pre><code class="language-js">// 常见类型
optionalFunc: PropTypes.func,
// 必填 只需要在类型后面串联一个isRequired
requiredFunc: PropTypes.func.isRequired,
// 特定结构的对象
optionalObjectWithShape: PropTypes.shape({
        color: PropTypes.string,
        fontSize: PropTypes.number
})
</code></pre>
<p>官网文档更多阅读:https://reactjs.org/docs/typechecking-with-proptypes.html</p>
<h2 id="props校验-默认值">props校验-默认值</h2>
<p><code>目标任务:</code>掌握如何给组件的props提供默认值</p>
<blockquote>
<p>通过 <code>defaultProps</code> 可以给组件的props设置默认值,在未传入props的时候生效</p>
</blockquote>
<h3 id="1-函数组件">1. 函数组件</h3>
<p>直接使用函数参数默认值</p>
<pre><code class="language-jsx">function List({pageSize = 10}) {
return (
    &lt;div&gt;
      此处展示props的默认值:{ pageSize }
    &lt;/div&gt;
)
}

// 不传入pageSize属性
&lt;List /&gt;
</code></pre>
<h3 id="2-类组件">2. 类组件</h3>
<p>使用类静态属性声明默认值,<code>static defaultProps = {}</code></p>
<pre><code class="language-jsx">class List extends Component {
static defaultProps = {
    pageSize: 10
}
   
render() {
    return (
      &lt;div&gt;
      此处展示props的默认值:{this.props.pageSize}
      &lt;/div&gt;
    )
}
}
&lt;List /&gt;
</code></pre>
<h2 id="生命周期---概述">生命周期 - 概述</h2>
<p><code>目标任务:</code>能够说出组件生命周期一共几个阶段</p>
<blockquote>
<p>组件的生命周期是指组件从被创建到挂载到页面中运行起来,再到组件不用时卸载的过程,注意,只有类组件才有生命周期(类组件 实例化函数组件 不需要实例化)</p>
</blockquote>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194847874-1922710324.png" alt="" loading="lazy"></p>
<p>http://projects.wojtekmaj.pl/react-lifecycle-methods-diagram/</p>
<h2 id="生命周期---挂载阶段">生命周期 - 挂载阶段</h2>
<p><code>目标任务:</code>能够说出在组件挂载阶段执行的钩子函数和执行时机</p>
<p><img src="assets/life1.png" alt="" loading="lazy"></p>
<table>
<thead>
<tr>
<th>钩子 函数</th>
<th>触发时机</th>
<th>作用</th>
</tr>
</thead>
<tbody>
<tr>
<td>constructor</td>
<td>创建组件时,最先执行,初始化的时候只执行一次</td>
<td>1. 初始化state2. 创建 Ref 3. 使用 bind 解决 this 指向问题等</td>
</tr>
<tr>
<td>render</td>
<td>每次组件渲染都会触发</td>
<td>渲染UI(<strong>注意: 不能在里面调用setState()</strong> )</td>
</tr>
<tr>
<td>componentDidMount</td>
<td>组件挂载(完成DOM渲染)后执行,初始化的时候执行一次</td>
<td>1. 发送网络请求   2.DOM操作</td>
</tr>
</tbody>
</table>
<h2 id="生命周期---更新阶段">生命周期 - 更新阶段</h2>
<p><code>目标任务:</code>能够说出组件的更新阶段的钩子函数以及执行时机</p>
<p><img src="assets/life2.png" alt="" loading="lazy"></p>
<table>
<thead>
<tr>
<th>钩子函数</th>
<th>触发时机</th>
<th>作用</th>
</tr>
</thead>
<tbody>
<tr>
<td>render</td>
<td>每次组件渲染都会触发</td>
<td>渲染UI(与 挂载阶段 是同一个render)</td>
</tr>
<tr>
<td>componentDidUpdate</td>
<td>组件更新后(DOM渲染完毕)</td>
<td>DOM操作,可以获取到更新后的DOM内容,<strong>不要直接调用setState</strong></td>
</tr>
</tbody>
</table>
<h2 id="生命周期---卸载阶段">生命周期 - 卸载阶段</h2>
<p><code>目标任务:</code>能够说出组件的销毁阶段的钩子函数以及执行时机</p>
<table>
<thead>
<tr>
<th>钩子函数</th>
<th>触发时机</th>
<th>作用</th>
</tr>
</thead>
<tbody>
<tr>
<td>componentWillUnmount</td>
<td>组件卸载(从页面中消失)</td>
<td>执行清理工作(比如:清理定时器等)</td>
</tr>
</tbody>
</table>
<h2 id="阶段小练习---todomvc">阶段小练习 - todoMVC</h2>
<p>案例仓库地址:https://gitee.com/react-course-series/react-todo-mvc</p>
<p><strong>项目演示步骤:</strong></p>
<ol>
<li>
<p>克隆项目到本地</p>
<pre><code class="language-bash">git clonehttps://gitee.com/react-course-series/react-todo-mvc.git
</code></pre>
</li>
<li>
<p>安装必要依赖</p>
<pre><code class="language-bash">yarn
</code></pre>
</li>
<li>
<p>开启mock接口服务,<strong>保持窗口不关闭</strong>!!!!!</p>
<pre><code class="language-bash"># 启动mock服务
yarn mock-serve
</code></pre>
</li>
<li>
<p><strong>另起一个bash窗口</strong>开启前端服务</p>
<pre><code class="language-bash">yarn start
</code></pre>
</li>
<li>
<p>浏览器输入 localhost:3000演示效果</p>
</li>
</ol>
<p><strong>项目开发步骤:</strong></p>
<ol>
<li>
<p>切换到todo-test分支</p>
<pre><code class="language-bash">git checkout todo-test
</code></pre>
</li>
<li>
<p>打开 app.js</p>
<p>已有基础样板代码,在这个基础上编写业务逻辑即可</p>
</li>
<li>
<p>接口文档</p>
<table>
<thead>
<tr>
<th>接口作用</th>
<th>接口地址</th>
<th>接口方法</th>
<th>接口参数</th>
</tr>
</thead>
<tbody>
<tr>
<td>获取列表</td>
<td>http://localhost:3001/data</td>
<td>GET</td>
<td>无</td>
</tr>
<tr>
<td>删除</td>
<td>http://localhost:3001/data/:id</td>
<td>DELETE</td>
<td>id</td>
</tr>
<tr>
<td>搜索</td>
<td>http://localhost:3001/data/?q=keyword</td>
<td>GET</td>
<td>name(以name字段搜索)</td>
</tr>
</tbody>
</table>
</li>
</ol>
<p><strong>实现功能</strong></p>
<table>
<thead>
<tr>
<th>功能</th>
<th>核心思路</th>
</tr>
</thead>
<tbody>
<tr>
<td>表格数据渲染</td>
<td>elementPlus el-table组件使用</td>
</tr>
<tr>
<td>删除功能</td>
<td>获取当前id调用接口</td>
</tr>
<tr>
<td>搜索功能</td>
<td>用的依旧是列表接口,多传一个name参数</td>
</tr>
<tr>
<td>清除搜索功能</td>
<td>清空搜索参数重新获取列表</td>
</tr>
</tbody>
</table>
<h1 id="hooks基础">Hooks基础</h1>
<h2 id="hooks概念理解">Hooks概念理解</h2>
<p><code>本节任务:</code> 能够理解hooks的概念及解决的问题</p>
<h3 id="1-什么是hooks">1. 什么是hooks</h3>
<blockquote>
<p>Hooks的本质:<strong>一套能够使函数组件更强大,更灵活的“钩子”</strong></p>
</blockquote>
<p>React体系里组件分为 类组件 和 函数组件</p>
<p>经过多年的实战,函数组件是一个更加匹配React的设计理念 <code>UI = f(data)</code>,也更有利于逻辑拆分与重用的组件表达形式,而先前的函数组件是不可以有自己的状态的,为了能让函数组件可以拥有自己的状态,所以从react v16.8开始,Hooks应运而生</p>
<p><strong>注意点:</strong></p>
<ol>
<li>有了hooks之后,为了兼容老版本,class类组件并没有被移除,俩者都可以使用</li>
<li>有了hooks之后,不能在把函数成为无状态组件了,因为hooks为函数组件提供了状态</li>
<li>hooks只能在函数组件中使用</li>
</ol>
<h3 id="2-hooks解决了什么问题">2. Hooks解决了什么问题</h3>
<p>Hooks的出现解决了俩个问题    1. 组件的状态逻辑复用2.class组件自身的问题</p>
<ol>
<li>
<p>组件的逻辑复用</p>
<p>在hooks出现之前,react先后尝试了 mixins混入,HOC高阶组件,render-props等模式</p>
<p>但是都有各自的问题,比如mixin的数据来源不清晰,高阶组件的嵌套问题等等</p>
</li>
<li>
<p>class组件自身的问题</p>
<p>class组件就像一个厚重的‘战舰’ 一样,大而全,提供了很多东西,有不可忽视的学习成本,比如各种生命周期,this指向问题等等,而我们更多时候需要的是一个轻快灵活的'快艇'</p>
</li>
</ol>
<h2 id="usestate">useState</h2>
<h3 id="1-基础使用">1. 基础使用</h3>
<p><code>本节任务:</code> 能够学会useState的基础用法</p>
<p><strong>作用</strong></p>
<p>​        useState为函数组件提供状态(state)</p>
<p><strong>使用步骤</strong></p>
<ol>
<li>导入 <code>useState</code> 函数</li>
<li>调用 <code>useState</code> 函数,并传入状态的初始值</li>
<li>从<code>useState</code>函数的返回值中,拿到状态和修改状态的方法</li>
<li>在JSX中展示状态</li>
<li>调用修改状态的方法更新状态</li>
</ol>
<p><strong>代码实现</strong></p>
<pre><code class="language-jsx">import { useState } from 'react'

function App() {
// 参数:状态初始值比如,传入 0 表示该状态的初始值为 0
// 返回值:数组,包含两个值:1 状态值(state) 2 修改该状态的函数(setState)
const = useState(0)
return (
    &lt;button onClick={() =&gt; { setCount(count + 1) }}&gt;{count}&lt;/button&gt;
)
}
export default App
</code></pre>
<h3 id="2-状态的读取和修改">2. 状态的读取和修改</h3>
<p><code>本节任务:</code> 能够理解useState下状态的读取和修改</p>
<p><strong>读取状态</strong></p>
<p>​        该方式提供的状态,是函数内部的局部变量,可以在函数内的任意位置使用</p>
<p><strong>修改状态</strong></p>
<ol>
<li>setCount是一个函数,参数表示<code>最新的状态值</code></li>
<li>调用该函数后,将使用新值替换旧值</li>
<li>修改状态后,由于状态发生变化,会引起视图变化</li>
</ol>
<p><strong>注意事项</strong></p>
<p>​        修改状态的时候,一定要使用新的状态替换旧的状态,不能直接修改旧的状态,尤其是引用类型</p>
<h3 id="3-组件的更新过程">3. 组件的更新过程</h3>
<p><code>本节任务:</code>能够理解使用hook之后组件的更新情况</p>
<p>函数组件使用 <strong>useState</strong> hook 后的执行过程,以及状态值的变化</p>
<ul>
<li>
<p>组件第一次渲染</p>
<ol>
<li>从头开始执行该组件中的代码逻辑</li>
<li>调用 <code>useState(0)</code> 将传入的参数作为状态初始值,即:0</li>
<li>渲染组件,此时,获取到的状态 count 值为: 0</li>
</ol>
</li>
<li>
<p>组件第二次渲染</p>
<ol>
<li>点击按钮,调用 <code>setCount(count + 1)</code> 修改状态,因为状态发生改变,所以,该组件会重新渲染</li>
<li>组件重新渲染时,会再次执行该组件中的代码逻辑</li>
<li>再次调用 <code>useState(0)</code>,此时 <strong>React 内部会拿到最新的状态值而非初始值</strong>,比如,该案例中最新的状态值为 1</li>
<li>再次渲染组件,此时,获取到的状态 count 值为:1</li>
</ol>
</li>
</ul>
<p>注意:<strong>useState 的初始值(参数)只会在组件第一次渲染时生效</strong>。也就是说,以后的每次渲染,useState 获取到都是最新的状态值,React 组件会记住每次最新的状态值</p>
<pre><code class="language-jsx">import { useState } from 'react'

function App() {
const = useState(0)
// 在这里可以进行打印测试
console.log(count)
return (
    &lt;button onClick={() =&gt; { setCount(count + 1) }}&gt;{count}&lt;/button&gt;
)
}
export default App
</code></pre>
<h3 id="4-使用规则">4. 使用规则</h3>
<p><code>本节任务:</code>能够记住useState的使用规则</p>
<ol>
<li>
<p><code>useState</code> 函数可以执行多次,每次执行互相独立,每调用一次为函数组件提供一个状态</p>
<pre><code class="language-js">function List(){
// 以字符串为初始值
const = useState('cp')
// 以数组为初始值
const = useState([])
}
</code></pre>
</li>
<li>
<p><code>useState</code> 注意事项</p>
<ol>
<li>
<p>只能出现在函数组件中</p>
</li>
<li>
<p>不能嵌套在if/for/其它函数中(react按照hooks的调用顺序识别每一个hook)</p>
<pre><code class="language-js">let num = 1
function List(){
num++
if(num / 2 === 0){
   const = useState('cp')
}
const = useState([])
}
// 俩个hook的顺序不是固定的,这是不可以的!!!
</code></pre>
</li>
<li>
<p>可以通过开发者工具查看hooks状态</p>
</li>
</ol>
</li>
</ol>
<h2 id="useeffect">useEffect</h2>
<h3 id="1-理解函数副作用">1. 理解函数副作用</h3>
<p><code>本节任务:</code> 能够理解副作用的概念</p>
<p><strong>什么是副作用</strong></p>
<p>​        副作用是相对于主作用来说的,一个函数除了主作用,其他的作用就是副作用。对于 React 组件来说,<strong>主作用就是根据数据(state/props)渲染 UI</strong>,除此之外都是副作用(比如,手动修改 DOM)</p>
<p><strong>常见的副作用</strong></p>
<ol>
<li>数据请求 ajax发送</li>
<li>手动修改dom</li>
<li>localstorage操作</li>
</ol>
<p>useEffect函数的作用就是为react函数组件提供副作用处理的!</p>
<h3 id="2-基础使用">2. 基础使用</h3>
<p><code>本节任务:</code> 能够学会useEffect的基础用法并且掌握默认的执行执行时机</p>
<p><strong>作用</strong></p>
<p>​        为react函数组件提供副作用处理</p>
<p><strong>使用步骤</strong></p>
<ol>
<li>导入 <code>useEffect</code> 函数</li>
<li>调用 <code>useEffect</code> 函数,并传入回调函数</li>
<li>在回调函数中编写副作用处理(dom操作)</li>
<li>修改数据状态</li>
<li>检测副作用是否生效</li>
</ol>
<p><strong>代码实现</strong></p>
<pre><code class="language-jsx">import { useEffect, useState } from 'react'

function App() {
const = useState(0)

useEffect(()=&gt;{
    // dom操作
    document.title = `当前已点击了${count}次`
})
return (
    &lt;button onClick={() =&gt; { setCount(count + 1) }}&gt;{count}&lt;/button&gt;
)
}

export default App

</code></pre>
<h3 id="3-依赖项控制执行时机">3. 依赖项控制执行时机</h3>
<p><code>本节任务:</code> 能够学会使用依赖项控制副作用的执行时机</p>
<p><strong>1. 不添加依赖项</strong></p>
<blockquote>
<p>组件首次渲染执行一次,以及不管是哪个状态更改引起组件更新时都会重新执行</p>
<ol>
<li>组件初始渲染</li>
<li>组件更新 (不管是哪个状态引起的更新)</li>
</ol>
</blockquote>
<pre><code class="language-jsx">useEffect(()=&gt;{
    console.log('副作用执行了')
})
</code></pre>
<p><strong>2. 添加空数组</strong></p>
<blockquote>
<p>组件只在首次渲染时执行一次</p>
</blockquote>
<pre><code class="language-jsx">useEffect(()=&gt;{
       console.log('副作用执行了')
},[])
</code></pre>
<p><strong>3. 添加特定依赖项</strong></p>
<blockquote>
<p>副作用函数在首次渲染时执行,在依赖项发生变化时重新执行</p>
</blockquote>
<pre><code class="language-jsx">function App() {
    const = useState(0)
    const = useState('zs')
   
    useEffect(() =&gt; {   
      console.log('副作用执行了')
    }, )
   
    return (   
      &lt;&gt;      
         &lt;button onClick={() =&gt; { setCount(count + 1) }}&gt;{count}&lt;/button&gt;      
         &lt;button onClick={() =&gt; { setName('cp') }}&gt;{name}&lt;/button&gt;   
      &lt;/&gt;
    )
}
</code></pre>
<p><strong>注意事项</strong></p>
<p>useEffect 回调函数中用到的数据(比如,count)就是依赖数据,就应该出现在依赖项数组中,如果不添加依赖项就会有bug出现</p>
<h2 id="阶段小练习---自定义hook">阶段小练习 - 自定义hook</h2>
<p><strong>需求描述</strong>:自定义一个hook函数,实现获取滚动距离Y</p>
<blockquote>
<p><code>const = useWindowScroll()</code></p>
</blockquote>
<pre><code class="language-js">import { useState } from "react"

export function useWindowScroll () {
const = useState(0)
window.addEventListener('scroll', () =&gt; {
    const h = document.documentElement.scrollTop
    setY(h)
})
return
}
</code></pre>
<p><strong>需求描述:</strong> 自定义hook函数,可以自动同步到本地LocalStorage</p>
<blockquote>
<p><code>const = useLocalStorage(key,defaultValue)</code></p>
<ol>
<li>message可以通过自定义传入默认初始值</li>
<li>每次修改message数据的时候 都会自动往本地同步一份</li>
</ol>
</blockquote>
<pre><code class="language-js">import { useEffect, useState } from 'react'

export function useLocalStorage (key, defaultValue) {
const = useState(defaultValue)
// 每次只要message变化 就会自动同步到本地ls
useEffect(() =&gt; {
    window.localStorage.setItem(key, message)
}, )
return
}
</code></pre>
<h1 id="hooks进阶">Hooks进阶</h1>
<h2 id="usestate---回调函数的参数">useState - 回调函数的参数</h2>
<p><code>本节任务:</code>能够理解useState回调函数作为参数的使用场景</p>
<p><strong>使用场景</strong></p>
<p>参数只会在组件的初始渲染中起作用,后续渲染时会被忽略。如果初始 state 需要通过计算才能获得,则可以传入一个函数,在函数中计算并返回初始的 state,此函数只在初始渲染时被调用</p>
<p><strong>语法</strong></p>
<pre><code class="language-jsx">const = useState(()=&gt;{    // 编写计算逻辑    return '计算之后的初始值'})
</code></pre>
<p><strong>语法规则</strong></p>
<ol>
<li>回调函数return出去的值将作为 <code>name</code> 的初始值</li>
<li>回调函数中的逻辑只会在组件初始化的时候执行一次</li>
</ol>
<p><strong>语法选择</strong></p>
<ol>
<li>如果就是初始化一个普通的数据 直接使用 <code>useState(普通数据)</code> 即可</li>
<li>如果要初始化的数据无法直接得到需要通过计算才能获取到,使用<code>useState(()=&gt;{})</code></li>
</ol>
<p><strong>来个需求</strong></p>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624194930856-1929800431.png" alt="" loading="lazy"></p>
<pre><code class="language-jsx">import { useState } from 'react'

function Counter(props) {
const = useState(() =&gt; {
    return props.count
})
return (
    &lt;div&gt;
      &lt;button onClick={() =&gt; setCount(count + 1)}&gt;{count}&lt;/button&gt;
    &lt;/div&gt;
)
}

function App() {
return (
    &lt;&gt;
      &lt;Counter count={10} /&gt;
      &lt;Counter count={20} /&gt;
    &lt;/&gt;
)
}

export default App
</code></pre>
<h2 id="useeffect---清理副作用">useEffect - 清理副作用</h2>
<p><code>本节任务:</code>能够掌握清理useEffect的方法</p>
<p><strong>使用场景</strong></p>
<p>在组件被销毁时,如果有些副作用操作需要被清理,就可以使用此语法,比如常见的定时器</p>
<p><strong>语法及规则</strong></p>
<pre><code class="language-js">useEffect(() =&gt; {   
    console.log('副作用函数执行了')   
    // 副作用函数的执行时机为: 在下一次副作用函数执行之前执行   
    return () =&gt; {      
      console.log('清理副作用的函数执行了')      
      // 在这里写清理副作用的代码   
    }
})
</code></pre>
<p><strong>定时器小案例</strong></p>
<blockquote>
<p>添加副作用函数前:组件虽然已经不显示了,但是定时器依旧在运行</p>
</blockquote>
<pre><code class="language-jsx">import { useEffect, useState } from 'react'
function Foo() {
    useEffect(() =&gt; {   
      setInterval(() =&gt; {   
            console.log('副作用函数执行了')   
      }, 1000)
    })
    return &lt;div&gt;Foo&lt;/div&gt;
}


function App() {
    const = useState(true)
    return (   
      &lt;&gt;      
          &lt;button onClick={() =&gt; setFlag(false)}&gt;click&lt;/button&gt;      
         {flag ? &lt;Foo/&gt; : null}   
      &lt;/&gt;
    )
}

export default App
</code></pre>
<blockquote>
<p>添加清理副作用函数后:一旦组件被销毁,定时器也被清理</p>
</blockquote>
<pre><code class="language-jsx">import { useEffect, useState } from 'react'

function Foo() {
    useEffect(() =&gt; {   
      const timerId = setInterval(() =&gt; {      
            console.log('副作用函数执行了')   
      }, 1000)   
      // 添加清理副租用函数   
      return () =&gt; {      
            clearInterval(timerId)   
      }
    })
    return &lt;div&gt;Foo&lt;/div&gt;
}
function App() {
    const = useState(true)
    return (   
      &lt;&gt;      
          &lt;button onClick={() =&gt; setFlag(false)}&gt;click&lt;/button&gt;      
         {flag ? &lt;Foo/&gt; : null}   
      &lt;/&gt;   
    )
}

export default App
</code></pre>
<h2 id="useeffect---发送网络请求">useEffect - 发送网络请求</h2>
<p><code>本节任务:</code>能够掌握使用useEffect hook发送网络请求</p>
<p><strong>使用场景</strong></p>
<p>如何在useEffect中发送网络请求,并且封装同步 async await操作</p>
<p><strong>语法要求</strong></p>
<p>不可以直接在useEffect的回调函数外层直接包裹 await ,因为<strong>异步会导致清理函数无法立即返回</strong></p>
<pre><code class="language-js">useEffect(async ()=&gt;{   
    const res = await axios.get('http://geek.itheima.net/v1_0/channels')   
    console.log(res)
},[])
</code></pre>
<p><strong>正确写法</strong></p>
<p>在内部单独定义一个函数,然后把这个函数包装成同步</p>
<pre><code class="language-jsx">useEffect(()=&gt;{   
    async function fetchData(){      
       const res = await axios.get('http://geek.itheima.net/v1_0/channels')                            console.log(res)   
    }
},[])
</code></pre>
<h2 id="useref">useRef</h2>
<p><code>本节任务:</code>能够掌握使用useRef获取真实dom或组件实例的方法</p>
<p><strong>使用场景</strong></p>
<p>在函数组件中获取真实的dom元素对象或者是组件对象</p>
<p><strong>使用步骤</strong></p>
<ol>
<li>导入 <code>useRef</code> 函数</li>
<li>执行 <code>useRef</code> 函数并传入null,返回值为一个对象 内部有一个current属性存放拿到的dom对象(组件实例)</li>
<li>通过ref 绑定 要获取的元素或者组件</li>
</ol>
<p><strong>获取dom</strong></p>
<pre><code class="language-jsx">import { useEffect, useRef } from 'react'
function App() {
    const h1Ref = useRef(null)
    useEffect(() =&gt; {   
      console.log(h1Ref)
    },[])
    return (   
      &lt;div&gt;      
            &lt;h1 ref={ h1Ref }&gt;this is h1&lt;/h1&gt;   
      &lt;/div&gt;
    )
}
export default App
</code></pre>
<p><strong>获取组件实例</strong></p>
<blockquote>
<p>函数组件由于没有实例,不能使用ref获取,如果想获取组件实例,必须是类组件</p>
</blockquote>
<p><code>Foo.js</code></p>
<pre><code class="language-jsx">class Foo extends React.Component {
    sayHi = () =&gt; {   
      console.log('say hi')
    }
    render(){   
      return &lt;div&gt;Foo&lt;/div&gt;
    }
}
   
export default Foo
</code></pre>
<p><code>App.js</code></p>
<pre><code class="language-jsx">import { useEffect, useRef } from 'react'
import Foo from './Foo'
function App() {
    const h1Foo = useRef(null)
    useEffect(() =&gt; {   
      console.log(h1Foo)
    }, [])
    return (   
      &lt;div&gt; &lt;Foo ref={ h1Foo } /&gt;&lt;/div&gt;
    )
}
export default App
</code></pre>
<h2 id="usecontext">useContext</h2>
<p><code>本节任务:</code>能够掌握hooks下的context使用方式</p>
<p><img src="https://img2022.cnblogs.com/blog/1947730/202206/1947730-20220624195011302-1645580327.png" alt="" loading="lazy"></p>
<p><strong>实现步骤</strong></p>
<ol>
<li>使用<code>createContext</code> 创建Context对象</li>
<li>在顶层组件通过<code>Provider</code> 提供数据</li>
<li>在底层组件通过<code>useContext</code>函数获取数据</li>
</ol>
<p><strong>代码实现</strong></p>
<pre><code class="language-jsx">import { createContext, useContext } from 'react'
// 创建Context对象
const Context = createContext()

function Foo() {
    return &lt;div&gt;Foo &lt;Bar/&gt;&lt;/div&gt;
}

function Bar() {
    // 底层组件通过useContext函数获取数据
    const name = useContext(Context)
    return &lt;div&gt;Bar {name}&lt;/div&gt;
}

function App() {
    return (   
      // 顶层组件通过Provider 提供数据   
      &lt;Context.Provider value={'this is name'}&gt;   
            &lt;div&gt;&lt;Foo/&gt;&lt;/div&gt;   
      &lt;/Context.Provider&gt;
    )
}

export default App
</code></pre>
<h2 id="阶段小练习-todomvc-hook版">阶段小练习-todoMvc-hook版</h2>
<p>案例仓库地址:https://gitee.com/react-course-series/react-tomvc-hook</p>
<p><strong>项目演示步骤:</strong></p>
<ol>
<li>
<p>克隆项目到本地</p>
<pre><code class="language-bash">git clonehttps://gitee.com/react-course-series/react-tomvc-hook.git
</code></pre>
</li>
<li>
<p>安装必要依赖</p>
<pre><code class="language-bash">yarn
</code></pre>
</li>
<li>
<p>开启mock接口服务,<strong>保持窗口不关闭</strong>!!!!!</p>
<pre><code class="language-bash"># 启动mock服务
yarn mock-serve
</code></pre>
</li>
<li>
<p><strong>另起一个bash窗口</strong>开启前端服务</p>
<pre><code class="language-bash">yarn start
</code></pre>
</li>
<li>
<p>浏览器输入 localhost:3000演示效果</p>
</li>
</ol>
<p><strong>项目开发步骤:</strong></p>
<ol>
<li>
<p>切换到todo-test分支</p>
<pre><code class="language-bash">git checkout todo-test
</code></pre>
</li>
<li>
<p>打开 app.js</p>
<p>已有基础样板代码,在这个基础上编写业务逻辑即可</p>
</li>
<li>
<p>接口文档</p>
<table>
<thead>
<tr>
<th>接口作用</th>
<th>接口地址</th>
<th>接口方法</th>
<th>接口参数</th>
</tr>
</thead>
<tbody>
<tr>
<td>获取列表</td>
<td>http://localhost:3001/data</td>
<td>GET</td>
<td>无</td>
</tr>
<tr>
<td>删除</td>
<td>http://localhost:3001/data/:id</td>
<td>DELETE</td>
<td>id</td>
</tr>
<tr>
<td>搜索</td>
<td>http://localhost:3001/data/?q=keyword</td>
<td>GET</td>
<td>name(以name字段搜索)</td>
</tr>
</tbody>
</table>
</li>
</ol>
<p><strong>实现功能</strong></p>
<table>
<thead>
<tr>
<th>功能</th>
<th>核心思路</th>
</tr>
</thead>
<tbody>
<tr>
<td>表格数据渲染</td>
<td>elementPlus el-table组件使用</td>
</tr>
<tr>
<td>删除功能</td>
<td>获取当前id调用接口</td>
</tr>
<tr>
<td>搜索功能</td>
<td>用的依旧是列表接口,多传一个name参数</td>
</tr>
<tr>
<td>清除搜索功能</td>
<td>清空搜索参数重新获取列表</td>
</tr>
</tbody>
</table><br><br>
来源:https://www.cnblogs.com/bingquan1/p/16410061.html
頁: [1]
查看完整版本: 快速入门React(基础)