React组件实例更改state状态值(四)

简介: 【8月更文挑战第14天】React组件实例更改state状态值(四)

代码演示

如果我们想通过函数来更改类中的一个state状态值,代码可能是这样的

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>hello_react</title>
  </head>
  <body>
    <!-- 准备好一个“容器” -->
    <div id="test"></div>
    <!-- 引入react核心库 -->
    <script type="text/javascript" src="./js/react.development.js"></script>
    <!-- 引入react-dom,用于支持react操作DOM -->
    <script type="text/javascript" src="./js/react-dom.development.js"></script>
    <!-- 引入babel,用于将jsx转为js -->
    <script type="text/javascript" src="./js/babel.min.js"></script>
    <script type="text/babel">
      // 1、创建类组件
      class MyComponent extends React.Component {
    
    
        constructor(props) {
    
    
          super(props);
          this.state = {
    
     isHot: true };
          this.tel = this.tel.bind(this);
        }
        render() {
    
    
          return <h1 onClick={
    
    this.tel}>今天天气很{
    
    this.state.isHot ? "热" : "冷"},我想吃冰激凌</h1>;
        }
        tel() {
    
    
          const isHot = this.state.isHot;
          this.setState({
    
     isHot: !isHot });
        }
      }
      // 渲染组件
      ReactDOM.render(<MyComponent />, document.getElementById("test"));
    </script>
  </body>
</html>

我们通过this.tel = this.tel.bind(this);改变函数中this的指向,通过setState来更新state状态值。
要理解这个代码的意义,我们需要深入理解一下类组件中的this指向。

类组件中的this指向

函数中的this指向

<script type="text/babel">
  // 1、创建类组件
  class MyComponent extends React.Component {
   
   
    constructor(props) {
   
   
      super(props);
      this.state = {
   
    isHot: true };
    }
    render() {
   
   
      return <h1 onClick={
   
   this.tel}>今天天气很{
   
   this.state.isHot ? "热" : "冷"},我想吃冰激凌</h1>;
    }
    tel() {
   
   
      console.log("类中的this", this);
    }
  }
  // 渲染组件
  ReactDOM.render(<MyComponent />, document.getElementById("test"));
</script>

我们直接打印下this的值
image.png
可以看到,当函数tel被执行时,其代码中的this指向是undefined。

根据类的知识,我们能梳理出一下信息:

  • tel方法是定义在MyComponent的原型对象上,供实例使用
  • tel方法作为onClick的回调,不是由实例对象直接调用的,而是直接调用的

我们知道, this的指向本质上是指向调用者的,即谁调用函数或方法,this就指向这个调用者!由于tel函数不是实例调用的,因此,必然没有指向MyComponent,this也就是undefined了。

更改函数中this的指向

我们需要通过函数来更改state中的值,必须在函数中可以访问组件实例对象才行。因此,我们需要通过bind改变其this指向
我们先复习一下bind的用法

bind(thisArg, arg1, arg2, arg3, ...)

fn.bind的作用是只修改this指向,但不会立即执行fn;会返回一个修改了this指向后的fn。需要调用才会执行:bind(thisArg, arg1, arg2, arg3, ...)()。bind的传参和call相同。

现在,我们在看这句代码,它的作用很明显

this.tel = this.tel.bind(this);

image.png
通过 this.tel在MyComponent上创建了一个tel属性,这个属性值是一个函数,函数来自MyComponen原型上的方法tel,且函数tel的this指向MyComponent自身
或者,我们这么写更便于理解

class MyComponent extends React.Component {
   
   
  constructor(props) {
   
   
    super(props);
    this.state = {
   
    isHot: true };
    this.telFuc = this.tel.bind(this);
  }
  render() {
   
   
    return <h1 onClick={
   
   this.telFuc}>今天天气很{
   
   this.state.isHot ? "热" : "冷"},我想吃冰激凌</h1>;
  }
  tel() {
   
   
    console.log(this);
    const isHot = this.state.isHot;
    this.setState({
   
    isHot: !isHot });
  }
}

image.png

更改类中的state状态值

根据最初的示例代码,我们知道,state状态值的更改是借助setState函数完成的,我们如果想使用react的双向数据绑定,就不能直接修改state的值,比如下面的我种方式不会让页面在点击时有视图更新

  // 1、创建类组件
  class MyComponent extends React.Component {
   
   
    constructor(props) {
   
   
      super(props);
      this.state = {
   
    isHot: true };
      this.telFuc = this.tel.bind(this);
    }
    render() {
   
   
      return <h1 onClick={
   
   this.telFuc}>今天天气很{
   
   this.state.isHot ? "热" : "冷"},我想吃冰激凌</h1>;
    }
    tel() {
   
   
      this.state.isHot = !this.state.isHot;
    }
  }
  // 渲染组件
  ReactDOM.render(<MyComponent />, document.getElementById("test"));

