当前位置: 首页 > news >正文

第七章 react组件实例中三大属性之props

  React中的props是一种用于组件之间传递数据的机制。它是一个对象,包含了组件的属性和值。当一个组件被渲染时,它的props对象会被传递给它。这样,组件就可以使用这些属性来渲染自己。


案例分析

以下案例是展示一个人的信息。

<!-- 准备好员工“容器” -->
  <div id="app"></div>

  <!-- 引入ReactJS核心库 -->
  <script type="text/javascript" src="../JS/react.development.js"></script>

  <!-- 引入React-DOM核心库,用于操作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>

  <!-- 此处类型为babel -->
  <script type="text/babel">
    class Person extends React.Component {
      state = {name:"Tom",sex:"man",age:"20"}
      
	render () {
        return (
          <ul>
            <li>name:{this.state.name}</li>
            <li>sex:{this.state.sex}</li>
            <li>age:{this.state.age}</li>
          </ul>
        )
      }
    }
    // 2、将虚拟DOM渲染到页面,标签必须闭合
   ReactDOM.render(<Person/>,document.getElementById('app'))
 </script>

此时页面展示的内容:

  • name:Tom
  • sex:man
  • age:20

以上组件的内容信息是使用状态(state)完成的,但是这样会出现一个问题:每个人的信息应该是不一样的,但是通过以上组件生成出来的人都是一模一样的的。

 ReactDOM.render(<Person/>,document.getElementById('app1'))
 ReactDOM.render(<Person/>,document.getElementById('app2'))
 ReactDOM.render(<Person/>,document.getElementById('app3'))

页面内容:

  • name:Tom

  • sex:man

  • age:20

  • name:Tom

  • sex:man

  • age:20

  • name:Tom

  • sex:man

  • age:20

我们渲染了三个人的组件信息,发现都是一模一样的,且不能改变。这是非常不合理的,所以我们要使用一个新的属性props来实现我们需要的功能。


使用props属性改造案例

   我们参考官网,使用props改造以上案例。

class Person extends React.Component {
      
      render () {
        const {name,age,sex} = this.props
        return (
          <ul>
            <li>name:{name}</li>
            <li>sex:{sex}</li>
            <li>age:{age}</li>
          </ul>
        )
      }
    }
    // 2、将虚拟DOM渲染到页面,标签必须闭合
    ReactDOM.render(<Person name="genius" age="25" sex="男"/>,document.getElementById('app'))

页面内容

  • name:genius
  • sex:男
  • age:25

以上就是我们使用props改造的案例,我们发现它和HTML标签属性的使用方式差不多,你传什么属性react就把你的属性放在props对象身上,组件内部使用的时候,就从props身上拿出来使用。

这个时候我们就可以通过多个组件实例生成不同的人的信息了

 ReactDOM.render(<Person name="Tom" age="20" sex="男"/>,document.getElementById('app1'))
 ReactDOM.render(<Person name="李琴" age="25" sex="女"/>,document.getElementById('app2'))
 ReactDOM.render(<Person name="张三" age="25" sex="男"/>,document.getElementById('app3'))

以上我们了解了props的基础使用方式。


批量传递标签属性

   以上案例我们使用的属性很少,但是在真实项目中,我们使用网络请求从数据库拿到的信息非常多,难道我们也要一个一个的往组件身上加属性吗?react为我们提供了一个便捷的方式。

const p = {name:"Tom",sex:"man",age: 20}

ReactDOM.render(<Person {...p} />,document.getElementById('app'))

我们可以使用扩展运算符将属性批量的添加到组件里面去,效果是一样的。

注意:需要我们注意的是你传入的属性名是什么,在组件内部使用的时候必须一致,否则是个空值。


控制props的数据类型与默认值

   现在我们加一个需求,让组件信息里面的年龄在真实年龄的基础上+1,并展示。

class Person extends React.Component {
      
      render () {
        const {name,age,sex} = this.props
        return (
          <ul>
            <li>name:{name}</li>
            <li>sex:{sex}</li>
            <li>age:{age + 1}</li>
          </ul>
        )
      }
    }
// 2、将虚拟DOM渲染到页面,标签必须闭合
ReactDOM.render(<Person name="Tom" age="25" sex="男"/>,document.getElementById('app'))

页面展示:

  • name:Tom
  • sex:男
  • age:251

我们可以看到,我们的年龄变成了251,说明该运算进行的是【字符串连接】而不是【数值运算】。说明是我们的props属性类型出了问题,我们修改一下

ReactDOM.render(<Person name="Tom" age={25} sex="男"/>,document.getElementById('app'))

