Recently, I was in a lull in the project. I have attended several interviews and found that interviewers still like to ask some questions about Redux, especially the best way to ask questions about this kind of development framework is to get to the bottom of the source code. I just took advantage of these two days to go over the Redux library from scratch, and there are some gains.
Redux source code I roughly divided into three pieces, from easy to difficult:
- State management core code
- The react – redux library
- The middleware
Handwritten source code is not the purpose, the main purpose is to see the code written by Daniel can open up the mind, and later when talking to the interviewer can fool him. The following from scratch, hand lu a set of their own Redux library, expected to achieve similar functions with the official library, and compare the official source code, see what their own writing methods have shortcomings. We’ll start today with the Redux core code.
Redux core code implementation
Before you start, review what Redux does and what problems it solves. Redux was created to address the react component’s state management. Redux internally manages a state tree and “sends” an “action” to update the state according to the Reducer provided by the developer, so that all data management is handled by Redux instead of the React component. Redux is a data management design idea, not a specific framework for React, so as long as our business is complex enough, redux can be used in any environment without React.
The Redux core has the following capabilities:
- Get current state (getState)
- Subscribe and unsubscribe
- Dispatch actions to update status (Dispatch)
- Generate actionCreator
- Merger of reducer
We implement them one by one.
Basic code structure
The core of Redux is state management. A state tree is maintained in a data warehouse. We want to provide developers with an interface to access states.
function createStore(reducer) {
var currentState; / / state
var currentReducer = reducer; // Reducer provided by external partners
/** * is exposed to the developer to get the current state */
function getState() {
return currentState;
}
return {
getState
}
}
export {
createStore
}
Copy the code
As you can see, the code is very simple. The createStore function receives a reducer, because the logic for updating the state is provided by the developer. Therefore, from the perspective of redux designer, I only receive the “logic” you give me, and the updated state is enclosed in the internal currentState object. And provide an interface function to access the object, thus protecting the internal state by means of a closure.
Implementation of distributed functions
In the Redux architecture, there is only one way to update the state, and that is to dispatch an action. The internal state object cannot be manually modified by the developer, so we also provide a Dispatch method to update the state.
function createStore(reducer) {
var currentState; / / state
var currentReducer = reducer; // Reducer provided by external partners
@param {Object} action Action Object */
function dispatch(action) {
currentState = currentReducer(currentState, action);
}
// Other code is omitted...
}
Copy the code
The above has realized the distribution function. This statement is the only one that calls the reducer function provided by the developer and passes in the action action object, that is, the updated new state overlays the old object.
However, this statement is not strict enough, so we will make our code more robust. If the action object passed is not valid (for example, there is no type attribute), our code will error.
function createStore(reducer) {
var currentState;
var currentReducer = reducer;
var isDispatching = true; // A tag is being sent
@param {Object} action Action Object */
function dispatch(action) {
// Verify the validity of the action object
if (typeof action.type === 'undefined') {
throw new Error('Action illegal ');
}
if (isDispatching) {
throw new Error('Current status being distributed... ');
}
try {
isDispatching = true;
currentState = currentReducer(currentState, action);
} finally {
isDispatching = false; }}// Other code is omitted...
}
Copy the code
The official source code also added a “distributive” flag, if the current redux call stack is distributive, will also throw an error, so that the core of the redux library distributive function has been implemented.
By the way, the Redux library calls the Dispatch method once by default, so why call it once first? Since the internal currentState object is undefined by default, to ensure that the state is initialized, we manually call the Dispatch method (since the initialization state is provided from outside) and pass in an initialization action:
// Perform a dispatch to ensure state initialization
dispatch({
type: '@@redux/INIT'
});
Copy the code
@@redux/INIT this action has no real meaning. Its purpose is to initialize the state object. I understand it’s just a name that comes to mind.
Subscribe and unsubscribe
When the state tree is updated, there may be some subsequent actions, such as updating the corresponding view in Web development, and it is not friendly to let the developer call it, so we can follow the publish-subscribe model to implement the subscription function.
The method is very simple, use an array to record the subscribed functions, when the dispatch action is complete, that is, execute “subscribe” in order:
function createStore(reducer) {
var listeners = []; // Save the subscription callback
/** * subscribe * @param {Function} listener listener listener * @returns {Function} returns unsubscribe Function */
function subscribe(listener) {
listeners.push(listener);
return function () {
listeners = listeners.filter(fn= > fn != listener);
}
}
// Other code is omitted...
}
Copy the code
Subscribe is a higher-order function that takes an external subscribe callback, appends it to the listener array, and returns the same function: unsubscribe.
Executing the unsubscribe function again thus filters out the current callback and completes the unsubscribe operation, which is implemented using the publish-subscribe pattern.
Finally, don’t forget to call the subscription function in the Dispatch method:
listeners.forEach(fn= > fn());
Copy the code
Generate actionCreator
Recall that in our development with Redux, we used a single function to return the action object. The advantage of this is to avoid writing long actionTypes by hand, which might cause errors:
/ / ActionCreator examples:
function displayBook(payload){
return {type:'DISPLAY_BOOK', payload};
}
Copy the code
This way displayBook(1001) returns the corresponding action object by calling the function. Store. Dispatch (displayBook(1001))
After getting action, the job is to distribute. It would be redundant to manually call store.dispatch() every time. Therefore, Redux provides bindActionCreator, which encapsulates the dispatch function into actionCreator. To save the developer a step to call dispatch, we implemented it.
Create a new bindactionactionine.js file and write out the function signature:
/** * Create ActionCreators * encapsulate the distributed action into the original actionCreator Object * @param {Object} ActionCreators Object collection * @param {Function} Dispatch Redux distribution method */
function bindActionCreators(actionCreators, dispatch) {}Copy the code
As you can see, the object passed in is wrapped by each actionCratore. The principle is very simple: Loop through each actionCreator method in the object and rewrite the call to the Dispatch method into the new function:
function bindActionCreators(actionCreators, dispatch) {
var boundActions = {};
Object.keys(actionCreators).forEach(key= > {
// Rewrite each actionCreator
boundActions[key] = function (. args) {
// Wrap the distribution method in the new functiondispatch(actionCreators[key](... args)); }; });return boundActions;
}
Copy the code
After processing by bindActionCreator, the code can be further simplified:
var actionCreator = bindActionCreators({displayBook},store.dispatch);
Copy the code
Direct call actionCreator. DisplayBook distribute DISPLAY_BOOK action (1001).
Merger of reducer
With the increasing complexity of redux project, there are more and more business logic of Reducer. It is obviously bad to put all the business into a Reducer function. Usually, when we develop React and Redux, Reducer and components correspond. Therefore, splitting the Reducer according to component functions can better manage the code.
Redux provides combineReducers to combine multiple reducer reducers into one. Let’s review its use:
import { combineReducers } from 'redux';
const chatReducer = combineReducers({
chatLog,
statusMessage,
userName
})
// The chatReducer function is the merged reducer
Copy the code
It can be seen that its usage is similar to the previous bindActionCreators, in that each reducer is packaged as an object and the returned result is the merged Reducer.
CombineReducers merges the reducer name into a final large state object:
Create a combinereducers. js to merge the Reducer method:
/** * Merge reducer * @param {Object} reducers reducer collection * @returns {Function} merged reducer */
function combineReducers(reducers) {
return function (state = {}, action) {
let combinedState = {}; // The synthesized state object
Object.keys(reducers).forEach(name= > {
// Execute each reducer, attach the returned state to the combinedState and name it with the reducer name
combinedState[name] = reducers[name](state[name], action);
});
returncombinedState; }}Copy the code
It can be seen that the principle and each reducer in the same loop object are merged into the final Reducer function using the reducer name.
In this way, the methods returned by the higher-order function must be classified according to the reducer name. So far the core code of the Redux library has been implemented.
In the next article, write another piece: redux middleware source code