我正在为多频道聊天应用创建React / Redux前端。在使用redux
,react-redux
和redux-thunk
进行状态更改后,我无法让一些React组件重新渲染。
我相信我的缩减器是非变异的,并且我通过react-redux
connect
订阅了。当我运行应用程序并查看浏览器控制台时,我看到组件的初始渲染(即具有初始,空状态),然后状态更改(由index.js
中的操作调度触发)....我然后会期望组件使用新道具重新渲染,但它不会发生。
我在这里张贴了一个回购: https://github.com/mattmoss/react-redux-no-update
node_modules
不在repo中,因此要运行,首先下载依赖项(运行yarn
就足够了),然后npm start
。
一些摘录(参见回购资料中的完整资料):
reducers/channelList.js
import * as c from '../actions/constants';
export default function channelList(state = [], action) {
switch (action.type) {
case c.FETCH_CHANNELS_SUCCESS:
return action.channels;
default:
return state;
}
}
actions/channelActions.js
export function fetchChannels() {
return (dispatch) => {
return ChannelApi.allChannels()
.then(channels => dispatch(fetchChannelsSuccess(channels)))
.catch(error => { throw(error); });
};
}
export function fetchChannelsSuccess(channels) {
return {
type: c.FETCH_CHANNELS_SUCCESS,
channels
};
}
components/ChannelListView.js
class ChannelListView extends React.Component {
render() {
const { channels, current, onSelect } = this.props;
console.log("channels:", channels, "current:", current);
return (
<ListGroup>
{channels.map(channel =>
<ListGroupItem
key={channel.id}
active={channel.id === this.props.current}
onClick={onSelect(channel.id)}
>
<strong>#{channel.name}</strong>
</ListGroupItem>
)}
</ListGroup>
);
}
}
export default ChannelListView;
containers/ChannelList.js
import ChannelListView from '../components/ChannelListView';
const mapStateToProps = (state, ownProps) => {
return {
channels: state.channelList,
current: state.currentChannel
};
};
const mapDispatchToProps = (dispatch) => {
return {
onSelect: (id) => () => {}
};
};
export default connect(mapStateToProps, mapDispatchToProps)(ChannelListView);
App.js
class App extends Component {
render() {
return (
<Grid>
<Row>
<Col>
<h1>Channels</h1>
<ChannelList />
</Col>
</Row>
</Grid>
);
}
}
index.js
const store = configureStore();
store.dispatch(fetchChannels());
ReactDOM.render(
<Provider store={configureStore()}>
<App />
</Provider>,
document.getElementById('root')
);
store/configureStore.js
import { createStore, applyMiddleware } from 'redux';
import rootReducer from '../reducers/rootReducer';
import thunk from 'redux-thunk';
import logger from 'redux-logger';
export default function configureStore() {
return createStore(
rootReducer,
applyMiddleware(thunk, logger)
);
}
答案 0 :(得分:4)
我不是100%,因为我自己对React来说还是比较新的。但请查看您的index.js
脚本。
// You configure the store, then dispatch the fetchChannels action
const store = configureStore();
store.dispatch(fetchChannels());
ReactDOM.render(
// But here, you're recreating the store again, which I think will re-initialise an empty store
// Change this to use the `store` variable from above.
<Provider store={configureStore()}>
<App />
</Provider>,
document.getElementById('root')
);