react-类组件1

news2025/1/8 5:22:13

类组件:

import { Component } from "react";

class App extends Component {
  constructor() {
    super();
    this.state = {
      message: "xxxxx",
    };
  }
  render() {
    return (
      <div>
        <div>{this.state.message}</div>
      </div>
    );
  }
}

export default App;
import React from "react";
import ReactDOM from "react-dom/client";

import App from "./App";
//react18
const root = ReactDOM.createRoot(document.getElementById("root"));
root.render(
  <React.StrictMode>
    <App />
  </React.StrictMode>
);

this指向问题

cli(){ console.log(this)}默认情况下this是undefined。
在正常的dom操作中监听点击事件,监听的函数的this是节点对象。
而react并不是直接渲染成真实dom,它本质上是一个React的Element对象。在监听的时候,函数并没有绑定this,所以默认情况想就是一个undefined。

直接调用相当于默认调用
在这里插入图片描述
可以通过bind给它绑定this,render里面有this,指向组件实例。

add(){
  console.log(this)
}
render(){
  console.log(this)
  return (
   <div>
    <button onClick={this.add.bind(this)}>+</button>
   </div>
  )
}

直接在html中绑定不太方便,可以在constructor中提前绑定

 constructor(){
   this.add = this.add.bind(this)
 }
 add(){
   console.log(this)
 }
 render(){
   return (
    <div>
     <button onClick={this.add}>+</button>
    </div>
  )
 }

还有更简单的方法使用箭头函数

//setState简短说明
//setState继承自Component,
//调用setState后,会将state中的xxx的值修改掉,然后自动重新执行render函数
 add(){
   console.log(this)
   this.setState({

   })
 }
 render(){
   return (
    <div>
     <button onClick={()=>this.add()}>+</button>
    </div>
  )
 }

jsx

jsx是js的拓展语法,也在许多地方成为JavaScript XML。
jsx顶层只能有一个根元素,jsx中的标签可以是单标签,也可以是双标签。

jsx嵌套变量作为子元素

当变量为Number,String,Array时,可以直接显示。
当变量为null,undefined,boolean时,内容为空,如果需要显示的话可以将它们转换为字符创,例如使用toString
当变量为Object时,不能作为子元素

class App extends Component {
  constructor() {
    super();
    this.state = {
      message: "xxxxx",
      age: 18,
      address: ["!1", "222"],
      isSelected: true,
      test: " ",
      //  test: null,   test:undefined
    };
  }
  render() {
    const { message, age, address, isSelected, test } = this.state;
    return (
      <div>
        <div>{message}</div>
        <div>{age}</div>
        <div>{address[0]}</div>
        <div>{isSelected.toString()}</div>
        <div>{test}</div>
      </div>
    );
  }
}

在这里插入图片描述

当变量为Object时
在这里插入图片描述

使用表达式

class App extends Component {
  constructor() {
    super();
    this.state = {
      message: "xxxxx",
      age: 18,
      address: ["!1", "222"],
      isSelected: true,
      test: null,
      obj: {
        name: "111",
        age: 19,
      },
    };
  }
  address() {
    return <span>test</span>;
  }
  render() {
    const { message, age, address, isSelected, test, obj } = this.state;
    if (isSelected) {
      this.setState({
        obj: {
          age: 100,
        },
      });
    }

    return (
      <div>
        <div>{message}</div>
        <div>{age}</div>
        <div>{address[0]}</div>
        <div>{isSelected.toString()}</div>
        <div>{test}</div>
        <div>{Object.keys(obj)[0]}</div>
        {/* 使用表达式*/}
        <div>{19 - 1}</div>
        <div>{isSelected ? "true" : "false"}</div>
        <div>
          {[1, 2, 3].map((item) => {
            return <li>{item}</li>;
          })}
        </div>
        <div>{obj.age}</div>
        <>{this.address()}</>
      </div>
    );
  }
}

在这里插入图片描述

