Redis uses a single thread combined with a non-blocking event polling mechanism to achieve efficient network IO and time event processing. In this article, we will deeply analyze the design and implementation of redis time events from the perspective of source code. Detailed explanation of time events in redisDefinition of time eventsTime events can be single expiration execution and destruction, or they can be scheduled tasks. For this, redis uniformly encapsulates time events as aeTimeEvent objects, uniquely identifies an event through id, and records the seconds and minutes of the task expiration execution in combination with when_sec and when_ms. The function that executes the time event is also executed by the function pointed to by the timeProc pointer. Let's take a redis scheduled task as an example, as shown below. The result records the time before the second and the time in milliseconds through when_sec and when_ms. Once this time is reached, the method serverCron pointed to by the timeProc function pointer will be executed. This function will periodically execute various tasks, which I will expand on later: Correspondingly, we give the code description of the time event, that is, the aeTimeEvent structure in the header file ae.h. This is the encapsulation structure of the time event. It can be seen that in addition to the core fields mentioned above, it also has a next pointer for connecting to the next registered time event: As mentioned above, redis time events are linked in the form of linked lists. Here we also give a unified time management object, namely the time poller aeEventLoop, which records the first time through timeEventHead and manages subsequent time uniformly with the next pointer of the time: Correspondingly, we also give the definition of this period of code, that is, the definition of aeEventLoop in ae.h: Registering time eventsDuring the server initialization phase, redis will register a timed event that is triggered approximately every 1 millisecond. The main purpose of this event is to:
Corresponding to the source code segment of the time event registration we gave, that is, aeCreateTimeEvent in the method initServer called when redis is initialized, you can see that it encapsulates the scheduled task as a time event timeEvent and sets the time interval to once every 1 millisecond: Polling for processing time eventsAfter redis processes all user requests, it will call the time processing function processTimeEvents to poll and process the ready time events, thereby ensuring that the time events are executed as punctually as possible. If the event time is not a scheduled task, it will be deleted directly after execution, otherwise it will set the next execution time. After all these steps are completed, the number of time events processed this time is returned: We give the entry point aeMain for processing the time loop. We can see that this function is the core function of redis. It will call aeProcessEvents in a loop to process various events: For example, aeProcessEvents can show that after executing all user requests, the function calls the processTimeEvents method to obtain and execute the ready time events: Finally, we can see the core code segment for processing time events. It polls the ready time events from timeEventHead, compares whether the current time is greater than or equal to the expiration time, and if so, executes the current time event. Then, it determines whether the event is a timed event. If so, it updates the next execution time, otherwise, it deletes it, and finally accumulates the time of this processing: Redis optimization for time event implementationBecause some time events require regular execution, redis has made the following two optimizations to ensure the real-time execution of time events:
Correspondingly, we give the first point in time for the core code segment of aof rewriting. You can see that serverCron internally determines that if there is no rdb and aof child process at present, and aof rewriting is required, the rewriteAppendOnlyFileBackground function is called to fork the child process for aof rewriting: The handler sendReplyToClient that replies to the client also has a section inside. If the write number totwritten is greater than REDIS_MAX_WRITE_PER_EVENT (the macro is defined as 64M), the write is directly terminated, and the break exit is waited until the next loop processing to avoid starvation of other time events due to this processing, which leads to delayed event execution: |
>>: What problems do HTTP/1, HTTP/2, and HTTP/3 solve?
2019 is the first year of 5G. With the issuance o...
China, which has no say in almost all modern scie...
Yesterday, the General Office of the Ministry of ...
This month, CMIVPS is offering a limited special ...
In a society ruled by law, it is normal to go to ...
[[386134]] Faced with fierce market competition, ...
After more than two years of construction, 5G mes...
iONcloud is a cloud hosting platform opened by Kr...
Another year has passed, winter has gone and spri...
In recent years, 5G and the Internet of Things ha...
With the popularity of microservice architecture ...
Yunbase is a Chinese hosting company. The domain ...
Designing and maintaining a network is complex, b...
[[185474]] RS485 bus is widely used in video surv...