【React】归纳篇(五)组件组合的使用 | 受控与非受控组件 | 组件的综合练习 | 获取表单数据

简介: 【React】归纳篇(五)组件组合的使用 | 受控与非受控组件 | 组件的综合练习 | 获取表单数据


今天我们来一个小练习,最常见的TODO练习,将组件进行组合使用。

首先明确以下3点:

功能界面的组件化编码流程(重要)

  • 1、拆分组件:例如把一个整体界面进行拆分,分为一个个单个组件
  • 2、实现静态组件:使用组件实现静态页面效果
  • 3、实现动态苏建:动态显示初始化数据;实现交互功能,如绑定事件监听

思考?

  • 数据应该保存在哪个组件内?

分析:

  • 1、看数据是某个组件需要,还是某些组件需要。某个组件需要就保存在某个组件。某些组件需要就保存在这些组件共同的父组件中;
  • 2、然后看数据是什么类型;

思考?

  • 子组件中改变父组件的状态:不能在子组件中直接改变父组件的状态
  • 方法:状态在哪个组件,更新状态的行为(函数)就应该定义在哪个组件
  • 子组件去调用父组件的方法
  • 父组件定义函数,传递给子组件,子组件进行调用

示例:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
</head>
<body>
    <div id="test1"></div>
    <div id="test2"></div>
    <script src="https://cdn.bootcss.com/react/16.4.0/umd/react.development.js"></script>
    <script src="https://cdn.bootcss.com/react-dom/16.4.0/umd/react-dom.development.js"></script>
    <script src="https://cdn.bootcss.com/babel-standalone/6.26.0/babel.min.js"></script>
    <script src="../js/prop-types.js"></script>
    <script type="text/babel">
       //Todo List
       //1、定义组件:将整体组件拆分为3个单个组件
       //将<Add />与<List />组件放在<App />组件中,实现一个静态组件
       //2、动态显示初始化数据——数据应该保存在哪个组件内
       //3、添加交互,监听事件绑定
       class App extends React.Component {
           constructor(props){
               super(props)
               //初始化状态设置
               this.state = {
                   todos: ['aaa','bbb','ccc']
               }
               this.addTodo = this.addTodo.bind(this);  
           }
           addTodo(todo) {
                const {todos} = this.state;
                todos.unshift(todo);
                //更新状态
                this.setState({todos});
           }
           render() {
               const {todos} = this.state;//解构赋值写法
               return (
                <div>
                    <h1>TODO LIST</h1>
                    <Add count={todos.length} addTodo={this.addTodo}/>  
                    <List todos={todos}/>
                </div>
               )
           }
       }
       class Add extends React.Component {
           constructor(props){
               super(props)
               this.handleAdd = this.handleAdd.bind(this);
           }
           handleAdd() {
                //1、读取输入的数据
                const todo = this.inputVal.value.trim()
                //2、检测合法性,合法则添加
                if(!todo){
                    return 
                }
                this.props.addTodo(todo);//因为addTodo被放在属性中传过来
                this.inputVal.value='';
           }
           render() {
               return (
                   <div>
                    <input type="text" ref={input => this.inputVal=input}/>
                    <button onClick={this.handleAdd}>增加 {this.props.count+1}</button>
                   </div>
               )
           }
       }
       Add.propTypes = {
           count: PropTypes.number.isRequired,
           addTodo:PropTypes.func.isRequired
       }
       class List extends React.Component {
           render() {
               return (
                   <ul>
                    {
                        this.props.todos.map((todo,index)=>{return <li key={index}>{todo}</li>})
                    }
                   </ul>
               )
           }
       }
       List.propTypes = {
           todos: PropTypes.array.isRequired
       }
       ReactDOM.render(<App />,document.getElementById('test1'));
    </script>
</body>
</html>

小结–组件组合编写的流程(记住流程)

  • 1、拆分组件:根据页面结构拆分组件
  • 2、实现静态组件:先给组件类指定render(),每个组件都指定。(此时没有动态数据与交互)
  • 3、实现动态组件:先初始化数据的动态显示;然后再添加交互,绑定(bind)事件监听。

在组件中收集表单数据

实际工作中,我们常常需要操作表单数据,如向服务端提交表单数据要发送请求,如检验表单数据合法性,等等。。那么这些操作之前,首先我们需要收集到表单数据。

实例

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
</head>
<body>
    <div id="test1"></div>
    <script src="https://cdn.bootcss.com/react/16.4.0/umd/react.development.js"></script>
    <script src="https://cdn.bootcss.com/react-dom/16.4.0/umd/react-dom.development.js"></script>
    <script src="https://cdn.bootcss.com/babel-standalone/6.26.0/babel.min.js"></script>
    <script src="../js/prop-types.js"></script>
    <script type="text/babel">
    //1、定义类组件
      class LoginForm extends React.Component {
          constructor(props) {
              super(props);
              //初始化状态
              this.state = {
                  psw : ''
              }
              this.handleSubmit = this.handleSubmit.bind(this);//绑定事件侦听
              this.handleChange = this.handleChange.bind(this);//绑定事件侦听
          }
          handleSubmit(event) {
            const name = this.nameInput.value;
            const {psw} = this.state;
            alert('username:'+name+' password:'+psw);
            //阻止事件的默认行为
            event.preventDefault();
          }
          //psw失去焦点事件
          handleChange(event) {
            //读取输入值
            const psw = event.target.value;
            //更新psw状态
            this.setState({psw});
          }
          render() {
              return (
                <form action="" method="" onSubmit={this.handleSubmit}>
                    <label htmlFor="username">用户名:</label>
                    <input type="text" id="username" ref={input => this.nameInput = input}/>
                    <label htmlFor="psw">密码:</label>
                    <input type="password" id="psw" value={this.state.psw} onChange={this.handleChange}/>
                    <input type="submit" value="Login In"/>
                </form>
              )
          }
      }
    //2、渲染组件
    ReactDOM.render(<LoginForm/>,document.getElementById('test1'));
    </script>