绑定 title,src,href,class,style

 this.state = {
      title: "xxx",
      href: "https://developer.mozilla.org/zh-CN/docs/Web/CSS/background-image",
      url: "https://img-blog.csdnimg.cn/direct/38f2d0b33be2462796e2a123165e3fd8.png",
    };
    render() {
    const { title, href, url } = this.state;

    return (
      <div>
        <h1 title={title}>111</h1>
        <a href={href}>a</a>
        <img src={url} />
      </div>
    );
  }

绑定className,style

 render() {
 
    const className = `aaa  ${isSelected ? "active" : ""}`;
    const classList = ["aa", "bb"];
    classList.push("vvv");
    return (
      <div>
        <div className={className}>11</div>
        <div style={{ fontSize: "20px", color: "red" }}>www</div>
        <div className={classList.join(" ")}></div>
      </div>
    );
  }

事件的绑定

事件名使用小驼峰命名,通过{}传入一个事件处理函数,这个函数会在事件发生时被执行

<button onClick={this.add.bind(this)}>+</button>

传递event和其他参数

class App extends Component {
  constructor() {
    super();
    this.state = {
    };
  }
  add(event, age, num) {
    console.log(event);
    console.log(age, num);
  }

  render() {
    return (
      <div>
        <button onClick={this.add.bind(this, "xx", 19)}>1</button>
        <button onClick={(event) => this.add(event, "zzz", 11)}>2</button>
      </div>
    );
  }
}

使用bind绑定的 event是最后一个
在这里插入图片描述

条件渲染

条件渲染的方式:

  1. 条件判断
  2. 三元运算符
  3. 与运算符&&

使用if条件判断

 render() {
    let ele;
    const { isShow } = this.state;
    if (isShow) {
      ele = <h1>show</h1>;
    } else {
      ele = <h1>hidden</h1>;
    }
    return (
      <div>
        <span>{ele}</span>
        <button onClick={(event) => this.add(event, "zzz", 11)}>2</button>
      </div>
    );
  }

在这里插入图片描述
三元运算符和与运算符

     <div>
        <span>{isShow ? ele : "test"}</span>
        <span>{isHidden + "" && "---11"}</span>
      </div>
/*  在 JavaScript 中,对 undefined 变量使用 && 运算符可能会导致将整个表达式的结果视为假值。
   这是因为在逻辑运算中,&& 运算符是一个短路逻辑运算符,如果左侧表达式的值为一个假值(例如 false、null、undefined、0、空字符串等),
那么整个表达式会立即返回左侧表达式的值,而不再计算右侧表达式。

如果想要确保对 undefined 变量进行逻辑运算时的预期行为,可以先将变量转换为字符串或者使用其他方法来处理。
*/
//模拟v-show
<div style={{display: isShow ? "block" : "hidden"}}> </div>

在这里插入图片描述

列表展示

使用map渲染列表

class App extends Component {
  constructor() {
    super();
    this.state = {
      list: [
        {
          id: 1,
          name: "111",
        },
        {
          id: 2,
          name: "222",
        },
        {
          id: 3,
          name: "333",
        },
        {
          id: 4,
          name: "444",
        },
      ],
    };
  }

  render() {
    const { list } = this.state;

    return (
      <div>
        {list.map((item) => {
          return <div key={item.id}>{item.name}</div>;
        })}
      </div>
    );
  }
}

export default App;
// jsx--ReactElement-真实dom

在这里插入图片描述

类组件

类组件的名称以大写字符开头,继承自React.Component,必须实现render函数。constructor是可选的。

import { Component } from "react";

class App extends Component {
  constructor() {
    super();
    this.state = {};
  }

  render() {
   

    return (
      <div>
       
      </div>
    );
  }
}

export default App;

render函数的返回值

当render被调用时,它会检查this.props和this.state的变化并返回以下类型之一:
1.React元素,通过jsx创建的
2. 数组或fragments
3. Portals,将子节点渲染到不同的dom中
4. 字符串或数值类型,它们在dom中会被渲染成文本节点
5. 布尔类型或null 什么都不渲染

函数组件的返回值类型也跟类组件的一样

生命周期

在这里插入图片描述

