admin管理员组文章数量:1340292
I have experienced useEffect function from React hook. When I use it with useSelector from React-redux, it worked so weird.
My reducer:
const initialState = { a: 0 };
function mainReducer(state = initialState, action) {
const { type, payload } = action;
switch (type) {
case 'A': return { ...state, a: payload }
default:
return state;
}
}
My Component:
function MyComponent(props) {
const { a } = useSelector(state => state.mainReducer);
const dispatch = useDispatch();
useEffect(() => {
console.log('did mount: ', a);
dispatch({ type: 'A', payload: 1 })
}, []);
useEffect(() => {
console.log('use effect: ', a);
dispatch({ type: 'A', payload: a });
}, [a])
return (<View style={styles.container}>
<Text style={styles.text}>{a}</Text>
</View>);
};
Result
Log:
did mount ran: 0
useEffect ran: 0
The last result is variable 'a' = 0
????
As I understand,
After the first render, both effect ran sequentially in their order in the code.
(Step 1) So the first effect run fist -> log 'did mount ran: 0'. Then it dispatch value 1 to store
(Step 2) The second effect run after -> log 'did mount ran: 0'. Then it dispatch value 0 to store
But what I don't understand is the second effect must track the change from variable 'a', so there will be:
In the following render time:
(Step 3) the second useEffect should be run when the value 'a' change from 0 to 1 (from Step 1).
And then:
(Step 4) it should have the third re-render when the value change again from 1 to 0 (from Step 2)
So the log should be:
did mount ran: 0
useEffect ran: 0
useEffect ran: 1
useEffect ran: 0
Could you please explain to me what I'm missing? Thank you
I have experienced useEffect function from React hook. When I use it with useSelector from React-redux, it worked so weird.
My reducer:
const initialState = { a: 0 };
function mainReducer(state = initialState, action) {
const { type, payload } = action;
switch (type) {
case 'A': return { ...state, a: payload }
default:
return state;
}
}
My Component:
function MyComponent(props) {
const { a } = useSelector(state => state.mainReducer);
const dispatch = useDispatch();
useEffect(() => {
console.log('did mount: ', a);
dispatch({ type: 'A', payload: 1 })
}, []);
useEffect(() => {
console.log('use effect: ', a);
dispatch({ type: 'A', payload: a });
}, [a])
return (<View style={styles.container}>
<Text style={styles.text}>{a}</Text>
</View>);
};
Result
Log:
did mount ran: 0
useEffect ran: 0
The last result is variable 'a' = 0
????
As I understand,
After the first render, both effect ran sequentially in their order in the code.
(Step 1) So the first effect run fist -> log 'did mount ran: 0'. Then it dispatch value 1 to store
(Step 2) The second effect run after -> log 'did mount ran: 0'. Then it dispatch value 0 to store
But what I don't understand is the second effect must track the change from variable 'a', so there will be:
In the following render time:
(Step 3) the second useEffect should be run when the value 'a' change from 0 to 1 (from Step 1).
And then:
(Step 4) it should have the third re-render when the value change again from 1 to 0 (from Step 2)
So the log should be:
did mount ran: 0
useEffect ran: 0
useEffect ran: 1
useEffect ran: 0
Could you please explain to me what I'm missing? Thank you
Share Improve this question edited Jun 20, 2020 at 9:12 CommunityBot 11 silver badge asked Feb 3, 2020 at 8:50 Nam LeNam Le 1011 gold badge2 silver badges4 bronze badges 5-
useSelector(state => state.mainReducer);
!? back to learning redux – xadm Commented Feb 3, 2020 at 8:58 - ``` import { bineReducers } from 'redux'; export default bineReducers({ mainReducer }); ``` – Nam Le Commented Feb 3, 2020 at 9:01
- I have this also @xadm – Nam Le Commented Feb 3, 2020 at 9:01
- what should be in selector? – xadm Commented Feb 3, 2020 at 9:02
- 1 use proper selector and change order of useEffects - it should work as expected (in some sense - react for redux store changes, not amount of changes) – xadm Commented Feb 3, 2020 at 9:38
2 Answers
Reset to default 5Your both dispatch are called after first render so even before your second render value is 0 so your second useEffect won't be able detect change as there is no change.
Let's see what is happening in your render method
First Render:
a = 0
first useEffect: dispatch({ a : 1 })
second useEffect: dispatch({ a : 0 })
so now in your redux store a is 0.
Second Render
a = 0
first useEffect: doesn't run as there is no dependency
second useEffect: doesn't run as a hasn't changed.
PLEASE, stop using
useSelector(state => state.mainReducer);
it doesn't make any sense
there should be a simple state transformation (subselection)
const a = useSelector(state => state.a)
taken directly from redux docs:
const counter = useSelector(state => state.counter)
update
you can see effect (from store change) with slightly changed ponent
function MyComponent(props) {
const a = useSelector(state => state.a);
const dispatch = useDispatch();
console.log('render: ', a);
useEffect(() => {
console.log('use effect: ', a);
dispatch({ type: 'A', payload: a });
}, [a])
useEffect(() => {
console.log('did mount: ', a);
dispatch({ type: 'A', payload: 1 })
}, []);
return (<View style={styles.container}>
<Text style={styles.text}>{a}</Text>
</View>);
};
It should result in log:
render: 0
// initial stateuse effect: 0
// first effect run- // dispatch
0
... processed in store by reducer but results in the same state ...
// ... and in our rendering process we still working on an 'old'a
readed from state on the beginning of render did mount: 0
// 'old'a
// dispatch1
... changed state in redux store.... rendered text
0
...
...//
useSelector
forces rerendering - change detectedrender: 1
// latest dispatched value, processed by reducers into new state, rereaded by selectoruse effect: 1
//useEffect
works AS EXPECTED as an effect ofa
change- .... rendered text
1
...
...
- no more rerenderings - latest dispach not changed state
Of course dispatch from other ponent will force update in this ponent ... if value will be different.
本文标签: javascriptuseEffect not working when dependency value changedStack Overflow
版权声明:本文标题:javascript - useEffect not working when dependency value changed - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743631364a2513164.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论