ChatGPT解决这个技术问题 Extra ChatGPT

一个组件正在将文本类型的不受控制的输入更改为 ReactJS 中的受控错误

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

以下是我的代码:

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>
  )
}
状态中 fields 的初始值是多少?
构造函数(道具){ 超级(道具); this.state = { 字段:{},错误:{} } this.onSubmit = this.onSubmit.bind(this); }
我刚刚发现您可以使用 useRef 有条件地将值设置为当前输入,例如 value={amountInputFocused ? amountRef.current?.value : amountState}。不确定这是否是设计使然,但它可以工作,并且可以消除错误。

B
Bondolin

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

this.state = { fields: {} }

字段作为空白对象,因此在第一次呈现期间 this.state.fields.name 将是 undefined,输入字段将获得其值:

value={undefined}

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

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

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

那时输入字段被转换为受控组件;这就是您收到错误的原因:

组件正在更改要控制的文本类型的不受控输入。

可能的解决方案:

1- 将状态中的 fields 定义为:

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

2-或使用 Short-circuit evaluation 定义 value 属性,如下所示:

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

是否有原因 undefined 是“不受控制的”而后者是“受控的” - 这些是什么意思?
因为 '' 是一个有效的字符串值。因此,当您将 '' 更改为“xyz”时,输入类型不会改变意味着受控到受控。但是在 undefinedxyz 的情况下,类型将从不受控制变为受控。
谢谢这对我有用:value={this.state.fields.name || ''}
我不知道一旦值变得未定义,输入就会自动不受控制,太好了。已经解决了我的问题
惊人!因为我没有在官方文档中看到这一点,还是我瞎了?请链接到来源:)
b
blueseal

value 更改为 defaultValue 将解决它。

笔记:

defaultValue 仅用于初始加载。如果你想初始化输入那么你应该使用defaultValue,但是如果你想使用state来改变值那么你需要使用value。阅读本文了解更多信息。

我在 input 中使用了 value={this.state.input ||""} 来消除该警告。


谢谢!我遇到了这个问题,其他解决方案不适用于我,但这个解决方案有帮助。
您也可以对功能组件应用相同的答案。示例:value={user || ""}
T
Tabish

在组件内部按以下方式放置输入框。

<input className="class-name"
              type= "text"
              id="id-123"
              value={ this.state.value || "" }
              name="field-name"
              placeholder="Enter Name"
              />

L
Lynden Noye

除了接受的答案之外,如果您使用的是 checkboxradio 类型的 input,我发现我还需要检查 checked 属性是否为空/未定义。

<input
  id={myId}
  name={myName}
  type="checkbox" // or "radio"
  value={myStateValue || ''}
  checked={someBoolean ? someBoolean : false}
  />

如果您使用的是 TS(或 Babel),则可以使用空值合并而不是逻辑 OR 运算符:

value={myStateValue ?? ''}
checked={someBoolean ?? false}

这是解决复杂问题的简单思维,酷@Lynden Noye
V
Vaibhav KB

很简单,您必须先设置初始状态

如果您不设置初始状态,react 会将其视为不受控制的组件


M
Michael Nelles
const [name, setName] = useState()

一旦您在文本字段中输入,就会生成错误

const [name, setName] = useState('') // <-- by putting in quotes 

将解决此字符串示例的问题。


A
Aindriú

如上所述,您需要设置初始状态,在我的情况下,我忘记在 setSate(); 中添加 ' ' 引号。

  const AddUser = (props) => {
  const [enteredUsername, setEnteredUsername] = useState()
  const [enteredAge, setEnteredAge] = useState()

给出以下错误

https://i.stack.imgur.com/AlFd4.png

正确的代码是简单地将初始状态设置为空字符串''

  const AddUser = (props) => {
  const [enteredUsername, setEnteredUsername] = useState('')
  const [enteredAge, setEnteredAge] = useState('')

原来是那个错误!!多谢。
我遇到了同样的问题,只是我没有使用空字符串作为初始值,而是使用 undefined。切换到空字符串解决了这个问题。谢谢
J
Jean Hilaire Messeroux

发生这种情况是因为该值不能是 undefined 或 null 来解决你可以这样做

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

n
noone

首先设置当前状态...this.state

这是因为当您要分配一个新状态时,它可能是未定义的。所以它也将通过设置状态提取当前状态来修复

this.setState({...this.state, field})

如果你的状态中有一个对象,你应该如下设置状态,假设你必须在用户对象中设置用户名。

this.setState({user:{...this.state.user, ['username']: username}})

据我了解, setState 已经只覆盖了字段,所以在第一种情况下,应该没有必要解构分配。在第二种状态下,它可能是必要的,因为 setState 将替换一个子对象批发。
A
Alec Mingione

解决此问题的最佳方法是将初始状态设置为 ''

constructor(props) {
 super(props)
  this.state = {
    fields: {
      first_name: ''
    }
  }
  this.onChange = this.onChange.bind(this);
}

onChange(e) {
  this.setState({
    fields:{
     ...this.state.fields,
     [e.target.name]: e.target.value
    }
  })
}


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

然后,您仍然可以像 if (field) 一样运行检查,并且如果您的值为 '',仍然可以获得相同的结果。

现在,因为您的值现在被归类为类型字符串而不是评估后未定义。因此,从一个大红色块😁😎的控制台清除错误。


M
Max Angelo Dapitilla Perin

如果值不存在或为空,则放置空值。

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

谢谢麦克斯!这就是问题的原因,但是,您的代码片段不会按原样适用于该用户(this.state.fields.namethis.state.value
m
m02ph3u5

如果您在 on 字段中使用多个输入,请遵循:例如:

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
        };
    });
};

