- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我遇到了一个关于基于属性的状态的问题。
我有一个组件父组件,它创建将属性传递给子组件。子组件根据收到的属性使用react。据我所知,在 React 中,更改组件状态的“唯一”正确方法是使用函数 componentWillMount 或 componentDidMount 和 componentWillReceiveProps(以及其他函数,但让我们重点关注这些函数,因为 getInitialState 仅执行一次)。
如果我从父级收到一个新属性并且想要更改状态,则只有函数 componentWillReceiveProps 才会被执行,并且允许我执行 setState。渲染不允许设置状态。
如果我想在开始时和接收新属性时设置状态怎么办?所以我必须在 getInitialState 或 componentWillMount/componentDidMount 上设置它。然后,您必须使用 componentWillReceiveProps 根据属性更改状态。
当您的状态高度依赖于您的属性时(这几乎总是如此),这就是一个问题。这可能会变得愚蠢,因为您必须根据新属性重复要更新的状态。
我创建了一个新方法,在 componentWillMount 和 componentWillReceiveProps 上调用它。我还没有发现在渲染之前更新属性之后以及第一次安装组件时调用任何方法。那么就不需要执行这种愚蠢的解决方法。
无论如何,这里有一个问题:在收到或更改新属性时是否有更好的选择来更新状态?
/*...*/
/**
* To be called before mounted and before updating props
* @param props
*/
prepareComponentState: function (props) {
var usedProps = props || this.props;
//set data on state/template
var currentResponses = this.state.candidatesResponses.filter(function (elem) {
return elem.questionId === usedProps.currentQuestion.id;
});
this.setState({
currentResponses: currentResponses,
activeAnswer: null
});
},
componentWillMount: function () {
this.prepareComponentState();
},
componentWillReceiveProps: function (nextProps) {
this.prepareComponentState(nextProps);
},
/*...*/
我觉得自己有点傻,我想我失去了一些东西......我想还有另一种解决方案可以解决这个问题。
是的,我已经知道了: https://facebook.github.io/react/tips/props-in-getInitialState-as-anti-pattern.html
最佳答案
我发现这种模式通常不是很有必要。 在一般情况(并非总是如此),我发现根据更改的属性设置状态有点反模式;相反,只需在渲染时导出必要的本地状态即可。
render: function() {
var currentResponses = this.state.candidatesResponses.filter(function (elem) {
return elem.questionId === this.props.currentQuestion.id;
});
return ...; // use currentResponses instead of this.state.currentResponses
}
但是,在某些情况下,缓存这些数据是有意义的(例如,计算它的成本可能非常昂贵),或者您只需要知道何时因其他原因设置/更改 Prop 。在这种情况下,我基本上会使用您在问题中编写的模式。
如果您真的不喜欢输入它,您可以将这个新方法形式化为 mixin。例如:
var PropsSetOrChangeMixin = {
componentWillMount: function() {
this.onPropsSetOrChange(this.props);
},
componentWillReceiveProps: function(nextProps) {
this.onPropsSetOrChange(nextProps);
}
};
React.createClass({
mixins: [PropsSetOrChangeMixin],
onPropsSetOrChange: function(props) {
var currentResponses = this.state.candidatesResponses.filter(function (elem) {
return elem.questionId === props.currentQuestion.id;
});
this.setState({
currentResponses: currentResponses,
activeAnswer: null
});
},
// ...
});
当然,如果您使用基于 class
的 React 组件,您需要找到一些替代解决方案(例如继承或自定义 JS mixin),因为它们不支持 React-现在就设计 mixin。
(就其值(value)而言,我认为使用显式方法代码会更清晰;我可能会这样写:)
componentWillMount: function () {
this.prepareComponentState(this.props);
},
componentWillReceiveProps: function (nextProps) {
this.prepareComponentState(nextProps);
},
prepareComponentState: function (props) {
//set data on state/template
var currentResponses = this.state.candidatesResponses.filter(function (elem) {
return elem.questionId === props.currentQuestion.id;
});
this.setState({
currentResponses: currentResponses,
activeAnswer: null
});
},
关于reactjs - 当属性更改并首次安装在 React 上时更改状态 - 缺少功能?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29397669/
你能比较一下属性吗 我想禁用文本框“txtName”。有两种方式 使用javascript,txtName.disabled = true 使用 ASP.NET, 哪种方法更好,为什么? 最佳答案 我
Count 属性 返回一个集合或 Dictionary 对象包含的项目数。只读。 object.Count object 可以是“应用于”列表中列出的任何集合或对
CompareMode 属性 设置并返回在 Dictionary 对象中比较字符串关键字的比较模式。 object.CompareMode[ = compare] 参数
Column 属性 只读属性,返回 TextStream 文件中当前字符位置的列号。 object.Column object 通常是 TextStream 对象的名称。
AvailableSpace 属性 返回指定的驱动器或网络共享对于用户的可用空间大小。 object.AvailableSpace object 应为 Drive 
Attributes 属性 设置或返回文件或文件夹的属性。可读写或只读(与属性有关)。 object.Attributes [= newattributes] 参数 object
AtEndOfStream 属性 如果文件指针位于 TextStream 文件末,则返回 True;否则如果不为只读则返回 False。 object.A
AtEndOfLine 属性 TextStream 文件中,如果文件指针指向行末标记,就返回 True;否则如果不是只读则返回 False。 object.AtEn
RootFolder 属性 返回一个 Folder 对象,表示指定驱动器的根文件夹。只读。 object.RootFolder object 应为 Dr
Path 属性 返回指定文件、文件夹或驱动器的路径。 object.Path object 应为 File、Folder 或 Drive 对象的名称。 说明 对于驱动器,路径不包含根目录。
ParentFolder 属性 返回指定文件或文件夹的父文件夹。只读。 object.ParentFolder object 应为 File 或 Folder 对象的名称。 说明 以下代码
Name 属性 设置或返回指定的文件或文件夹的名称。可读写。 object.Name [= newname] 参数 object 必选项。应为 File 或&
Line 属性 只读属性,返回 TextStream 文件中的当前行号。 object.Line object 通常是 TextStream 对象的名称。 说明 文件刚
Key 属性 在 Dictionary 对象中设置 key。 object.Key(key) = newkey 参数 object 必选项。通常是 Dictionary 
Item 属性 设置或返回 Dictionary 对象中指定的 key 对应的 item,或返回集合中基于指定的 key 的&
IsRootFolder 属性 如果指定的文件夹是根文件夹,返回 True;否则返回 False。 object.IsRootFolder object 应为&n
IsReady 属性 如果指定的驱动器就绪,返回 True;否则返回 False。 object.IsReady object 应为 Drive&nbs
FreeSpace 属性 返回指定的驱动器或网络共享对于用户的可用空间大小。只读。 object.FreeSpace object 应为 Drive 对象的名称。
FileSystem 属性 返回指定的驱动器使用的文件系统的类型。 object.FileSystem object 应为 Drive 对象的名称。 说明 可
Files 属性 返回由指定文件夹中所有 File 对象(包括隐藏文件和系统文件)组成的 Files 集合。 object.Files object&n
我是一名优秀的程序员,十分优秀!