我有一个简单的react组件的形式,我相信有一个受控的输入:
import React from 'react';
export default class MyForm extends React.Component {
constructor(props) {
super(props);
this.state = {}
}
render() {
return (
<form className="add-support-staff-form">
<input name="name" type="text" value={this.state.name} onChange={this.onFieldChange('name').bind(this)}/>
</form>
)
}
onFieldChange(fieldName) {
return function (event) {
this.setState({[fieldName]: event.target.value});
}
}
}
export default MyForm;
当我运行我的应用程序,我得到以下警告:
警告:MyForm正在将文本类型的未受控输入更改为
控制。输入元素不应该从不受控切换到
控制(反之亦然)。决定是使用受控的还是
在组件的生命周期内不受控制的输入元素
我相信我的输入是受控的,因为它有一个值。我在想我做错了什么?
我使用的是React 15.1.0
当你在你的输入中使用onChange={this. onfieldchange ('name').bind(this)}时,你必须声明你的状态空字符串作为属性字段的值。
不正确的方法:
this.state ={
fields: {},
errors: {},
disabled : false
}
正确的方法:
this.state ={
fields: {
name:'',
email: '',
message: ''
},
errors: {},
disabled : false
}
我相信我的输入是受控的,因为它有一个值。
对于一个被控制的输入,它的值必须与状态变量的值相对应。
在您的示例中,这个条件最初没有满足,因为this.state.name最初没有设置。因此,输入最初是不受控制的。一旦onChange处理程序第一次被触发,this.state.name就会被设置。此时,上述条件满足,则认为输入是受控的。这种从不受控到受控的转变产生了上面所见的错误。
通过在构造函数中初始化this.state.name:
e.g.
this.state = { name: '' };
输入将从一开始就受到控制,从而解决问题。更多示例请参见React受控组件。
与此错误无关,您应该只有一个默认导出。上面的代码有两个。
简而言之,如果你正在使用类组件,你必须使用状态初始化输入,就像这样:
this.state = { the_name_attribute_of_the_input: "initial_value_or_empty_value" };
你必须对所有的输入都这么做你想在代码中改变它们的值。
在使用功能组件的情况下,你将使用钩子来管理输入值,并且你必须为以后想要操作的每个输入输入初始值,如下所示:
const [name, setName] = React.useState({name: 'initialValue'});
如果你不想有初始值,你可以放一个空字符串。
当你第一次呈现你的组件时,this.state.name没有设置,所以它的值为undefined或null,你最终将value={undefined}或value={null}传递给你的输入。
当ReactDOM检查一个字段是否受控制时,它会检查value != null(注意它是!=,而不是!==),由于JavaScript中的undefined == null,它决定它是不受控制的。
因此,当onFieldChange()被调用时,this.state.name被设置为一个字符串值,你的输入从不受控制变成受控制。
如果你这样做。在你的构造函数中State = {name: "},因为" != null,你的输入将一直有一个值,并且该消息将消失。