我们在修改代码以后,在看结果:

  • name:Tom
  • sex:男
  • age:26

修改了标签属性的数据类型后,年龄运算进行了【数值运算】。

但是这是我们知道该组件该传入什么数据类型的情况下进行的修改,但是别人不一定知道,有什么方法提示别人传入标签属性的数据类型呢?如果不传或者漏传一个属性,其默认值该如何设置呢?


步骤1:引入prop-types 用于组件的属性类型限制

 <!-- 引入prop-types 用于组件的属性类型限制 -->
 <script src="../JS/prop-types.js"></script>

引入了该脚本后,全局多了一个叫PropTypes的对象。

步骤2:给组件的标签属性设置类型限制

 // 设置属性类型以及是否必填
Person.propTypes = {
    name: PropTypes.string.isRequired, // 限制字符串,且必传
    sex: PropTypes.string, // 限制字符串
    age: PropTypes.number, // 限制数值
    speak: PropTypes.func // 限制函数
}

通过给组件类一个特定的属性propTypes来规范标签属性的数据类型限制。

步骤3:给组件的标签属性设置默认值

// 对属性值设置默认值
Person.defaultProps = {
    sex: '不男不女',
    age: 18
}

通过给组件类一个特定的属性defaultProps来设置标签属性的默认值,当没有传入该属性值时显示其默认值。

步骤4:测试是否有效

const p = {name:9527, sex:'man', age:20}
ReactDOM.render(<Person {...p}  />,document.getElementById('app'))

以上配置会报错:

 Failed prop type: Invalid prop `name` of type `number` supplied to `Person`, expected `string`.

错误说明:提供给“Person”的“number”类型的无效道具“name”,预期为“string”。

说明我们的类型限制生效了。当我们将name属性的值改回字符串形式,错误就会消失。

在测试一下默认值是否生效:

const p = {name:'Tom'}
ReactDOM.render(<Person {...p}  />,document.getElementById('app'))

此时我们不传sexage属性,看看页面内容。

  • name:Tom
  • sex:不男不女
  • age:19

说明sex属性展示的是默认值,age为什么是19呢?因为我们在内部+1了,所以age也是使用的默认值18

到此说明我们的类型限制和默认值设置都已经生效了。


简化props的写法

   之前我们使用propTypesdefaultProps 都是通过类调用的,说明这是类的静态属性,我们以此修改代码:

 class Person extends React.Component {

     // 设置属性类型以及是否必填
    static propTypes = {
      name: PropTypes.string.isRequired, // 限制字符串,且必传
      sex: PropTypes.string, // 限制字符串,默认值:不男不女
      age: PropTypes.number, // 限制数值,默认值:18
      speak: PropTypes.func // 限制函数
    }

    // 对属性值设置默认值
    static defaultProps = {
      sex: '不男不女',
      age: 18
    }
      
      render () {
        console.log(this)
        const {name,age,sex} = this.props
        return (
          <ul>
            <li>name:{name}</li>
            <li>sex:{sex}</li>
            <li>age:{age + 1}</li>
          </ul>
        )
      }
 }

   
// 2、将虚拟DOM渲染到页面,标签必须闭合
ReactDOM.render(<Person name="Tom"  />,document.getElementById('app'))

扩展构造函数与props

在react官网中对构造函数是这样描述的:

如果不初始化 state 或不进行方法绑定,则不需要为 React 组件实现构造函数。

在 React 组件挂载之前,会调用它的构造函数。在为 React.Component 子类实现构造函数时,应在其他语句之前前调用 super(props)。否则,this.props 在构造函数中可能会出现未定义的 bug。

通常,在 React 中,构造函数仅用于以下两种情况:

  • 通过给 this.state 赋值对象来初始化内部 state。
  • 为事件处理函数绑定实例

constructor() 函数中不要调用 setState() 方法。如果你的组件需要使用内部 state,请直接在构造函数中为 this.state 赋值初始 state

constructor(props) {
super(props);
// 不要在这里调用 this.setState()
this.state = { counter: 0 };
this.handleClick = this.handleClick.bind(this);
}

大概意思是如果你创建了构造函数,如果不调用super(props)语句,在构造函数中的this.props出出现未定义的bug。

我们尝试一下:

constructor(props){
    super()
    console.log('constructor',this.props)
}

这里我们打印的this.props的结果是:undefined


将类式组件改造为函数式组件

   在函数组件中,props是作为函数的参数传递的。所以我们的函数式可以使用props