class App extends Component {
  constructor() {
    super();
    this.state = {};
    //初始化state
  }
  componentDidMount() {
    //组件已经渲染到dom中
  }
  componentDidUpdate(proProps,preState,snapshot) {
    //dom发生更新  ,snapshot getSnapshotBeforeUpdate传递的值
  }
  componentDidMount() {
    // 组件卸载,
  }
  shouldComponentUpdate(){
    return true  //是否更新,返回false就不会更新
  }
  getSnapshotBeforeUpdate(){
    //在更新前进行一些操作,返回的数据可以在 componentDidUpdate中获取
  return {
    name:"aaa"
  }
  }
  change(){
   this.setState({
   })  //调用setState后会触发render,然后重新渲染,在执行componentDidUpdate
  }
  render() {
    return <div>222</div>;
  }
}

组件间的通信

父组件向子组件传参

// 父组件
      <div>
        <Son name={"sssxasa"} age={18}></Son>
      </div>
------------------------
//子组件
class Son extends Component {
  constructor(props) {  // constructor 这一步可以省略
    super(props);
  }
  render() {
    const { name, age } = this.props;
    return (
      <div>
        <div>
          {name}:{age}
        </div>
      </div>
    );
  }
}

propTypes 参数类型
import { Component } from "react";
import PropTypes from "prop-types";
class Son extends Component {
  render() {
    const { name, age } = this.props;
    return (
      <div>
        <div>
          {name}:{age}
        </div>
      </div>
    );
  }
}

Son.propTypes = {
  name: PropTypes.string,
  age: PropTypes.number,
};
//设置默认值
Son.defaultProps = {
  name: "default",
  age: 18,
};

子组件向父组件传值

父组件向子组件传递一个方法,这个方法接受参数,子组件使用时把参数传递过来。

父组件

  getSon(val) {
    console.log(val)
  }
  render() {
    return (
      <div>
        <Son getSon={(val) => this.getSon(val)}></Son>
      </div>
    );
  }

子组件

class Son extends Component {
  render() {
    const { name, age, getSon } = this.props;
    return (
      <div>
        <div>
          {name}:{age}
        </div>
        <button onClick={() => getSon("传递的值")}>传递值</button>
      </div>
    );
  }
}

react中的插槽

react中有两种方式实现插槽:
组件的children 和props传递react元素

组件的children

这些元素会放到组件实例上,有多个元素时children是一个数组,
在这里插入图片描述
只有一个元素时,children就是对应的元素

       const { children } = this.props;
       <div>
          <div>{children}</div>
       </div>


     //通过设置chilidren可以限制传入单个元素或多个元素
     Son.propTypes={
       children:PropTypes.array //:PropTypes.element
    }
props传递react元素

在这里插入图片描述

可以通过传递一个函数,由子组件决定渲染的内容
在这里插入图片描述

context

React.createContext 创建 需要共享的Context对象

import React from "react";

const myContext = React.createContext();

export default myContext;

Context.Provider,每个context对象都会返回一个Provider React组件,它允许组件订阅context的变化。
Provider可以接受一个value,传递给组件

 render() {
    return (
      <div>
        <myContext.Provider value={{ color: "red", name: "sss" }}>
          <Son></Son>
        </myContext.Provider>
      </div>
    );
  }

Class.context.Type 挂载在class的contextType 属性会被重新赋值为由React.createContext()创建的Context对象,通过this.context可以使用Context上的值。

import { Component } from "react";

import myContext from "./context";
class Son extends Component {
  render() {
    const { name } = this.context;
    return <div>{name}</div>;
  }
}
Son.contextType = myContext;
export default Son;

函数式组件使用context需要使用context.Consumer
  return (
      <div>
        <myContext.Provider value={{ color: "red", name: "sss" }}>
          <Son></Son>
          <Test></Test>
        </myContext.Provider>
      </div>
    );
----------------------------------------
import myContext from "./context";
export function Test() {
  return (
    <div>
      <myContext.Consumer>
        {(value) => {
          return <h1>{value.name}</h1>;
        }}
      </myContext.Consumer>
    </div>
  );
}
使用多个provider

createContext可以设置默认值

import React from "react";

const myContext = React.createContext();

