Enter the long winded section before learning
React source code Analysis All articles are original, combined with the summary and analysis of others online source code. I think the article may not be the best one, but I’m sure it’s a more detailed one. If something is wrong, please correct it, because you are also a fool.
I think the Scheduler module is the difficult part, and every time I try to chew on it, I always want to find a reason to back off. Luck held on. Thanks also to the article shockw4ver, which inspired me the most, and gave me the train of thought for learning. Since I was summarizing while learning, I underestimated the learning content of this part. I planned to end the article in one, but I found that the content was really too much, so I decided to divide it into two summaries. Let’s begin our first study:
- Tips: The source code in this article does not show all the code associated with this part of react, only the main flow. Because of the amount of code, it is not easy to understand the whole. You can read the article while reading the source code, will understand faster.
1. MessageChannel versus requestIdleCallback
In React Fiber, we examine why time sharding is necessary. RequestIdleCallback can be executed in priority order when the browser is idle. React did start with requestIdleCallback. However, React enabled the new task scheduling method in a release on September 27, 2019. In PR, you can see why it was replaced with MessageChannel.
- Due to the
requestIdleCallback
Depending on the refresh rate of the monitor, you need to look at it when using itVsync cycle (hardware device frequency)
The face of - In order to perform as many tasks as possible in each frame, the
5ms interval message event
To initiate a dispatch, using the postMessage approach - The postMessage approach also carries risks, as more frequent scheduling of tasks intensifies
The main thread
withOther Browser Tasks
Competition for resources - Compared to the
requestIdleCallback
andsetTimeout
The extent to which the browser throttles the message event under the background TAB needs to be further determined. This experiment assumes that it has the same priority as the timer.
By adopting MessageChannel, you abandon the browser’s frame-dependent strategy and instead control the frequency of scheduling. Improves task processing speed and optimizes the React runtime performance.
1.1 What is a MessageChannel
RequestIdleCallback API has been studied in detail in the React Fiber framework, so we will also learn the basic usage of MessageChannel. A MessageChannel is a point-to-point conduit for communication. It allows blocks of code to run independently in different browser contexts. Different context, through which it establishes the pipeline for message transmission. MessageChannel has the following features:
- Two read-only
MessagePort
: port1 and port2. These two ports form the bridge over which the message channel takes over - One port is used locally and the other port is passed to the remote window
- Messages are delivered as DOM events without interrupting or blocking task tasks in the event loop
var channel = new MessageChannel();
var port1 = channel.port1;
var port2 = channel.port2;
port1.onmessage = function(event) {
console.log("Port1 receives data from Port2:" + event.data);
}
port2.onmessage = function(event) {
console.log("Port2 receives data from port1:" + event.data);
}
port1.postMessage(Port2 "hello");
port2.postMessage("Port1 I got your regards.");
Copy the code
We’ll take a peek at MessageChannel here, but it’s enough to understand this step to read the React source code. Learn more about it later
2. Principle of Schedule
I have read the Schedule source code, but THERE is no overall concept. Later, I found react Scheduler on the Internet and analyzed this article again, which I think is very good. Xiuyan is the object of my worship, and this article was written by Xiufan. I wonder if I will become a pious disciple of a family. (Hee hee) Of course, there is no doubt that I will use the knowledge I understand from the passage.
2.1 priority type timeoutForPriorityLevel (/ scheduler/SRC/scheduler. Js)
Let’s start by looking at the method of getting the task expiration time. You can see the division of task priorities in Schedule, with 5 total priorities in total.
function timeoutForPriorityLevel(priorityLevel) {
switch (priorityLevel) {
case ImmediatePriority:
return IMMEDIATE_PRIORITY_TIMEOUT;
case UserBlockingPriority:
return USER_BLOCKING_PRIORITY_TIMEOUT;
case IdlePriority:
return IDLE_PRIORITY_TIMEOUT;
case LowPriority:
return LOW_PRIORITY_TIMEOUT;
case NormalPriority:
default:
returnNORMAL_PRIORITY_TIMEOUT; }}// maxSigned31BitInt: an integer of up to 31 bits. For 32-bit systems, the maximum integer size in V8.
var maxSigned31BitInt = 1073741823;
// Times out immediately
var IMMEDIATE_PRIORITY_TIMEOUT = -1;
// Eventually times out
var USER_BLOCKING_PRIORITY_TIMEOUT = 250;
var NORMAL_PRIORITY_TIMEOUT = 5000;
var LOW_PRIORITY_TIMEOUT = 10000;
// Never times out
var IDLE_PRIORITY_TIMEOUT = maxSigned31BitInt;
Copy the code
In the following table, you can see the corresponding way: The higher the number, the longer the timeout period and the lower the priority.
Priority priorityLevel | The field names | describe |
---|---|---|
NoPriority = 0: indicates NoPriority | React initializes and resets root; User-defined use | |
ImmediatePriority = 1: Immediate | IMMEDIATE_PRIORITY_TIMEOUT = -1 | Times out immediately: indicates that the command Times out immediately |
UserBlockingPriority = 2: UserBlockingPriority | USER_BLOCKING_PRIORITY_TIMEOUT = 250 | Timeout period = Current time + 250 |
NormalPriority = 3: indicates the NormalPriority | NORMAL_PRIORITY_TIMEOUT = 5000 | Timeout period = Current time + 5000 |
LowPriority = 4: indicates the LowPriority | LOW_PRIORITY_TIMEOUT = 10000 | Timeout period = Current time + 10000 |
IdlePriority = 5: indicates the priority of idle reexecution | IDLE_PRIORITY_TIMEOUT = maxSigned31BitInt | There is no timeout and execution may be blocked forever |
2.2 Task Queue
Scheduler plays an important role in React updates. Scheduler is used for both synchronous and asynchronous rendering, for first rendering and for updating nodes. Scheduler uses the Heap to hold the task queue. And adopt certain rules for task scheduling. Let’s start by learning the generation rules for this task queue.
2.1 unstable_scheduleCallback (priorityLevel, callback, the options)
Scheduler’s entry function is unstable_scheduleCallback. The function calculates the start time and expiration time of the task, generates the task, and queues the task.
/** * priorityLevel: specifies the priority of the task. Corresponding value and the priority described above. * options: {delay: XXX, timeout: XXX} Determines the startTime and timeout of the task. * /
unstable_scheduleCallback(priorityLevel, callback, options){}Copy the code
2.2.1 Calculating startTime and timeout
The calculation time of unstable_scheduleCallback can be divided into the following three steps:
- StartTime = options.delay(existing) + currentTime
- Timeout = Expiration time returned based on priority (value corresponding to timeoutForPriorityLevel method above)
- expirationTime = startTime + timeout
function unstable_scheduleCallback(priorityLevel, callback, options) {
// Get the current time
var currentTime = getCurrentTime();
var startTime;
if (typeof options === 'object'&& options ! = =null) {
var delay = options.delay;
if (typeof delay === 'number' && delay > 0) {
startTime = currentTime + delay;
} else{ startTime = currentTime; }}else {
startTime = currentTime;
}
// timeout = IMMEDIATE_PRIORITY_TIMEOUT | USER_BLOCKING_PRIORITY_TIMEOUT | ....
var timeout = timeoutForPriorityLevel(priorityLevel);
// Expiration time = startTime + timeout
var expirationTime = startTime + timeout;
}
Copy the code
According to the bureau, actuallyTask execution time = Current time currentime + Delay + Priority timing (xxx_PRIORITY_TIMEOUT)
According to the code, draw the following flow chart:
2.2.2 Task Queue Maintenance
To understand a little bit about execution, Scheduler is used to schedule tasks. In order to prevent the main thread of the browser from running something for a long time, critical tasks are postponed, so tasks need to be prioritized. Perform the task with the highest priority each time. Scheduler uses the Heap to hold the task queue.
Shceudler swaps tasks in the heap by bit-picking the median node based on their priority. Tools in Shceduler (such as push,peek, etc.) are used to add and remove tasks. Finally, two task queues, taskQueue and timeQueue, are constructed.
- TaskQueue: Stores the tasks to be processed
- TimeQueue: Stores deferred tasks
Before each task places the Heap, it prioritizes the Heap according to the sortIndex and ID attributes.
- SortIndex: The smaller the value, the higher the priority
- Id: indicates the task creation sequence. A smaller ID indicates a higher priority
Scheduler first peek a taskQueue task and executes it, then pops it off the task list. During the task execution, check whether the current time meets the delay criteria of the timeQueue task. If yes, remove the timeQueue task and add it to taskQueue…. Let’s look at the task maintenance process in code.
2.2.2.1 Task Creation
TaskIdCounter = taskIdCounter = taskIdCounter = taskIdCounter = taskIdCounter = taskIdCounter
var taskIdCounter = 1;
function unstable_scheduleCallback(priorityLevel, callback, options) {
// Create the task expiration time code above
/ /...
// Create new task
var newTask = {
id: taskIdCounter++,
callback, // callback: this is the task we want to perform
priorityLevel,
startTime,
expirationTime,
sortIndex: -1};/ /... Deposit task
}
Copy the code
Task field parsing:
field | describe |
---|---|
id | Id of the task node, passed when creating a tasktaskIdCounter On the 1 |
callback | Task function Perform content |
priorityLevel | Task priority. The priority should be ImmediatePriority, UserBlockingPriority, NormalPriority, LowPriority, and IdlePriority in descending order |
startTime | Time stamps, tasksExpected to perform Time. The default value is the current time, that is, the synchronization task. Options. delay can be set to an asynchronous delay task |
expirationTime | Expiration time, based on which scheduler schedules asynchronous tasks. Timeout is set to expirationTime or priorityLevel is used to calculate the timeout value. Timeout and startTime are called expirationTime |
sortIndex | The default value is -1. For asynchronous delayed tasks, this value would be assigned to expirationTime |
2.2.2.1 Adding a Task
// Create a task queue
var taskQueue = []; // Store timely tasks
var timerQueue = []; // Store 'delay task' delay task, i.e. 'startTime > currentTime' task
var isSchedulerPaused = false; // Whether the scheduler breaks: Pausing the scheduler is useful for debugging.
var currentTask = null;
var currentPriorityLevel = NormalPriority; // The current priority is normal by default (this variable is used globally, often with getCurrentPriorityLevel, which returns this value)
var isPerformingWork = false; // This is set during work to prevent re-entry.
// Whether the callback function is scheduled (whether the main task is scheduled)
var isHostCallbackScheduled = false;
var isHostTimeoutScheduled = false;
function unstable_scheduleCallback(priorityLevel, callback, options) {
// Calculate the expiration time
// ...
// Create new task
// ...
// startTime > currentTime: indicates that the execution time has not reached. It's a delayed task
if (startTime > currentTime) {
// Set startTime to sortIndex, which is used to sort tasks in taskQueue/ timeQueue, as mentioned earlier
newTask.sortIndex = startTime; // This is a delayed task.
// Store asynchronous tasks (sort)
push(timerQueue, newTask);
// taskQueue Has no immediate task and the new task has the highest priority
if (peek(taskQueue) === null && newTask === peek(timerQueue)) {
// All tasks are delayed, and this is the task with the earliest delay.
// All tasks are deferred because this is the highest priority deferred task.
if (isHostTimeoutScheduled) {
// Cancel the previous deferred task
cancelHostTimeout();
} else {
isHostTimeoutScheduled = true;
}
//startTime -currentTime After seconds, run handleTimeout to push tasks in timeQueue to taskQueuerequestHostTimeout(handleTimeout, startTime - currentTime); }}else {
// Set expirationTime as sortIndex for a taskQueue that is about to be executed
newTask.sortIndex = expirationTime;
push(taskQueue, newTask);
IsHostCallbackScheduled: Indicates whether a primary task is being executed
// isPerformingWork: a flag to confirm that performWorkUntilDeadline is in the recursive execution state
if(! isHostCallbackScheduled && ! isPerformingWork) {// No primary task is executed and performWorkUntilDeadline is not recursively called into the process
isHostCallbackScheduled = true; requestHostCallback(flushWork); }}return newTask;
}
Copy the code
Global variable resolution:
field | describe |
---|---|
taskQueue | Storing timely tasks |
timerQueue | storedelay task Delayed task, i.estartTime > currentTime The task of |
currentTask | Current Task |
currentPriorityLevel | By default, the current priority is normal (this variable is used globally, often with getCurrentPriorityLevel, which returns this value). |
isPerformingWork | This is set at work time to prevent re-entry. |
2.2.2.2 unstable_scheduleCallback General process
At this point the task has been created and the expiration time has been calculated. The task queue is created:
- Calculate expiration time
- Create a task
StartTime (currentTime + delay) > currentTime
: indicates a delayed task. In the timeQunene. If it’s priorityHighest latency task
andNo timely task (taskQueue is empty)
byrequestHostTimeout
willtimeQueue
Push task intaskQueue
In the- If it is
Real-time tasks
Is stored to a taskQueue. If no primary task is executed andperformWorkUntilDeadline
Also no recursive call, then callrequestHostCallback
The task scheduling function is displayed
The flow chart is as follows:
The 2.3 Scheduler MessageChannel (SchedulerHostConfig. Default. Js)
Finally, we get to the MessageChannel step, where the Scheduler module manages the execution logic for the callback after the redraw is complete. The methods mentioned above used requestHostCallback, requestHostTimeout, and cancelHostTimeout are all part of this section. Let’s start studying.
2.3.1 Overview of method variables
Here is Sheduler’s core function for task sharding. Let’s learn one by one.
- RequestHostCallback:
redraw
After completion according toThread idleness
withTask timeout
In theSpecific time
Perform tasks. - RequestHostTimeout:
- CancelHostCallback: Cancels the task.
- ShouldYieldToHost: whether to interrupt. This is used to determine whether the task times out and needs to be interrupted. Timeout is suspended.
- GetCurrentTime: Gets the current time.
- RequestPaint: Requests to draw. If the browser supports
navigator.scheduling.isInputPending
If there is a new input and draw, this function will return true, otherwise return false.
Related global variable parsing:
field | describe |
---|---|
isMessageLoopRunning | Indicates whether the current message loop is open |
scheduledHostCallback | The current task will assign the callback function to this variable in requestHostCallback |
taskTimeoutID | Current task ID (variable returned by setTimeout) |
yieldInterval | The interval between task sharding. As we mentioned above,requestIdleCallback Is to use the idle time of the browser to schedule tasks. React now has to control itselfTime slicing . This variable is the interval time. The connection is defined as 5ms. |
deadline | Deadline. I’ll look it up later in the codedeadline = currentTime + yieldInterval; So I have a rough idea of what this thing is |
needsPaint | Whether to draw. (According to the browserUser input andBrowser drawing To judge, more on that later) |
maxYieldInterval | Maximum interval between terminations |
// These methods are the state of the export function, which is called externally
export let requestHostCallback; // The thread is free to execute tasks in the taskQueue
export let cancelHostCallback; // Cancel the task
export let requestHostTimeout;
export let cancelHostTimeout;
export let shouldYieldToHost; // Whether the task needs to be interrupted (based on timeout)
export let requestPaint; / / empty function
export let getCurrentTime; // Get the current time
export let forceFrameRate;
// Indicates whether the current message loop is open
let isMessageLoopRunning = false;
// The current task will assign the callback function to this variable in requestHostCallback
let scheduledHostCallback = null;
// Current task ID (variable returned by setTimeout)
let taskTimeoutID = -1;
// The interval between task fragments. The connection is defined as 5ms.
let yieldInterval = 5;
let deadline = 0;
// Maximum interval between terminations
const maxYieldInterval = 300;
// Whether to draw
let needsPaint = false;
// Mount the window variable to the local variable
const setTimeout = window.setTimeout;
const clearTimeout = window.clearTimeout;
const requestAnimationFrame = window.requestAnimationFrame;
const cancelAnimationFrame = window.cancelAnimationFrame;
Copy the code
2.3.2 getCurrentTime Obtains the current time
// Support performance now()
const hasPerformanceNow = typeof performance === 'object' && typeof performance.now === 'function';
if (hasPerformanceNow) {
const localPerformance = performance;
getCurrentTime = () = > localPerformance.now();
} else {
const localDate = Date;
// initialTime Initialization time of application startup
const initialTime = localDate.now();
getCurrentTime = () = > localDate.now() - initialTime;
}
Copy the code
performance.now()
There is,currentTime = performance.now
.performance.now()
Is not present,currentTime = new Date.now() - initialTime
.initialTime
Is a variable generated by the React application initialization (as is the value)Date.now()
), but this value does not change in the later stage.Date.now() - initialTime
Application = the ReactInitialize the
Time spent.
2.3.2.1 Difference between performance.now() and date.now ()
The timestamp returned by performing.now () is not limited to the accuracy of one millisecond, but uses a floating point number to achieve the accuracy of microseconds. Perform.now () increases slowly at a constant rate and is not affected by system time (it may be adjusted by other software)
2.3.3 shouldYieldToHost and requestPaint
Navigator. scheduling is mentioned here, but I don’t really know much about it. A quick Google search and Scheduler parsing:
- React has a further consideration for environments that support the navigator.scheduling property, namely
Browser drawing
和User input
Do it first. That’s actually how React was designedScheduling sectionOf course, since this attribute is not universally supported, the definition in the else branch is purely to determine whether the deadline has passed. Consider the robustness of the API.
What if the browser supports Navigator? .scheduling? IsInputPending, so that the browser is more accurate to confirm the presence of browser drawing and user input. This case should take precedence, and the requestPaint function is called to set needsPaint = true.
Please use the following code to stabilize, I think shouldYieldToHost is basically clear.
/ / support the navigator. The scheduling
if(enableIsInputPending && navigator? .scheduling? .isInputPending ! = =undefined
) {
const scheduling = navigator.scheduling;
shouldYieldToHost = function() {
// Get the current time (performance.now())
const currentTime = getCurrentTime();
// Current time > End time
if (currentTime >= deadline) {
// There is no time left. We need to temporarily relinquish control of the main thread. Because browsers have more advanced tasks. Mainly rendering and user input.
// If a render is pending or input is pending,
// Then we should pause. But if neither exists, then we can
// Reduce the frequency of pauses while maintaining the response speed.
// We'll end up pausing anyway, because there might be pending renders that haven't been committed yet
// Accompany calls to 'requestPaint' or other main thread tasks
// Just like cyber events.
if (needsPaint || scheduling.isInputPending()) {
// There is either a pending paint or a pending input.
// There is waiting drawing and input
return true;
}
// When there is no input to be processed. Yield terminates only when a maximum value is reached
return currentTime >= maxYieldInterval;
} else {
// Do not terminate, there is still time left
return false; }};// Return true if input and draw are pending
requestPaint = function() {
needsPaint = true;
};
} else {
// 'isInputPending' is unavailable. We don't know if the input is pending, so we always pause at the end of the frame
shouldYieldToHost = function() {
return getCurrentTime() >= deadline;
};
// 'requestPaint' is invalid because we pause every frame anyway.
requestPaint = function() {};
}
Copy the code
3.4.5 forceFrameRate
The yieldInterval is the interval that the React controls the execution of time-sharding tasks. This method is a public API provided to developers. You can set the scheduling interval interval depending on the refresh of different devices.
forceFrameRate = function(fps) {
if (fps < 0 || fps > 125) {
console['error'] ('forceFrameRate takes a positive int between 0 and 125, ' + 'forcing frame rates higher than 125 fps is not supported');
return;
}
if (fps > 0) {
yieldInterval = Math.floor(1000 / fps);
} else {
// Obviously, if there is no transmission or a negative transmission, reset to 5ms
yieldInterval = 5; }};Copy the code
3.4.6 MessageChannel(requestHostCallback、requestHostTimeout、cancelHostCallback / performWorkUntilDeadline)
Read the code with comments first:
const channel = new MessageChannel(); // Define a channel object
const port = channel.port2; // Define port to execute channel port2
channel.port1.onmessage = performWorkUntilDeadline;
// It is used to execute the task at a specific time after the redraw is completed according to the idle degree of the thread and the timeout time of the task
requestHostCallback = function(callback) {
// Set the task to be executed immediately
scheduledHostCallback = callback;
if(! isMessageLoopRunning) {// If the state is closed, a task is coming, turn on the message loop flag
isMessageLoopRunning = true;
// Initiates the message, and the performWorkUntilDeadline function receives the message and executes the task
port.postMessage(null);
}
// else what happens?
};
// Cancel the task
cancelHostCallback = function() {
// Set the immediate task to null
scheduledHostCallback = null;
};
// The next iteration of the browser eventLoop's timer phase executes the callback, or if a specific time is passed in
requestHostTimeout = function(callback, ms) {
taskTimeoutID = setTimeout(() = > {
callback(getCurrentTime());
}, ms);
};
// The related cancel method is to clear the timer and reset the taskTimoutID
cancelHostTimeout = function() {
clearTimeout(taskTimeoutID); // Clear timer
taskTimeoutID = -1; // Set taskTimeoutID to -1
};
}
const performWorkUntilDeadline = () = > {
// Check whether 'execute task now' is empty
if(scheduledHostCallback ! = =null) {
// Get the current time (performance. Now () or date.now () - React startup time)
const currentTime = getCurrentTime();
// Pause after the 'yieldInterval' ms, no matter where we are in the vsync loop.
// This means that there is always time left before the message event.
deadline = currentTime + yieldInterval;
// Whether there is any time left
const hasTimeRemaining = true;
try {
const hasMoreWork = scheduledHostCallback(
hasTimeRemaining,
currentTime,
);
// Whether there are more tasks to be done
if(! hasMoreWork) {// Set the message loop flag to false if there is no task
// Clear execute the task immediately
isMessageLoopRunning = false;
scheduledHostCallback = null;
} else {
If there is more work to do, schedule the next message event at the end of the previous event.
port.postMessage(null); }}}else {
// Execute task now is empty, close the loop flag
isMessageLoopRunning = false;
}
// Set needsPaint flag to false
needsPaint = false;
};
Copy the code
We learned how to use MessageChannel earlier, and use it in Scheduler to control shard task execution. Let’s look at the basic steps:
- To define a
channel
theMessageChannel
And specify a port variable to point toport2
- will
performWorkUntilDeadline
A message event handler for the port1 port of a channel - in
requestHostCallback
Set the current task to be executed, start the message scheduling cycle, and passport2
thepostMessage
Send a message - call
performWorkUntilDeadline
Method to start the task
Sorting out the variables involved:
field | describe |
---|---|
scheduledHostCallback | Current task to be performed |
isMessageLoopRunning | Indicates whether the current message loop is open. Used to constantly check for new messages (tasks) |
Instead of drawing a picture, draw a picture from someone else:
4. The conclusion
In order to avoid being too long, the React Scheduler Scheduler principle 2 will be summarized next time. I look forward to continuing to attach the previous source code series:
- React source series 1: React API
- React Render FiberRoot
- React source series 3: React.Render process 2 update
- React Fiber
- React Scheduler Scheduler
- React Scheduler….