浏览器内置的 <input> 组件 允许你渲染不同类型的表单输入框。

<input />

参考

<input>

渲染 浏览器内置的 <input> 组件以展示输入框。

<input name="myInput" />

参见下面更多示例

属性

<input> 支持所有 常见的元素属性

你可以传递以下属性中的任意一个,以将其变为 受控输入框

  • checked:布尔值,控制复选框或单选按钮是否被选中。
  • value:字符串,控制文本框的输入文本(如果是单选按钮,则为其表单数据)。

当你传递它们之一时,你必须同时传递 onChange 处理函数,用于更新传递的值。

以下 <input> 属性仅在非受控输入框中有效:

  • defaultChecked:布尔值,指定 type="checkbox"type="radio" 输入的 初始值
  • defaultValue:字符串,指定文本框的 初始值

以下 <input> 属性均可用于受控与非受控输入框:

  • accept:字符串,指定 type="file" 输入框所接受的文件类型。
  • alt:字符串,指定 type="image" 输入框的替代图像文本。
  • capture:字符串,指定 type="file" 输入框所捕获的媒体(麦克风、视频或摄像头)。
  • autoComplete:字符串,指定可能的 自动填充行为 之一。
  • autoFocus:布尔值。如果为 true,React 将在挂载时聚焦于此元素。
  • dirname:字符串,指定元素的方向性的表单字段名称。
  • disabled:布尔值。如果为true,输入框将无法交互并显示为不可用(dimmed)。
  • children<input> 不接受子元素。
  • form:字符串,指定此输入框所属的 <form>id。如果未指定,则为最近的父表单。
  • formAction:字符串。输入框指定此值并指定 type="submit"type="image" 后将覆盖父表单对应属性 <form action>
  • formEnctype:字符串。输入框指定此值并指定 type="submit"type="image" 后将覆盖父表单对应属性 <form enctype>
  • formMethod:字符串。输入框指定此值并指定 type="submit"type="image" 后将覆盖父表单对应属性 <form method>
  • formNoValidate:字符串。输入框指定此值并指定 type="submit"type="image" 后将覆盖父表单对应属性 <form noValidate>
  • formTarget:字符串。输入框指定此值并指定 type="submit"type="image" 后将覆盖父表单对应属性 <form target>
  • height:字符串,指定 type="image" 的图像高度。
  • list:字符串,指定带有自动完成选项的 <datalist>id
  • max:数字,指定数值和日期时间输入的最大值。
  • maxLength:数字,指定文本和其他输入的最大长度。
  • min:数字,指定数值和日期时间输入的最小值。
  • minLength:数字,指定文本和其他输入的最小长度。
  • multiple:布尔值,指定是否允许 <type="file"type="email" 指定多个值。
  • name:字符串,指定此输入框的名称,它将 随表单一起提交
  • onChange:一个 Event 处理函数。如果这是 受控输入框,则必须提供。在用户更改输入框的值时立即触发(例如,每次按键时触发)。行为类似于浏览器的 input 事件。
  • onChangeCapture:与 onChange 类似,但是是在 捕获阶段 触发。
  • onInput:一个 Event 处理函数。在用户更改值时立即触发。由于历史原因,在 React 中习惯于使用 onChange,工作方式类似。
  • onInputCapture:与 onInput 类似,但是是在 捕获阶段 触发。
  • onInvalid:一个 Event 处理函数。在表单提交时,如果输入框未通过验证将触发。与内置的 invalid 事件不同,React 的 onInvalid 事件可以进行冒泡。
  • onInvalidCapture:与 onInvalid 类似,但是是在 捕获阶段 触发。
  • onSelect:一个 Event 处理函数。在 <input> 内的选择更改后触发。React 扩展了 onSelect 事件,使其也能在选择为空和编辑时触发(可能会影响选择)。
  • onSelectCapture:与 onSelect 类似,但是是在 捕获阶段 触发。
  • pattern:字符串,指定 value 必须匹配的模式。
  • placeholder:字符串,当输入值为空时,以暗淡的颜色显示的内容。
  • readOnly:布尔值。如果为 true,用户无法编辑输入。
  • required:布尔值。如果为 true,提交表单时必须提供此输入框的值。
  • size:数字,类似于设置宽度,但单位取决于控件。
  • src:字符串,指定 type="image" 输入框的图像源。
  • step:正数或 'any' 字符串,指定有效值之间的距离。
  • type:字符串,指定其中之一的 输入类型
  • width:字符串,指定 type="image" 输入框的图像宽度。