</body>
</html>

小结

包含表单的组件分为2类:

  • 受控组件:表单项输入数据能自动收集成状态state,如password,读状态state
  • 非受控组件: 需要时才手动读取表单的数据,如username,需要手动设置

“受控”与“非受控”两个概念,区别在于这个组件的状态是否可以被外部修改。一个设计得当的组件应该同时支持“受控”与“非受控”两种形式,即当开发者不控制组件属性时,组件自己管理状态,而当开发者控制组件属性时,组件该由属性控制。而开发一个复杂组件更需要注意这点,以避免只有部分属性受控,使其变成一个半受控组件。

大多数情况请,推荐使用“受控组件”来实现表单,比较符合React的思想,但是这两者的效率并没有太大的差异。



相关文章
|
3月前
|
移动开发 前端开发 JavaScript
React 表单与事件
10月更文挑战第10天
51 1
|
3月前
|
人工智能 自然语言处理 前端开发
SpringBoot + 通义千问 + 自定义React组件:支持EventStream数据解析的技术实践
【10月更文挑战第7天】在现代Web开发中,集成多种技术栈以实现复杂的功能需求已成为常态。本文将详细介绍如何使用SpringBoot作为后端框架,结合阿里巴巴的通义千问(一个强大的自然语言处理服务),并通过自定义React组件来支持服务器发送事件(SSE, Server-Sent Events)的EventStream数据解析。这一组合不仅能够实现高效的实时通信,还能利用AI技术提升用户体验。
253 2
|
2月前
|
前端开发 UED 开发者
React 数据表格分页实现
本文详细介绍了如何在React中实现数据表格的分页功能,包括基础实现、常见问题及解决方案。通过状态管理和事件处理,我们可以有效地减少页面加载时间,提升用户体验。文章提供了完整的代码示例,帮助开发者解决分页按钮样式、按钮过多和初始加载慢等问题,并给出了相应的优化方案。
110 53
|
3月前
|
前端开发 JavaScript 数据安全/隐私保护
深入探索研究React表单
【10月更文挑战第6天】
91 57
|
28天前
|
存储 前端开发 JavaScript
React 表单输入组件 Input:常见问题、易错点及解决方案
本文介绍了在 React 中使用表单输入组件 `Input` 的基础概念,包括受控组件与非受控组件的区别及其优势。通过具体代码案例,详细探讨了创建受控组件、处理多个输入字段、输入验证和格式化的方法,并指出了常见易错点及避免方法,旨在提升表单的健壮性和用户体验。
37 4
|
2月前
|
前端开发 搜索推荐 测试技术
React 数据表格排序与过滤
本文介绍了如何在 React 中实现数据表格的排序和过滤功能,从基础概念到实际代码实现,涵盖排序和过滤的基本原理、实现步骤、常见问题及解决方法。通过合理管理状态、优化性能和避免常见错误,帮助开发者提高用户体验和开发效率。
53 4
|
2月前
|
前端开发 JavaScript
React 表单处理技巧
【10月更文挑战第24天】本文从初学者角度出发,详细介绍了 React 中表单处理的基本概念、常见问题及解决方案。涵盖受控组件与非受控组件的区别、状态更新、表单验证、多字段管理及高级技巧,通过代码示例帮助读者更好地理解和应用。
87 7
|
4月前
|
前端开发 JavaScript UED
react或者vue更改用户所属组,将页面所有数据进行替换(解决问题思路)____一个按钮使得页面所有接口重新请求
在React或Vue中,若需在更改用户所属组后更新页面所有数据但不刷新整个页面,可以通过改变路由出口的key值来实现。在用户切换组成功后,更新key值,这会触发React或Vue重新渲染路由出口下的所有组件,从而请求新的数据。这种方法避免了使用`window.location.reload()`导致的页面闪烁,提供了更流畅的用户体验。
60 1
react或者vue更改用户所属组,将页面所有数据进行替换(解决问题思路)____一个按钮使得页面所有接口重新请求
|
4月前
|
前端开发
react中使用Modal.confirm数据不更新的问题解决
文章讨论了在React中使用Ant Design的`Modal.confirm`时更新数据不生效的问题,并提供了解决方案。原因是React状态更新可能是异步的,导致Modal的内容更新后不会立即反映在UI上。解决办法是在状态更新后使用`useEffect`钩子来调用Modal实例的`update`方法,从而更新Modal的内容。
131 0
react中使用Modal.confirm数据不更新的问题解决
|
4月前
|
前端开发 JavaScript
react学习(21)受控组件
react学习(21)受控组件