export default myContext;
export const testContext = React.createContext({
  age: 18,
  address: "asdasf",
});
  return (
      <div>
        <testContext.Provider value={{ age: 18, address: "fwefes" }}>
          <myContext.Provider value={{ color: "red", name: "sss" }}>
            <Son></Son>
            <Test></Test>
          </myContext.Provider>
        </testContext.Provider>
      </div>
    );

有多个context时,可以使用Consumer使用对应的context

import { Component } from "react";
import myContext, { testContext } from "./context";

class Son extends Component {
  render() {
    const { name } = this.context;
    return (
      <div>
        {name}
        <testContext.Consumer>
          {(value) => {
            return <h1>{value.address}</h1>;
          }}
        </testContext.Consumer>
      </div>
    );
  }
}
Son.contextType = myContext;
export default Son;

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1908328.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如 何 避 开 职 场 雷 区

01 管理好自己的情绪 在职场中切忌过于情绪化&#xff0c;学会管理好自己的情绪是一个职场人必备的技能之一。坏情绪并不会帮助你解决任何问题&#xff0c;过多的抱怨&#xff0c;只会使你成为负能量的传播者。 在职场中&#xff0c;需要做的是管理好自己的情绪&#xff0c;积…

Python--并发编程--协程

概念 协程是轻量级的线程&#xff0c;它是程序员管理的并发机制&#xff0c;使得在一个线程中程序可以在多个函数之间交替运行。 Python中主要通过asyncio模块实现协程。 协程函数 用async修饰的函数 import asyncio# func为协程函数 async def func():await asyncio.slee…

【C语言】extern 关键字详解

在C语言中&#xff0c;extern关键字用于声明一个变量或函数是定义在另一个文件中的。它使得在多个文件之间共享变量或函数成为可能。extern关键字常见于大型项目中&#xff0c;通常用于声明全局变量或函数&#xff0c;这些变量或函数的定义位于其他文件中。 基本用法 变量声明…

[极客大挑战 2019]RCE ME

