警告:组件正在更改要控制的文本类型的非受控输入。输入元件不应从非受控切换到受控(反之亦然)。决定在部件的使用寿命内使用受控或非受控输入元件*

以下是我的代码:

constructor(props) {
  super(props);
  this.state = {
    fields: {},
    errors: {}
  }
  this.onSubmit = this.onSubmit.bind(this);
}

....

onChange(field, e){
  let fields = this.state.fields;
  fields[field] = e.target.value;
  this.setState({fields});
}

....

render() {
  return(
    <div className="form-group">
      <input
        value={this.state.fields["name"]}
        onChange={this.onChange.bind(this, "name")}
        className="form-control"
        type="text"
        refs="name"
        placeholder="Name *"
      />
      <span style={{color: "red"}}>{this.state.errors["name"]}</span>
    </div>
  )
}

当前回答

如果在字段中使用多个输入,请执行以下操作:例如:

class AddUser extends React.Component {
   constructor(props){
     super(props);

     this.state = {
       fields: { UserName: '', Password: '' }
     };
   }

   onChangeField = event => {
    let name = event.target.name;
    let value = event.target.value;
    this.setState(prevState => {
        prevState.fields[name] =  value;
        return {
           fields: prevState.fields
        };
    });
  };

  render() { 
     const { UserName, Password } = this.state.fields;
     return (
         <form>
             <div>
                 <label htmlFor="UserName">UserName</label>
                 <input type="text" 
                        id='UserName' 
                        name='UserName'
                        value={UserName}
                        onChange={this.onChangeField}/>
              </div>
              <div>
                  <label htmlFor="Password">Password</label>
                  <input type="password" 
                         id='Password' 
                         name='Password'
                         value={Password}
                         onChange={this.onChangeField}/>
              </div>
         </form>
     ); 
  }
}

在以下位置搜索您的问题:

onChangeField = event => {
    let name = event.target.name;
    let value = event.target.value;
    this.setState(prevState => {
        prevState.fields[name] =  value;
        return {
            fields: prevState.fields
        };
    });
};

其他回答

发生这种情况是因为该值不能为未定义或null来解析。您可以这样做

value={ this.state.value ?? "" }

在我的情况下,这几乎是玛雅克·舒克拉的最高答案所说的。唯一的细节是我的州完全没有我所定义的财产。

例如,如果您的状态为:

state = {
    "a" : "A",
    "b" : "B",
}

如果您正在扩展代码,您可能需要添加一个新属性,因此,在代码的其他地方,您可能会创建一个新的属性c,其值不仅在组件的状态中未定义,而且属性本身也未定义。

要解决这个问题,只需确保将c添加到状态中,并给它一个适当的初始值。

例如。,

state = {
    "a" : "A",
    "b" : "B",
    "c" : "C", // added and initialized property!
}

希望我能解释我的边缘案例。

原因是,在您定义的状态中:

this.state = { fields: {} }

字段作为空白对象,因此在第一次渲染过程中,this.state.fields.name将未定义,输入字段的值将为:

value={undefined}

因此,输入字段将变得不受控制。

在输入中输入任何值后,状态中的字段将更改为:

this.state = { fields: {name: 'xyz'} }

此时,输入场被转换为受控分量;这就是为什么会出现错误:

组件正在将文本类型的非受控输入更改为受约束的。

可能的解决方案:

1-将状态中的字段定义为:

this.state = { fields: {name: ''} }

2-或使用短路评估定义值属性,如下所示:

value={this.state.fields.name || ''}   // (undefined || '') = ''

如果您正在将value属性设置为对象的属性,并希望确保该属性未定义,则可以组合空合并运算符??带有可选的链接运算符?。如下所示:

<input
  value={myObject?.property ?? ''}
/>

对我来说,这是一个错误:

<input onChange={onClickUpdateAnswer} value={answer.text}>
  {answer.text}
</input>

如您所见,我已将字符串传递到Input标记的主体中,

Fix:

<input onChange={onClickUpdateAnswer} value={answer.text}></input>