z
zrna

使用 React Hooks 也不要忘记设置初始值。
我使用的是 <input type='datetime-local' value={eventStart} />,初始 eventStart 就像

const [eventStart, setEventStart] = useState();
改为
const [eventStart, setEventStart] = useState('');

括号中的空字符串是差异。此外,如果您像我一样在提交后重置表单,您需要再次将其设置为空字符串,而不仅仅是空括号。

这只是我对这个话题的一点贡献,也许它会对某人有所帮助。


J
Justine Mahinyila

我使用反应钩子遇到了同样的警告,尽管我之前已经将初始状态初始化为:-

const [post,setPost] = useState({title:"",body:""})

但后来我在 onChange 事件处理程序上覆盖了预定义状态对象的一部分,

 const onChange=(e)=>{
        setPost({[e.target.name]:e.target.value})
    }

解决方案我通过首先处理前一个状态的整个对象(通过使用扩展运算符)然后在它之上进行编辑来解决这个问题,

 const onChange=(e)=>{
        setPost({...post,[e.target.name]:e.target.value})
    }

1
10110

就我而言,这几乎就是 Mayank Shukla 的最佳答案所说的。唯一的细节是我的状态完全没有我定义的属性。

例如,如果您有这种状态:

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

如果您正在扩展您的代码,您可能想要添加一个新道具,因此,您可以在代码中的其他地方创建一个新属性 c,其值不仅在组件的状态上未定义,而且属性 本身< /em> 未定义。

要解决这个问题,只需确保将 c 添加到您的状态中并为其提供适当的初始值。

例如,

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

希望我能够解释我的边缘情况。


N
Nicolas Hedoire

警告:组件正在更改要控制的文本类型的不受控输入。输入元素不应从不受控切换到受控(反之亦然)。决定在组件的生命周期内使用受控输入元素还是不受控输入元素。

解决方案:检查值是否未定义

React / Formik / Bootstrap / TypeScript

例子 :

{ values?.purchaseObligation.remainingYear ?
  <Input
   tag={Field}
   name="purchaseObligation.remainingYear"
   type="text"
   component="input"
  /> : null
}

M
Muhammad Minhaj

当输入字段值未定义时出现此问题的原因然后从反应中抛出警告。如果您为多个输入字段创建一个 changeHandler 并且您想使用 changeHandler 更改状态,那么您需要使用扩展运算符分配先前的值。就像我在这里的代码一样。

constructor(props){
    super(props)
    this.state = {
        user:{
            email:'',
            password:''
        }
    }
}

// This handler work for every input field
changeHandler = event=>{
    // Dynamically Update State when change input value
    this.setState({
        user:{
            ...this.state.user,
            [event.target.name]:event.target.value
        }
    })
}

submitHandler = event=>{
    event.preventDefault()

    // Your Code Here...
}

render(){
    return (
        <div className="mt-5">
       
            <form onSubmit={this.submitHandler}>
                <input type="text" value={this.state.user.email} name="email" onChage={this.changeHandler} />
                
                <input type="password" value={this.state.user.password} name="password" onChage={this.changeHandler} />

                <button type="submit">Login</button>
            </form>
      

        </div>
    )
}

L
Latief Anwar

像这样

value={this.state.fields && this.state.fields["name"] || ''}

为我工作。

但我这样设置初始状态:

this.state = {
  fields: [],
}

a
andromeda

