表单与状态管理

表单与状态管理

  1. 使用 useState 初始化状态

    • 可以通过 useState 将数据存储在组件的状态中,例如将笔记数组存储在 notes 状态中。

    • 初始值可以通过 props 传递,也可以直接设置为空数组等默认值。

    • 示例:

      javascript 复制代码
      const [notes, setNotes] = useState(props.notes);
  2. 受控组件(Controlled Component)

    • 通过将输入框的 value 属性绑定到组件的状态(如 newNote),实现对输入框的控制。

    • 需要为输入框添加 onChange 事件处理函数,将输入框的值同步到组件状态中。

    • 示例:

      javascript 复制代码
      const [newNote, setNewNote] = useState('');
      const handleNoteChange = (event) => {
        setNewNote(event.target.value);
      };
      <input value={newNote} onChange={handleNoteChange} />;
  3. 表单提交

    • 为表单添加 onSubmit 事件处理函数,防止默认提交行为(event.preventDefault())。

    • 在提交时,可以根据状态创建新对象(如笔记对象),并更新状态。

    • 示例:

      javascript 复制代码
      const addNote = (event) => {
        event.preventDefault();
        const noteObject = {
          content: newNote,
          date: new Date().toISOString(),
          important: Math.random() < 0.5,
          id: notes.length + 1,
        };
        setNotes(notes.concat(noteObject));
        setNewNote('');
      };
      <form onSubmit={addNote}>...</form>;

数据过滤与显示

  1. 条件显示数据

    • 通过添加一个状态(如 showAll)来控制显示的数据。

    • 使用条件运算符(? :)或数组的 filter 方法来筛选数据。

    • 示例:

      javascript 复制代码
      const [showAll, setShowAll] = useState(true);
      const notesToShow = showAll ? notes : notes.filter(note => note.important);
  2. 切换显示状态

    • 通过按钮点击事件更新状态,从而切换显示的内容。

    • 示例:

      javascript 复制代码
      <button onClick={() => setShowAll(!showAll)}>
        show {showAll ? 'important' : 'all'}
      </button>;

练习:电话簿应用

  1. 添加名字

    • 使用受控组件管理输入框,通过表单提交将名字添加到电话簿中。

    • 示例:

      javascript 复制代码
      const [persons, setPersons] = useState([{ name: 'Arto Hellas' }]);
      const [newName, setNewName] = useState('');
      const addPerson = (event) => {
        event.preventDefault();
        const personObject = { name: newName, id: persons.length + 1 };
        setPersons(persons.concat(personObject));
        setNewName('');
      };
  2. 防止重复添加

    • 在添加名字前,检查电话簿中是否已存在该名字。

    • 如果存在,使用 alert 提示用户。

    • 示例:

      javascript 复制代码
      const addPerson = (event) => {
        event.preventDefault();
        if (persons.some(person => person.name === newName)) {
          alert(`${newName} is already added to phonebook`);
        } else {
          const personObject = { name: newName, id: persons.length + 1 };
          setPersons(persons.concat(personObject));
          setNewName('');
        }
      };
  3. 添加电话号码

    • 为表单添加第二个输入框,用于输入电话号码。

    • 示例:

      javascript 复制代码
      const [newNumber, setNewNumber] = useState('');
      const addPerson = (event) => {
        event.preventDefault();
        const personObject = { name: newName, number: newNumber, id: persons.length + 1 };
        setPersons(persons.concat(personObject));
        setNewName('');
        setNewNumber('');
      };
  4. 搜索功能

    • 添加一个搜索输入框,用于按名字过滤电话簿。

    • 使用 filter 方法实现大小写不敏感的搜索。

    • 示例:

      javascript 复制代码
      const [searchTerm, setSearchTerm] = useState('');
      const personsToShow = persons.filter(person =>
        person.name.toLowerCase().includes(searchTerm.toLowerCase())
      );
  5. 组件拆分与重构

    • 将应用拆分为多个独立组件,如搜索框、添加表单、人员列表等。

    • 在根组件中维护状态和事件处理函数。

    • 示例:

      javascript 复制代码
      const App = () => {
        return (
          <div>
            <h2>Phonebook</h2>
            <Filter ... />
            <PersonForm ... />
            <Persons ... />
          </div>
        );
      };

调试与工具

  1. React DevTools

    • 使用 React DevTools 查看组件的状态和属性,方便调试。
    • 可以直接在 DevTools 中修改状态,观察应用的变化。
  2. 调试输出

    • 在组件中临时渲染状态或其他变量,方便调试。

    • 示例:

      javascript 复制代码
      <div>debug: {newName}</div>;

注意事项

  • 状态更新的不可变性
    • 在 React 中,状态更新必须是不可变的,不能直接修改原始状态。
    • 使用数组的 concat 或对象的展开运算符等方法来创建新状态。
  • 表单默认行为
    • 表单提交时会触发页面刷新,需要通过 event.preventDefault() 阻止默认行为。
  • 组件定义位置
    • 不要在另一个组件的定义中直接定义子组件,避免因作用域或生命周期问题导致错误。
相关推荐
学嵌入式的小杨同学5 小时前
从零打造 Linux 终端 MP3 播放器!用 C 语言实现音乐自由
linux·c语言·开发语言·前端·vscode·ci/cd·vim
weixin_425543736 小时前
TRAE CN3.3.25 构建的Electron简易DEMO应用
前端·typescript·electron·vite·nestjs
Mr Xu_7 小时前
【Vue3 + ECharts 实战】正确使用 showLoading、resize 与 dispose 避免内存泄漏
前端·信息可视化·vue·echarts
0思必得07 小时前
[Web自动化] Selenium设置相关执行文件路径
前端·爬虫·python·selenium·自动化
雯0609~7 小时前
hiprint:实现项目部署与打印1-官网提供普通html版本
前端·html
不绝1918 小时前
UGUI——进阶篇
前端
Exquisite.8 小时前
企业高性能web服务器(4)
运维·服务器·前端·网络·mysql
2501_944525549 小时前
Flutter for OpenHarmony 个人理财管理App实战 - 账户详情页面
android·java·开发语言·前端·javascript·flutter
2601_949857439 小时前
Flutter for OpenHarmony Web开发助手App实战:快捷键参考
前端·flutter
wangdaoyin20109 小时前
若依vue2前后端分离集成flowable
开发语言·前端·javascript