注意

  • 复选框需要使用 checkeddefaultChecked,而不是 valuedefaultValue
  • 如果文本框接收到字符串类型的 value 属性,则会被 视为受控组件
  • 如果复选框或单选按钮接收到布尔类型的 checked 属性,则会被 视为受控组件
  • 一个输入框不能同时既是受控组件又是非受控组件。
  • 一个输入框在其生命周期中不能在受控和非受控之间切换。
  • 每个受控组件都需要一个 onChange 事件处理函数,用于同步更新其值。

用法

展示不同类型的输入框

渲染 input 组件展示输入框。默认情况下,这是一个文本框。你可以传递 type="checkbox" 将其指定为多选框;或者传递 type="radio" 将其指定为单选按钮;你也可以将其指定为其他类型

export default function MyForm() {
  return (
    <>
      <label>
        文本框:<input name="myInput" />
      </label>
      <hr />
      <label>
        多选框<input type="checkbox" name="myCheckbox" />
      </label>
      <hr />
      <p>
        单选按钮:
        <label>
          <input type="radio" name="myRadio" value="option1" />
          选项一
        </label>
        <label>
          <input type="radio" name="myRadio" value="option2" />
          选项二
        </label>
        <label>
          <input type="radio" name="myRadio" value="option3" />
          选项三
        </label>
      </p>
    </>
  );
}


为输入框提供 label 属性

一般而言,应该将每个 <input> 都放置在 <label> 内,表示此标签与该选择框相关联。当用户单击标签时,浏览器将自动聚焦选择框。这对于可访问性也非常重要:当用户聚焦选择框时,屏幕阅读器将宣布标签标题。

如果无法将 <input> 放置在 <label> 内,请通过将相同的 ID 传递给 <input id><label htmlFor> 来将它们关联起来。为了避免一个组件的多实例之间的冲突,使用 useId 生成这样的 ID。

import { useId } from 'react';

export default function Form() {
  const ageInputId = useId();
  return (
    <>
      <label>
        你的名称:
        <input name="firstName" />
      </label>
      <hr />
      <label htmlFor={ageInputId}>你的年龄:</label>
      <input id={ageInputId} name="age" type="number" />
    </>
  );
}


为输入框提供初始值

你可以为任何输入框指定初始值。如果要为文本框指定初始值,传递字符串 defaultValue;如果要为复选框和单选按钮指定初始值,传递布尔值 defaultChecked

export default function MyForm() {
  return (
    <>
      <label>
        文本框:<input name="myInput" defaultValue="Some initial value" />
      </label>
      <hr />
      <label>
        多选框:<input type="checkbox" name="myCheckbox" defaultChecked={true} />
      </label>
      <hr />
      <p>
        单选按钮:
        <label>
          <input type="radio" name="myRadio" value="option1" />
          选项一
        </label>
        <label>
          <input
            type="radio"
            name="myRadio"
            value="option2"
            defaultChecked={true} 
          />
          选项二
        </label>
        <label>
          <input type="radio" name="myRadio" value="option3" />
          选项三
        </label>
      </p>
    </>
  );
}


提交表单时读取输入框的值

在输入框周围添加一个包含 <button type="submit"> 按钮的 <form> 组件。这将调用 <form onSubmit> 事件处理程序。默认情况下,浏览器将向当前 URL 发送表单数据并刷新页面。你可以通过调用 e.preventDefault() 取消此默认行为,并使用 new FormData(e.target) 读取表单数据。

export default function MyForm() {
  function handleSubmit(e) {
    // 阻止浏览器重新加载页面
    e.preventDefault();

    // 读取表单数据
    const form = e.target;
    const formData = new FormData(form);

    // 你可以直接将 formData 作为 fetch 的请求 body:
    fetch('/some-api', { method: form.method, body: formData });

    // 也可以使用普通的对象:
    const formJson = Object.fromEntries(formData.entries());
    console.log(formJson);
  }

  return (
    <form method="post" onSubmit={handleSubmit}>
      <label>
        文本框:<input name="myInput" defaultValue="Some initial value" />
      </label>
      <hr />
      <label>
        多选框: <input type="checkbox" name="myCheckbox" defaultChecked={true} />
      </label>
      <hr />
      <p>
        单选按钮:
        <label><input type="radio" name="myRadio" value="option1" /> 选项一</label>
        <label><input type="radio" name="myRadio" value="option2" defaultChecked={true} /> 选项二</label>
        <label><input type="radio" name="myRadio" value="option3" /> 选项三</label>
      </p>
      <hr />
      <button type="reset">重置表单</button>
      <button type="submit">提交表单</button>
    </form>
  );
}

