This article is reprinted from the WeChat public account "Source Code Apprentice", the author is Huangfu Aoaojiao. Please contact the Source Code Apprentice public account to reprint this article. NioEventLoopGroup initialization source code1. The process of finding source codeAs we mentioned earlier, we can almost regard NioEventLoopGroup as a thread pool, which will execute tasks one by one. There are two kinds of NioEventLoopGroup we commonly use, NioEventLoopGroup(int nThreads), NioEventLoopGroup(), one is to specify the number of threads, and the other is to specify the number of threads by default! Here we analyze it with the parameterless construction as the entry point!
When we use the default amount, it will pass a 0, so let's continue!
Note that the parameters passed here are: 0, null
Here, an additional SelectorProvider.provider() is passed. This method is an API provided by JDK NIO and can mainly obtain NIO selectors or Channels, as shown in the following figure: We return to the main thread and continue with:
Here, a DefaultSelectStrategy selection strategy is passed, which will be explained in detail later when explaining NioEventLoop, so I won't elaborate on it!
We will find that a rejection strategy RejectedExecutionHandlers.reject() is passed by default. What is this rejection strategy for?
We have come to a conclusion that when certain conditions trigger this rejection strategy, it will throw a RejectedExecutionException. The specific time of triggering will be explained in detail later. Just remember it here! Let's go back to the main line. Here we start calling the parent class. Do you remember who is the parent class of NioEventLoopGroup that we analyzed in the last class? That's right: MultithreadEventLoopGroup. We will enter MultithreadEventLoopGroup:
Do you remember what nThreads is? Is it 0? Here is a judgment. When the number of your threads is 0, DEFAULT_EVENT_LOOP_THREADS will be used as the number of thread pools. What is DEFAULT_EVENT_LOOP_THREADS?
The default is twice the CPU, so we now come to a conclusion that when we use the default NioEventLoopGroup, the system will use the system CPU core number * 2 as the number of thread pools by default! The selectorProvider, rejection strategy, and selectStrategyFactory we passed in the previous step are encapsulated as an array and placed at args[0], args[1], and args[2]! Let's go back to the main thread and call the parent class again, MultithreadEventExecutorGroup:
Note that an extra parameter is passed here again: DefaultEventExecutorChooserFactory, a selector factory, which returns a selector of type DefaultEventExecutorChooserFactory. The specific analysis will be analyzed later! Let's go back to the main line:
Here we finally see a large section of code. Here is the main logic of EventLoopGroup. Let's analyze it line by line: 2. Building a thread executor1. Source code analysis
Here we will determine whether the executor we passed in is empty, otherwise we will create a new one. Do we remember what the value of executor is? It is null, right? So it will enter the logic here. Let's go into the newDefaultThreadFactory source code and take a look: newDefaultThreadFactory() main logic
As you can see, a DefaultThreadFactory, a default thread factory, is passed into the executor! ThreadPerTaskExecutor main logic:
We found that a thread factory we passed in was called here, a new thread was created and the start method was called to start it. So how was it created? We go into the newThread source code to check. Since the thread factory we use by default is DefaultThreadFactory, we will go to DefaultThreadFactory#newThread
There are not many operations here, just encapsulating a Runnable as a Thread and returning it. Let's focus on this Thread. Is it the same as the Thread we traditionally use? Let's follow up with the newThread method and take a look:
The logic is very simple, that is, wrap a Thread as Netty's customized FastThreadLocalThread. As for why, we will not explain it further for now, and the subsequent chapters will explain it in detail! 2. Thread Executor Summary Here we will create a thread executor ThreadPerTaskExecutor, using the default thread factory DefaultThreadFactory. The thread executor will wrap a task as a FastThreadLocalThread object, and then call the start method to start a new thread to execute the task! 3. Create the corresponding number of executors
1. Source code analysis
First he will create an empty EventExecutor executor array, and then iterate and fill it! Remember what nThreads is? The default is CPU*2, so CPU*2 executors will be created here! We found that the main logic filled in the for loop is newChild, so we enter the newChild method. Here is a hint, what kind of object is the Group object we created? It is a NioEventLoopGroup object, right? So here we will enter the NioEventLoopGroup#newChild method:
The length of the args we passed is 3, which has been parsed before: args[0] is selectorProvider, args[1] is rejection strategy, args[2] is selectStrategyFactory So queueFactory is null. Then we focus on the NioEventLoop object. It can be seen that the newChild method returns NioEventLoop, so we can preliminarily determine that the NioEventLoop object exists in the EventExecutor array! So far, we will not go into it. I will analyze the initialization source code of NioEventLoop in the next lesson. Here we can be sure of one thing. The NioEventLoop object exists in the EventExecutor array! Let's go back to the main line: 2. Executor Array Summary After the for loop is completed, the EventExecutor[nThreads] array is filled, and each element in it is a NioEventLoop object, and each NioEventLoop object contains a ThreadPerTaskExecutor thread executor object! 4. Create an executor selector1. Source code analysis
Do you remember what type chooserFactory is? It is of type DefaultEventExecutorChooserFactory. If you forget, you can look up the code in the source code or debug it! We enter the DefaultEventExecutorChooserFactory#newChooser source code logic and pass in the array we just looped and filled:
As you can see, there seem to be two cases here, and different strategy objects are returned. When the length of your array is an idempotent power of 2, the PowerOfTwoEventExecutorChooser object is returned, otherwise the GenericEventExecutorChooser object is returned. Let's analyze both cases: I. PowerOfTwoEventExecutorChooser
The main logic of this code is to take a self-increasing CAS class and perform an & operation with the array length, which will eventually result in a loop to obtain the number: As can be seen from the above picture, this function can realize a cyclic function of fetching numbers. Every time the tail of the array is reached, it will go back to the head and fetch again! Code example:
Result Set
II. GenericEventExecutorChooser When the number of your threads is not a power of 2, a general selector will be used. The specific implementation source code is as follows:
I don't need to demonstrate this code. Its function is the same as the above one. It can achieve the function of cyclically obtaining numbers. think Why does Netty have to be implemented in two strategy classes? Can't we just use the second one? Netty's official requirements for performance have reached the extreme. Everyone should know that the speed of bit operations is higher than that of direct modulus operations, so Netty has made an optimization even for this point! 2. Actuator Selector Summary From the above, we can understand that a selector will be created through a selector factory and saved in NioEvenetLoopGroup. Calling the next method of the selector will return a NioEventLoop object, which is obtained by continuous looping and obtaining NioEventLoop objects in turn. This is also the basis for NioEventLoop to SocketChannel to be one-to-many! This is all later! NioEventLoopGroup source code summary
|
A few days ago, we shared the news that RAKsmart&...
Domestic 5G construction is still in full swing, ...
【51CTO.com Quick Translation】 Undoubtedly, mobile...
In the Internet world, HTTP and HTTPS are the two...
The Internet of Things (IoT) is widely regarded b...
The intelligentization of weak-current electricit...
On April 9, 2019, Juniper Networks, a provider of...
Bandwagonhost has just restocked a limited editio...
Research has found that when page access performa...
[[419187]] As Cisco's global strategic partne...
Huawei has always been an active promoter and pra...
HUAWEI CONNECT 2017 opened on September 5 at the ...
What is structured cabling? In short, it is a sta...
As high-speed cellular networks become mainstream...
CloudCone has released a new promotional package,...