我是Redux的新手,我有一个问题,我制作了一个简单的Redux应用程序,它具有两种不同的功能,一种是基于计数的增量递减,其次是单击按钮然后更改图像,所以我真正的问题是当我先更新状态增量减量,然后工作良好,但是如果我按按钮单击,然后再次按增量或减量按钮,然后向我显示错误NAN,而不是增量递减结果1 2 0r其他,然后在控制台中向我显示波纹管错误:
Received NaN for the `children` attribute. If this is expected, cast the value to a string.
“我尝试了许多方式,例如以多种不同方式调度动作,但无论如何对我来说都是行不通的。”
请检查代码并指导我
constants.js
export const CLICK_ME = 'CLICK_ME'
export const INCREMENT = 'INCREMENT'
export const DECREMENT = 'DECREMENT'
action.js
import {CLICK_ME, INCREMENT, DECREMENT} from '../constant/constant'
export const clicking = () => {
return {
type:CLICK_ME
}
}
export const increment =()=>{
return {
type : INCREMENT
}
}
export const decrement =()=>{
return {
type : DECREMENT
}
}
reducer.js
import {CLICK_ME, INCREMENT, DECREMENT} from '../constant/constant'
const initialState = {
click : true,
count: 0
}
const reducer = (state = initialState , action) => {
if(action.type === CLICK_ME){
return {click : !state.click}
}
else if(action.type === INCREMENT ){
return { count: state.count + 1 }
}
else if(action.type === DECREMENT ){
return { count: state.count - 1 }
}
else return state
}
export default reducer
store.js
import reducer from '../reducers/reducer'
import { createStore } from 'redux'
const store = createStore(reducer)
export default store
index.js
import React from 'react';
import ReactDOM from 'react-dom';
import './index.css';
import App from './App';
import * as serviceWorker from './serviceWorker';
import 'bootstrap/dist/css/bootstrap.css'
import store from './store/store'
import { Provider } from 'react-redux'
ReactDOM.render(
<Provider store={store} >
<App />
</Provider>
, document.getElementById('root'));
serviceWorker.unregister();
App.js
import React from 'react';
import {connect} from 'react-redux'
import { CLICK_ME, INCREMENT, DECREMENT } from './constant/constant';
function App(props) {
return (
<div className="container pt-5 mt-5 ">
<div className="row">
<div className="col col-sm-8 col-md-8 col-lg-8 col-xl-8 ">
{ props.click ? <img src="" className="img-fluid rounded-circle" width="30%" height="auto" alt=""/> : <img src="https://cdn2.iconfinder.com/data/icons/user-needs-19/16/45_false_delete_remove_cross_wrong_2-512.png" className="img-fluid rounded-circle" alt="" width="30%" height="auto" />}
</div>
<div className="col col-sm-4 col-md-4 col-lg-4 col-xl-4 ">
<p className="text-danger" >THE NUMBER IS COUNT ON EVERY CLICK </p>
<code className="p-5 " style={{fontSize:"100px"}} >{props.count}</code>
</div>
</div>
<hr className="bg-info p-1 m-1" />
<button className="btn btn-danger m-1" onClick={props.increment} >Increment</button>
<button className="btn btn-danger m-1" onClick={props.decrement} >Decrement</button>
<button className="btn btn-primary bg-info float-right" onClick={props.clicks} >Click me to change the state</button>
</div>
);
}
const mapStateToProps = (state)=>{
return {
click : state.click,
count : state.count
}
}
const dispatchToProps = (dispatch) => {
return {
clicks:()=>dispatch({type:CLICK_ME}),
increment :()=>dispatch({type:INCREMENT}),
decrement :()=>dispatch({type:DECREMENT})
}
}
export default connect(mapStateToProps,dispatchToProps)(App)
答案 0 :(得分:1)
您的问题出在reducer内部,当您调度操作并更新redux存储时,您将返回已更新的状态,仅返回要更新的字段,而不是该状态的先前值以及已更新的字段加值。
const reducer = (state = initialState , action) => {
if(action.type === CLICK_ME){
return {
...state,
click : !state.click
}
}
else if(action.type === INCREMENT ){
return {
...state,
count: state.count + 1
}
}
else if(action.type === DECREMENT ){
return {
...state,
count: state.count - 1
}
}
else return state
}
我建议使用这种方式编写化简: Is this redux reducer OK
答案 1 :(得分:1)
缺少的关键是您需要从化简函数中返回整个状态,而不仅是通过该操作修改过的片段。这意味着复制您未修改的属性。
按如下所示更改reducer函数(我还建议您使用switch
语句,它的键入较少,并且与其他Redux代码一致):
import {CLICK_ME, INCREMENT, DECREMENT} from '../constant/constant'
const initialState = {
click : true,
count: 0
}
const reducer = (state = initialState , action) => {
switch (action.type) {
case CLICK_ME:
return { ...state, click: !state.click };
case INCREMENT:
return { ...state, count: state.count + 1 };
case DECREMENT:
return { ...state, count: state.count - 1 };
default:
return state;
}
}
export default reducer;
如果您不熟悉它,则...state
使用 object spread 运算符,该运算符会将state
中的所有属性及其值复制到新对象上。或者,您可以使用等效的Object.assign({}, state, { modified: 'state' })
。