function Person (props) {
     const {name,age,sex} = props
     return (
          <ul>
            <li>name:{name}</li>
            <li>sex:{sex}</li>
            <li>age:{age + 1}</li>
          </ul>
        );
    }

    // 设置属性类型以及是否必填
    Person.propTypes = {
      name: PropTypes.string.isRequired, // 限制字符串,且必传
      sex: PropTypes.string, // 限制字符串,默认值:不男不女
      age: PropTypes.number, // 限制数值,默认值:18
      speak: PropTypes.func // 限制函数
    }

    // 对属性值设置默认值
    Person.defaultProps = {
      sex: '不男不女',
      age: 18
    }
    // 2、将虚拟DOM渲染到页面,标签必须闭合
    ReactDOM.render(<Person name="Tom"  />,document.getElementById('app'))

将其改为函数式组件,运行控制台不报错,且数据类型和默认值生效。


总结Props

  • 1、props是一种用于组件之间传递数据的机制。它是一个对象,包含了组件的属性和值。
  • 2、可以通过扩展运算符({…p})来批量传入标签属性
  • 3、可以通过propTypes属性来限制标签属性的数据类型
  • 4、可以通过defaultProps属性来设置标签属性的默认值
  • 5、标签属性不能修改,只能读取
  • 6、函数式组件也能使用props

相关文章:

  • 面试官:Tomcat 在 SpringBoot 中是如何启动的(一)
  • 【人工智能与深度学习】判别性循环稀疏自编码器和群体稀疏性
  • 【计算机视觉 | 目标检测】BARON:pseudo words 和 linear layer 的理解
  • 【Git从入门到精通】分支机制
  • 【jvm系列-04】精通运行时数据区共享区域---堆
  • 【机器学习】P10 从头到尾实现一个线性回归案例
  • 四个常见的Linux面试问题
  • 【面试】TCP、UDP、Socket、HTTP网络编程面试题
  • Mysql查询截取分析_慢查询日志
  • Qt 中的信息输出机制:QDebug、QInfo、QWarning、QCritical 的简单介绍和用法
  • 50 Projects 50 Days - Expanding Cards 学习记录
  • Go语言精修(尚硅谷笔记)第十一章
  • 【EHub_tx1_tx2_E100】不止科技NVISTAR ROC 300激光雷达Ubuntu18.04+ROS1ROS2 评测
  • WebRTC API
  • 大数据之Spark开发环境准备
  • [分享]iOS开发 - 实现UITableView Plain SectionView和table不停留一起滑动
  • 2017-08-04 前端日报
  • Asm.js的简单介绍
  • ComponentOne 2017 V2版本正式发布
  • docker容器内的网络抓包
  • Druid 在有赞的实践
  • iOS 颜色设置看我就够了
  • JavaScript 事件——“事件类型”中“HTML5事件”的注意要点
  • Markdown 语法简单说明
  • Object.assign方法不能实现深复制
  • Service Worker
  • Swoft 源码剖析 - 代码自动更新机制
  • web标准化(下)
  • 初探 Vue 生命周期和钩子函数
  • 高性能JavaScript阅读简记(三)
  • 如何设计一个比特币钱包服务
  • Java总结 - String - 这篇请使劲喷我
  • ​Kaggle X光肺炎检测比赛第二名方案解析 | CVPR 2020 Workshop
  • ​无人机石油管道巡检方案新亮点:灵活准确又高效
  • #每日一题合集#牛客JZ23-JZ33
  • (1)(1.19) TeraRanger One/EVO测距仪
  • (a /b)*c的值
  • (附源码)spring boot儿童教育管理系统 毕业设计 281442
  • (机器学习-深度学习快速入门)第一章第一节:Python环境和数据分析
  • (六)Hibernate的二级缓存
  • (顺序)容器的好伴侣 --- 容器适配器
  • (提供数据集下载)基于大语言模型LangChain与ChatGLM3-6B本地知识库调优:数据集优化、参数调整、Prompt提示词优化实战
  • (转)Linux整合apache和tomcat构建Web服务器
  • (转)用.Net的File控件上传文件的解决方案
  • .CSS-hover 的解释
  • .gitignore文件设置了忽略但不生效
  • .NET Remoting Basic(10)-创建不同宿主的客户端与服务器端
  • .Net 知识杂记
  • .NET 中小心嵌套等待的 Task,它可能会耗尽你线程池的现有资源,出现类似死锁的情况
  • .NET/C# 判断某个类是否是泛型类型或泛型接口的子类型
  • .Net面试题4
  • .Net转Java自学之路—SpringMVC框架篇六(异常处理)
  • .php结尾的域名,【php】php正则截取url中域名后的内容
  • ::前边啥也没有
  • @RequestMapping处理请求异常