setState函数来自承自父类React.Component原型上的方法,因此,我们可以直接访问到。
image.png
更新状态值的基础方法如下

class MyComponent extends React.Component {
   
   
  constructor(props) {
   
   
    super(props);
    this.state = {
   
    isHot: true };
    this.telFuc = this.tel.bind(this);
  }
  render() {
   
   
    return <h1 onClick={
   
   this.telFuc}>今天天气很{
   
   this.state.isHot ? "热" : "冷"},我想吃冰激凌</h1>;
  }
  tel() {
   
   
    console.log(this);
    this.setState({
   
    isHot: !this.state.isHot });
  }
}

构造函数中每个函数执行的次数

<script type="text/babel">
  // 1、创建类组件
  class MyComponent extends React.Component {
   
   
    constructor(props) {
   
   
      console.log("constructor执行了");
      super(props);
      this.state = {
   
    isHot: true };
      this.telFuc = this.tel.bind(this)  }
    render() {
   
   
      console.log("render函数执行了");
      return <h1 onClick={
   
   this.telFuc}>今天天气很{
   
   this.state.isHot ? "热" : "冷"},我想吃冰激凌</h1>;
    }
    tel() {
   
   
      console.log("tel函数执行了");
      this.setState({
   
    isHot: !this.state.isHot });
    }
  }
  // 渲染组件
  ReactDOM.render(<MyComponent />, document.getElementById("test"));
</script>

GIF 2023-5-10 14-38-56.gif
通过上图我们可以看出,由于MyComponent组件被实例化一次,因此constructor函数执行一次
render函数和原型上的 tel方法则在每次点击页面时执行一次

state的简写方法

state更改值的普通写法

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <title>hello_react</title>
  </head>
  <body>
    <!-- 准备好一个“容器” -->
    <div id="test"></div>
    <!-- 引入react核心库 -->
    <script type="text/javascript" src="./js/react.development.js"></script>
    <!-- 引入react-dom,用于支持react操作DOM -->
    <script type="text/javascript" src="./js/react-dom.development.js"></script>
    <!-- 引入babel,用于将jsx转为js -->
    <script type="text/javascript" src="./js/babel.min.js"></script>
    <script type="text/babel">
      // 1、创建类组件
      class MyComponent extends React.Component {
   
   
        constructor(props) {
   
   
          super(props);
          this.state = {
   
    isHot: true };
          this.telFuc = this.tel.bind(this);
        }
        render() {
   
   
          return <h1 onClick={
   
   this.telFuc}>今天天气很{
   
   this.state.isHot ? "热" : "冷"},我想吃冰激凌</h1>;
        }
        tel() {
   
   
          this.setState({
   
    isHot: !this.state.isHot });
        }
      }
      // 渲染组件
      ReactDOM.render(<MyComponent />, document.getElementById("test"));
    </script>
  </body>
</html>

要更改一个组件中的状态值state,我们需要写一个构造器,在构造器内部初始化state

this.state = {
   
    isHot: true };

同时,需要更改MyComponent原型上函数tel的this指向

this.telFuc = this.tel.bind(this);

state更改值的简便写法

要精简这种写法,我们先复习下类的写法。

class Parent {
   
   
  constructor(name) {
   
   
    this.name = name;
  }
  speak() {
   
   
    console.log("请讲话");
  }
  age = 3;
  eat = function () {
   
   
    console.log("吃饭");
  };
}
const people = new Parent("小明");
console.log("people: ", people);

通过上述代码,我们可以发现,通过constructor我们可以在类自身添加静态属性,通过 age = 3;的形式也可以直接在类上自身添加属性;我们如果直接在类里写函数,则函数是定义在Parent类的原型prototype上的,但如果通过eat = function () {};的形式可以直接在类自身添加静态方法。
image.png
根据这个简写规则,我们可以直接更改类中state更新的代码

<script type="text/babel">
  // 1、创建类组件
  class MyComponent extends React.Component {
   
   
    state = {
   
    isHot: true };
    render() {
   
   
      return <h1 onClick={
   
   this.tel}>今天天气很{
   
   this.state.isHot ? "热" : "冷"},我想吃冰激凌</h1>;
    }
    tel = function () {
   
   
      this.setState({
   
    isHot: !this.state.isHot });
    };
  }
  // 渲染组件
  ReactDOM.render(<MyComponent />, document.getElementById("test"));
</script>

这样看起来代码精简多了!但很可惜,this.setState({ isHot: !this.state.isHot });这句代码不会生效,因为这里的this依旧找不到。
但是,我们将function改成箭头函数的形式就好了

