如何在没有缩减器的情况下从深层嵌套的子组件更新祖先状态
How to update ancestor state from deeply nested child component without reducers
我有一个 React 应用程序,其结构类似于下面的代码(但更复杂)。实际上,我要更改的只是孙组件中的数据(本例中为 grandchild.name
)。但是,我必须在层次结构中的所有组件上传递 getters (props) 和 setters (onChange) 处理程序,使其不必要地复杂化(特别是如果添加了更多嵌套组件(例如 GreatGrandChild
)。
// Parent
const Parent = () => {
const [child, setChild] = useState({
name: 'Jack',
grandChild: {
name: 'John'
}
});
return (
<div>
<Child data={child} onChange={newChild => setChild(newChild)} />
</div>
);
};
// Child
const Child = props => {
const handleChange = newGrandChild => {
props.onChange({ name: props.data.name, grandChild: newGrandChild });
};
return (
<div>
<GrandChild data={props.data.grandChild} onChange={handleChange} />
</div>
);
};
// Grand Child
const GrandChild = props => {
const handleChange = e => {
props.onChange({ name: e.target.name });
};
return (
<div>
<input type='text' onChange={handleChange} />
</div>
);
};
那么如何在不在子组件中添加handleChange函数的情况下更新父状态。听说“函数柯里化”可以解决这个问题,但我不知道在这种情况下如何使用它。
P.S。我需要一个不包含 Contexts and/or Reducers 的解决方案,因为这对我的用例来说太过分了。此外,在父组件中创建“updateGrandChild”函数并将其传递给孙组件也不起作用,因为我的数据结构包含一个数组 属性,其中索引是动态的。
import { useState } from "react";
// Parent
const Parent = () => {
const [child, setChild] = useState({
name: "Jack",
grandChild: {
name: "John"
}
});
return (
<div>
<pre>{JSON.stringify(child, null, 2)}</pre>
<Child data={child} onChange={setChild} />
</div>
);
};
// Child
const Child = ({ data, onChange }) => {
return (
<div>
<GrandChild data={data.grandChild} onChange={onChange} />
</div>
);
};
// Grand Child
const GrandChild = ({ data, onChange }) => {
return (
<div>
<input
type="text"
value={data.name}
onChange={(e) =>
onChange((data) => ({
...data,
grandChild: {
...data.grandChild,
name: e.target.value
}
}))
}
/>
</div>
);
};
export default function App() {
return <Parent />;
}
我有一个 React 应用程序,其结构类似于下面的代码(但更复杂)。实际上,我要更改的只是孙组件中的数据(本例中为 grandchild.name
)。但是,我必须在层次结构中的所有组件上传递 getters (props) 和 setters (onChange) 处理程序,使其不必要地复杂化(特别是如果添加了更多嵌套组件(例如 GreatGrandChild
)。
// Parent
const Parent = () => {
const [child, setChild] = useState({
name: 'Jack',
grandChild: {
name: 'John'
}
});
return (
<div>
<Child data={child} onChange={newChild => setChild(newChild)} />
</div>
);
};
// Child
const Child = props => {
const handleChange = newGrandChild => {
props.onChange({ name: props.data.name, grandChild: newGrandChild });
};
return (
<div>
<GrandChild data={props.data.grandChild} onChange={handleChange} />
</div>
);
};
// Grand Child
const GrandChild = props => {
const handleChange = e => {
props.onChange({ name: e.target.name });
};
return (
<div>
<input type='text' onChange={handleChange} />
</div>
);
};
那么如何在不在子组件中添加handleChange函数的情况下更新父状态。听说“函数柯里化”可以解决这个问题,但我不知道在这种情况下如何使用它。
P.S。我需要一个不包含 Contexts and/or Reducers 的解决方案,因为这对我的用例来说太过分了。此外,在父组件中创建“updateGrandChild”函数并将其传递给孙组件也不起作用,因为我的数据结构包含一个数组 属性,其中索引是动态的。
import { useState } from "react";
// Parent
const Parent = () => {
const [child, setChild] = useState({
name: "Jack",
grandChild: {
name: "John"
}
});
return (
<div>
<pre>{JSON.stringify(child, null, 2)}</pre>
<Child data={child} onChange={setChild} />
</div>
);
};
// Child
const Child = ({ data, onChange }) => {
return (
<div>
<GrandChild data={data.grandChild} onChange={onChange} />
</div>
);
};
// Grand Child
const GrandChild = ({ data, onChange }) => {
return (
<div>
<input
type="text"
value={data.name}
onChange={(e) =>
onChange((data) => ({
...data,
grandChild: {
...data.grandChild,
name: e.target.value
}
}))
}
/>
</div>
);
};
export default function App() {
return <Parent />;
}