[极客大挑战 2019]RCE ME <?php error_reporting(0); if(isset($_GET[code])){$code$_GET[code];if(strlen($code)>40){die("This is too Long.");}if(preg_match("/[A-Za-z0-9]/",$code)){die("NO.");}eval($code); } else{highlight_f…

考CISP,不要踩坑的几点建议

当你立志要在信息安全领域闯出一片天&#xff0c;可能多少都会听行内人说&#xff0c;搞本CISP。但这个认证究竟该怎么拿&#xff1f;需要培训吗&#xff1f;培训又是怎么一回事&#xff1f;价格如何&#xff1f;还有&#xff0c;什么时候开始准备最好&#xff1f;这些问题可能…

简易Qt串口助手

界面显示如下 关于串口类 初始化 设置串口号 设置波特率 打开串口 发送按钮功能实现 接收数据显示在控件中 关闭串口

FFmpeg 实现从麦克风获取流并通过RTMP推流

使用FFmpeg库实现从麦克风获取流并通过RTMP推流&#xff0c;FFmpeg版本为4.4.2-0。RTMP服务器使用的是SRS&#xff0c;我这边是跑在Ubuntu上的&#xff0c;最好是关闭掉系统防火墙。拉流端使用VLC。如果想要降低延时&#xff0c;请看我另外一篇博客&#xff0c;里面有说降低延时…

【密码学基础】基于LWE(Learning with Errors)的全同态加密方案

学习资源&#xff1a; 全同态加密I&#xff1a;理论与基础&#xff08;上海交通大学 郁昱老师&#xff09; 全同态加密II&#xff1a;全同态加密的理论与构造&#xff08;Xiang Xie老师&#xff09; 现在第二代&#xff08;如BGV和BFV&#xff09;和第三代全同态加密方案都是基…

数据集 | 人脸公开数据集的介绍及下载地址

本文介绍了人脸相关算法的数据集。 1.人脸数据集详情 1.1.Labeled Faces in the Wild (LFW) 论文 下载地址&#xff1a;LFW Face Database : Main (umass.edu) 是目前人脸识别的常用测试集&#xff0c;其中提供的人脸图片均来源于生活中的自然场景&#xff0c;因此识别难度会…

表情包原理

https://unicode.org/Public/emoji/12.1/emoji-zwj-sequences.txt emoji 编码规则介绍_emoji编码-CSDN博客 UTS #51: Unicode Emoji C UTF-8编解码-CSDN博客 创作不易&#xff0c;小小的支持一下吧&#xff01;

数据结构练习

1. 快速排序的非递归是通过栈来实现的&#xff0c;则前序与层次可以通过控制入栈的顺序来实现&#xff0c;因为递归是会一直开辟栈区空间&#xff0c;所以非递归的实现只需要一个栈的大小&#xff0c;而这个大小是小于递归所要的&#xff0c; 非递归与递归的时间复杂度是一样的…

Docker Desktop如何换镜像源?

docker现在很多镜像源都出现了问题,导致无法拉取镜像,所以找到一个好的镜像源,尤为重要。 一、阿里镜像源 经过测试,目前,阿里云镜像加速地址还可以使用。如果没有阿里云账号,需要先注册一个账号。 地址:https://cr.console.aliyun.com/cn-hangzhou/instances/mirrors 二…

[Flink]三、Flink1.13

11. Table API 和 SQL 如图 11-1 所示&#xff0c;在 Flink 提供的多层级 API 中&#xff0c;核心是 DataStream API &#xff0c;这是我们开发流 处理应用的基本途径&#xff1b;底层则是所谓的处理函数&#xff08; process function &#xff09;&#xff0c;可以访…

User parameters 用户参数与Web监控

目录 一. 自定义键介绍 二. 制作步骤 1. 添加无可变部分参数 2. 添加有可变参数 3. 使用用户参数监控php-fpm 服务的状态 三. Web页面导入应用监控 四. Web监控 主要功能和操作&#xff1a; 开启方式 官方预定义监控项文档https://www.zabbix.com/documentation/6…

fastjson-1.2.24漏洞复现

文章目录 0x01 前言0x02 环境0x03漏洞复现环境准备 0x04 漏洞分析利用链源码分析 0x05 总结0x06 可能遇到的坑 0x01 前言 影响版本 fastjson < 1.2.24 本文出于学习fastjson漏洞的目的&#xff0c;为了能更好的复现漏洞&#xff0c;需要有以下前置知识。 springbootfastj…

刷代码随想录有感(129):动态规划——两个字符串的删除操作

题干&#xff1a; 代码&#xff1a; class Solution { public:int minDistance(string word1, string word2) {vector<vector<int>>dp(word1.size() 1, vector<int>(word2.size() 1, 0));for(int i 1; i < word1.size(); i){for(int j 1; j < wor…

15个最佳WooCommerce商城网站及其主要功能

正在寻找的WooCommerce商城网站来激发灵感&#xff1f; 在动态的在线购物世界中&#xff0c;WooCommerce 就像企业的超级英雄。它帮助他们轻松创建强大而可靠的在线商店&#xff0c;并与WordPress顺畅协作。 从创新的产品展示到简化的结账流程&#xff0c;每个特色网站都拥有…

724.力扣每日一题7/8 Java

博客主页&#xff1a;音符犹如代码系列专栏&#xff1a;算法练习关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ 目录 思路 解题方法 时间复杂度 空间复杂度 Code 思路 主要基于数组的…

解析Xml文件并修改QDomDocument的值

背景&#xff1a; 我需要解决一个bug&#xff0c;需要我从xml中读取数据到QDomDocument&#xff0c;然后获取到我想要的目标信息&#xff0c;然后修改该信息。 ---------------------------------------------------------------------------------------------------------…

PPO控制人形机器人行走举例

PPO控制人形机器人行走 Proximal Policy Optimization (PPO) 是一种策略优化算法,在强化学习中广泛使用。它通过改进策略梯度方法,使得训练过程更加稳定和高效。 PPO算法原理介绍 PPO算法主要有两种变体:PPO-Clip 和 PPO-Penalty。这里主要介绍PPO-Clip,因为它更常用。 …