<script type="text/babel">
  // 1、创建类组件
  class MyComponent extends React.Component {
   
   
    state = {
   
    isHot: true };
    render() {
   
   
      return <h1 onClick={
   
   this.tel}>今天天气很{
   
   this.state.isHot ? "热" : "冷"},我想吃冰激凌</h1>;
    }
    tel = () => {
   
   
      console.log(this);
      this.setState({
   
    isHot: !this.state.isHot });
    };
  }
  // 渲染组件
  ReactDOM.render(<MyComponent />, document.getElementById("test"));
</script>

使用箭头函数时,函数内的this指向父级,也就是MyComponent,因此,这个问题就迎刃而解了。
页面效果:GIF 2023-5-10 15-12-55.gif

相关文章
|
2天前
|
前端开发 JavaScript 开发者
React 按钮组件 Button
本文介绍了 React 中按钮组件的基础概念,包括基本的 `&lt;button&gt;` 元素和自定义组件。详细探讨了事件处理、参数传递、状态管理、样式设置和可访问性优化等常见问题及其解决方案,并提供了代码示例。帮助开发者避免易错点,提升按钮组件的使用体验。
102 77
|
3天前
|
前端开发 UED 开发者
React 对话框组件 Dialog
本文详细介绍了如何在 React 中实现一个功能完备的对话框组件(Dialog),包括基本用法、常见问题及其解决方案,并通过代码案例进行说明。从安装依赖到创建组件、添加样式,再到解决关闭按钮失效、背景点击无效、键盘导航等问题,最后还介绍了如何添加动画效果和处理异步关闭操作。希望本文能帮助你在实际开发中更高效地使用 React 对话框组件。
97 75
|
1月前
|
前端开发 JavaScript 测试技术
React 分页组件 Pagination
本文介绍了如何在 React 中从零构建分页组件,涵盖基础概念、常见问题及解决方案。通过示例代码详细讲解了分页按钮的创建、分页按钮过多、初始加载慢、状态管理混乱等常见问题的解决方法,以及如何避免边界条件、性能优化和用户反馈等方面的易错点。旨在帮助开发者更好地理解和掌握 React 分页组件的开发技巧,提升应用的性能和用户体验。
69 0
|
8天前
|
前端开发 Java API
React 进度条组件 ProgressBar 详解
本文介绍了如何在 React 中创建进度条组件,从基础实现到常见问题及解决方案,包括动态更新、状态管理、性能优化、高级动画效果和响应式设计等方面,帮助开发者构建高效且用户体验良好的进度条。
37 18
|
23天前
|
存储 前端开发 测试技术
React组件的最佳实践
React组件的最佳实践
|
21天前
|
前端开发 API 开发者
React 文件上传组件 File Upload
本文详细介绍了如何在 React 中实现文件上传组件,从基础的文件选择和上传到服务器,再到解决文件大小、类型限制、并发上传等问题,以及实现多文件上传、断点续传和文件预览等高级功能,帮助开发者高效构建可靠的应用。
48 12
|
16天前
|
存储 前端开发 JavaScript
React 表单输入组件 Input:常见问题、易错点及解决方案
本文介绍了在 React 中使用表单输入组件 `Input` 的基础概念,包括受控组件与非受控组件的区别及其优势。通过具体代码案例,详细探讨了创建受控组件、处理多个输入字段、输入验证和格式化的方法,并指出了常见易错点及避免方法,旨在提升表单的健壮性和用户体验。
27 4
|
23天前
|
前端开发 JavaScript API
React 文件下载组件 File Download
本文介绍了在React中实现文件下载组件的方法,包括使用`a`标签和JavaScript动态生成文件,解决了文件路径、文件类型、大文件下载及文件名乱码等问题,并展示了使用第三方库`file-saver`和生成CSV文件的高级用法。
35 6
|
20天前
|
前端开发 JavaScript API
React 文件下载组件:File Download
本文详细介绍了如何在React应用中实现文件下载组件,包括基本概念、实现步骤和代码示例。同时,探讨了常见问题如文件类型不匹配、文件名乱码等及其解决方法,旨在提升用户体验和代码可维护性。
39 2
|
24天前
|
存储 前端开发 JavaScript
React 文件上传组件 File Upload
本文介绍了如何在 React 中实现文件上传组件,包括基本的概念、实现步骤、常见问题及解决方案。通过 `&lt;input type=&quot;file&quot;&gt;` 元素选择文件,使用 `fetch` 发送请求,处理文件类型和大小限制,以及多文件上传和进度条显示等高级功能,帮助开发者构建高效、可靠的文件上传组件。
66 2