如果您将 value 属性设置为对象的属性并希望确保该属性不是未定义的,则可以将空值合并运算符 ?? 与可选的链接运算符 ?. 组合起来,如下所示:

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

k
khizer

可以应用多种方法:

基于类的方法:使用本地状态并使用默认值定义现有字段:

constructor(props) {
    super(props);
    this.state = {
      value:''
    }
  }
<input type='text'
                  name='firstName'
                  value={this.state.value}
                  className="col-12"
                  onChange={this.onChange}
                  placeholder='Enter First name' />

在功能样式组件中使用 Hooks React > 16.8:

[value, setValue] = useState('');
<input type='text'
                  name='firstName'
                  value={value}
                  className="col-12"
                  onChange={this.onChange}
                  placeholder='Enter First name' />

如果在功能样式的 HOC 组件的情况下使用 propTypes 并为 propTypes 提供默认值。

 HOC.propTypes = {
    value       : PropTypes.string
  }
  HOC.efaultProps = {
    value: ''
  }

function HOC (){

  return (<input type='text'
                  name='firstName'
                  value={this.props.value}
                  className="col-12"
                  onChange={this.onChange}
                  placeholder='Enter First name' />)

}



D
Demetrio Gomez

改变这个

  const [values, setValues] = useState({intialStateValues});

为了这

  const [values, setValues] = useState(intialStateValues);

s
srigu

我也面临同样的问题。我的解决方案是我错过了向元素添加“名称”属性。

<div className="col-12">
   <label htmlFor="username" className="form-label">Username</label>
        <div className="input-group has-validation">
             <span className="input-group-text">@</span>
                  <input 
                      type="text" 
                      className="form-control" 
                      id="username"
                      placeholder="Username" 
                      required=""
                      value={values.username}
                      onChange={handleChange}
                  />
                  <div className="invalid-feedback">
                      Your username is required.
                  </div>
        </div>
</div>

在我在属性的输入列表中引入 name = username 之后,它工作正常。


A
Alexej Holad

对于功能组件:

const SignIn = () => {

  const [formData, setFormData] = useState({
    email: "",
    password: ""
  });

  
  const handleChange = (event) => {
    const { value, name } = event.target;
    setFormData({...formData, [name]: value });
  };


  const handleSubmit = (e) => {
    e.preventDefault();
    console.log("Signed in");
    setFormData({
      email: "",
      password: ""
    });
  };


  return (
    <div className="sign-in-container">
      <form onSubmit={handleSubmit}>
        <FormInput
          name="email"
          type="email"
          value={formData.email}
          handleChange={handleChange}
          label="email"
          required
        />
        <FormInput
          name="password"
          type="password"
          value={formData.password}
          handleChange={handleChange}
          label="password"
          required
        />
        <CustomButton type="submit">Sign in</CustomButton>
      </form>
    </div>
  );
};

export default SignIn;

A
Akaisteph7

虽然这听起来很疯狂,但为我解决这个问题的方法是添加一个额外的 div。部分代码为例:

  ... [Other code] ...
  const [brokerLink, setBrokerLink] = useState('');
  ... [Other code] ...

  return (
    ... [Other code] ...
              <div styleName="advanced-form" style={{ margin: '0 auto', }}>
                {/* NOTE: This div */}
                <div>
                  <div styleName="form-field">
                    <div>Broker Link</div>
                    <input
                      type="text"
                      name="brokerLink"
                      value={brokerLink}
                      placeholder=""
                      onChange={e => setBrokerLink(e.target.value)}
                    />
                  </div>
                </div>
              </div>
    ... [Other code] ...
  );
... [Other code] ...

很奇怪。如果没有这个额外的 div,看起来 react 最初渲染的 input 元素没有 value 属性,但由于某种原因有一个空的 style 属性。您可以通过查看 html 来了解这一点。这导致控制台警告..

更奇怪的是,添加一个不是空字符串的默认值或执行类似 value={brokerLink || ''} 的操作会导致完全相同的问题。

另一个奇怪的事情是我有 30 个其他元素几乎完全相同,但没有导致这个问题。唯一的区别是这个brokerLink没有那个外部div..并且将它移动到代码的其他部分而不更改任何内容由于某种原因删除了警告..

如果没有我的确切代码,可能几乎不可能复制。如果这不是反应中的错误或其他什么,我不知道是什么。


A
Adnan Siddique

我是 reactjs 新手,我正在使用 reactjs 版本 17 我遇到了这个问题

我解决了:

而不是这个

const [email, setEmail] = useState();

我添加了这个

const [email, setEmail] = useState("");

在 useState 函数中,我添加了引号来初始化数据并且错误消失了。