注意

<input> 添加 name 属性,例如 <input name="firstName" defaultValue="Taylor" />。指定的 name 将作为表单数据中的一个键,例如 { firstName: "Taylor" }

陷阱

默认情况下,<form> 内的任何 <button> 都可以提交表单。这可能会让人感到惊讶!如果你有自定义 Button 组件,请考虑使用 <button type="button"> 而不是 <button>。如果你想要明确指定提交表单的按钮,请使用 <button type="submit">


使用 state 控制输入框

<input /> 这样的输入框是非受控的。即使你像 <input defaultValue="Initial text" /> 一样 传递了初始值,你的 JSX 也只是指定了初始值,而非当前时刻的值。

如果要渲染一个受控输入框,请传递 value 属性(或者向多选框和单选按钮传递 checked。React 将强制传递 value 属性给输入框。通常,你可以通过声明一个 state 来控制输入框。

function Form() {
const [firstName, setFirstName] = useState(''); // 声明一个 state 变量...
// ...
return (
<input
value={firstName} // ...强制输入框的值与 state 相匹配...
onChange={e => setFirstName(e.target.value)} // ... 并在每次改变(change)时更新 state!
/>
);
}

当你需要 state 时,受控输入框都将非常有用——比如,每次编辑时都重新渲染 UI:

function Form() {
const [firstName, setFirstName] = useState('');
return (
<>
<label>
你的名称:
<input value={firstName} onChange={e => setFirstName(e.target.value)} />
</label>
{firstName !== '' && <p>你的名称是 {firstName}</p>}
...

如果你想提供多种方式来调整输入框 state(例如,通过单击按钮),它也会很有用:

function Form() {
// ...
const [age, setAge] = useState('');
const ageAsNumber = Number(age);
return (
<>
<label>
年龄:
<input
value={age}
onChange={e => setAge(e.target.value)}
type="number"
/>
<button onClick={() => setAge(ageAsNumber + 10)}>
增加 10 年
</button>

传递给受控组件的 value 属性不能是 undefinednull。如果你需要初始值为空(例如,下面的 firstName 字段),请将你的 state 变量初始化为空字符串('')。

import { useState } from 'react';

export default function Form() {
  const [firstName, setFirstName] = useState('');
  const [age, setAge] = useState('20');
  const ageAsNumber = Number(age);
  return (
    <>
      <label>
        名:
        <input
          value={firstName}
          onChange={e => setFirstName(e.target.value)}
        />
      </label>
      <label>
        年龄:
        <input
          value={age}
          onChange={e => setAge(e.target.value)}
          type="number"
        />
        <button onClick={() => setAge(ageAsNumber + 10)}>
          增加 10 年
        </button>
      </label>
      {firstName !== '' &&
        <p>你的名称是 {firstName}</p>
      }
      {ageAsNumber > 0 &&
        <p>你的年龄是 {ageAsNumber}</p>
      }
    </>
  );
}

陷阱

如果传递了 value 但没有传递 onChange,那么将无法输入内容。当你通过传递 value 来控制输入框时,你需要保证输入框始终具有你传递的值。因此,如果你将一个 state 作为 value 传递,但在 onChange 事件处理程序中忘记同步更新该状态变量,React 将在每次输入后将输入框恢复到你指定的 value。


优化每次按键时的重新渲染

当你使用受控输入框时,每次按键都会设置 state。如果包含 state 的组件重新渲染了大型树形结构,这可能会变得很慢。有几种方法可以优化重新渲染的性能。

例如,假设表单在每次按键时会重新渲染所有页面内容:

function App() {
const [firstName, setFirstName] = useState('');
return (
<>
<form>
<input value={firstName} onChange={e => setFirstName(e.target.value)} />
</form>
<PageContent />
</>
);
}

由于 <PageContent /> 不依赖于输入框 state,因此可以将输入框 state 移入其自己的组件中:

function App() {
return (
<>
<SignupForm />
<PageContent />
</>
);
}

function SignupForm() {
const [firstName, setFirstName] = useState('');
return (
<form>
<input value={firstName} onChange={e => setFirstName(e.target.value)} />
</form>
);
}

这样可以显著提高性能,因为每次按键时只有 SignupForm 会重新渲染。

如果无法避免重新渲染(例如,如果 PageContent 依赖于搜索输入框的值),useDeferredValue 可以帮助你在大型重新渲染过程中保持受控输入框的响应性(responsive)。


故障排除

输入时文本框未更新

如果传递了 value 属性给输入框,而没有传递 onChange 属性,那么你将在控制台看到错误信息:

// 🔴 Bug:没有 onChange 事件处理程序的受控文本框
<input value={something} />
Console
你传递了 value 属性给表单,但是没有传递 onChange 事件处理程序。这将使其变为只读。如果该字段应该是可变的,请使用 defaultValue。否则,设置 onChangereadOnly

正如错误信息所提示的,如果你仅仅是想要 指定初始值,请传递 defaultValue

// ✅ Good:有初始值的非受控输入框
<input defaultValue={something} />

如果你想要 使用 state 变量控制输入框,指定 onChange 事件处理程序:

// ✅ Good:具有 onChange 事件处理程序的受控输入框
<input value={something} onChange={e => setSomething(e.target.value)} />

如果输入框是只读的,额外传递 readOnly 属性:

// ✅ Good:没有 onChange 事件处理程序的受控只读输入框
<input value={something} readOnly={true} />

点击时多选框未更新

如果传递了 checked 属性给多选框,而没有传递 onChange 属性,那么你将在控制台看到错误信息:

// 🔴 Bug:没有 onChange 事件处理程序的受控多选框
<input type="checkbox" checked={something} />
Console
你传递了 checked 属性给表单,但是没有传递 onChange 事件处理程序。这将使其变为只读。如果该字段应该是可变的,请使用 defaultValue。否则,设置 onChangereadOnly

正如错误信息所提示的,如果你仅仅是想要 指定初始值,请传递 defaultValue

// ✅ Good:有初始值的非受控多选框
<input type="checkbox" defaultChecked={something} />

If you want to control this checkbox with a state variable, specify an onChange handler:

// ✅ Good:具有 onChange 事件处理程序的受控多选框
<input type="checkbox" checked={something} onChange={e => setSomething(e.target.checked)} />

陷阱

如果你想要读取多选框中选中的值,应该使用 e.target.checked 而不是 e.target.value

如果多选框是只读的,额外传递 readOnly 属性:

// ✅ Good:没有 onChange 事件处理程序的受控只读输入框
<input type="checkbox" checked={something} readOnly={true} />

当我输入时,输入框光标会跳到开头

如果你想要 控制输入框,你应该在 onChange 期间将对应的 state 变量更新为来自 DOM 的输入框的值。

你不应该将它更新为 e.target.value(对于多选框应该是 e.target.checked)以外的值:

function handleChange(e) {
// 🔴 Bug:将输入框更新为 e.target.value 以外的值
setFirstName(e.target.value.toUpperCase());
}

你也不应该异步更新:

function handleChange(e) {
// 🔴 Bug:异步更新输入框
setTimeout(() => {
setFirstName(e.target.value);
}, 100);
}

将 state 同步更新为 e.target.value 以解决此问题:

function handleChange(e) {
// ✅ 将受控输入框的值同步更新为 e.target.value
setFirstName(e.target.value);
}

如果这不能解决问题,有可能是因为每次输入时输入框都从 DOM 中删除并重新添加。同样,如果在每次重新渲染时不小心 重置了 state,就会发生这种情况。例如,如果输入框或其祖先组件总是接收不同的 key,或者嵌套使用组件(这在 React 中是不允许的,并且会导致“内部”组件在每次渲染时重新挂载),就会发生这种情况。


收到错误:“A component is changing an uncontrolled input to be controlled”

提供的 value 属性必须在整个生命周期中都为字符串。

你不能一会传递 value={undefined} 一会传递 value="some string",这会导致 React 不清楚你是想指定受控组件还是非受控组件。受控组件的 value 属性应该始终接收字符串,而不是 nullundefined

如果 value 来自 API 或 state,它可能会被初始化为 nullundefined。在这种情况下,要么最初将其设置为空字符串(''),要么传递 value={someValue ?? ''} 以确保 value 是一个字符串。

类似的,如果传递 checked 属性给多选框,请确保它始终是布尔值。