Overview The original intention of the project was to independently create a mature and distinctive IOC container, but because the process referred to Spring too much and could not make too many improvements, the purpose became to use this project as a springboard to understand Spring. Unlike some frameworks that imitate Spring on the Internet, this project is mainly aimed at annotations. The address is Thales process In Spring, the formation of a bean is divided into three major stages: Bean definition phase (including BeanDefinition loading, parsing, and registration) Class Design If you just want to understand how a bean is born and dies, you only need to debug it step by step. If you don't understand, debug it several times. But if you want to implement a similar container, you must understand the class design, responsibility allocation, and interface implementation inheritance (unless you want several classes to do everything). Following is the class diagram of DefaultListableBeanFactory Can't stand it? Let’s look at another picture The first picture is for Spring 5.0, and the second picture is for Spring 0.9, so there is no need to introduce too much design complexity at the beginning. Let's look at a set of comparison pictures It is clear at a glance which one is 0.9 and which one is 5.0. The purpose of saying so much is to show that we don’t have to aim for the most perfect goal from the beginning. We can do it step by step and add functions step by step. Implementing simple IOC Let's first define a BeanFactory interface
beanDefinition Because it is in the form of annotations, we can no longer give a resource file and then parse it like XML, but should scan all classes with @Component under the classPath. At this time, the parameter we need to give changes from the file path to the package path. We only need to scan the qualified classes in this package and its subpackages, convert them into BeanDefinition and then register them. The class that performs this function is ClassPathBeanDefinitionScanner. At this step, it is different from the traditional process. We will pass in a ResourceLoader to implement the specific scanning function (ie positioning), but there will be no special class to handle the parsing step.
But the PathMatchingResourcePatternResolver is not used directly in the end Instead, use it as a property of ClassPathBeanDefinitionScanner and call it in this class. Now that we have a Resource, how do we get the corresponding BeanDefinition? First consider this question, what kind of class can be registered BeanDefinition? Added @Component annotation or met other registration conditions Not an interface or abstract class So we can abstract a method boolean isCandidateComponent(Class<?> clazz) to determine whether it is registered Now it is time to register. We still adhere to the concept of interface-oriented programming and take single responsibility into consideration. We abstract the registration bean definition separately.
As mentioned above, the location, parsing, and registration of Bean definitions are all done in ClassPathBeanDefinitionScanner, so BeanDefinitionRegistry naturally becomes one of the attributes of ClassPathBeanDefinitionScanner. So the ClassPathBeanDefinitionScanner is constructed
Instantiation When is it instantiated? We say that it is instantiated when getBean() is called and there is no existing bean.
Object creation There are two ways, through the default reflection implementation of Jdk, or through the cglib proxy implementation. The default is naturally a no-parameter constructor, but if parameters are passed in, you need to match the corresponding constructor according to the type and number of parameters and use it to instantiate So we abstract InstantiationStrategy as the instantiation interface. Both instantiation methods need to implement this interface. When we actually use it, we only need to call the method of this interface.
Property Injection Get field value There are two ways to get field values: Directly annotate Autowired or Value
Field value filling After getting the field value, fill it into the corresponding field through reflection
initialization Call the specified initialization method to initialize the resources. How to get the initialization method? In XML mode, just add a tag. If it is annotation mode, add an annotation or add a parameter to an annotation to represent the initialization method. This has not been implemented yet. Function filling Postprocessor added Above we have implemented a Bean container that can perform dependency lookup and dependency injection. Let's review the Spring process and see what we are missing. The easiest thing to think of is the post-processor, including BeanFactoryPostProcessor and BeanPostProcessor. The former modifies the beanFactory, while the latter modifies the bean. It is also interface-oriented programming. First, create a BeanPostProcessor
At present, what needs to be done by BeanPostProcessor? We can separate the code that processes annotations and obtains injection properties and use a BeanPostProcessor to process it. All custom implemented BeanPostProcessors need to inherit this interface. Since the function of BeanPostProcessor is to process other beans, it must be created before other processed beans are instantiated. So we add registerBeanPostProcessors(beanFactory) before finishBeanFactoryInitialization(beanFactory); to instantiate all BeanPostProcessors The importance of these beanPostProcessors is different. For example, the priority of the BeanPostProcessor that handles annotation injection is higher than that of the general BeanPostProcessor, so it needs to be instantiated first. Aware interface added In fact, now we can completely hand over an object to the IOC container, but the relationship between the object and the container is one-way. The container can operate the bean, but the bean cannot use the container. In order to solve this problem, we add an Aware interface as a marker interface, which is inherited by each more specific Aware. After the attributes are instantiated, the initialization method is executed to complete the injection of related container attributes. Event listener added Listener is an implementation of the observer pattern Let's first define the following basic interfaces
The specific calling process is that a specific listener is added to the broadcaster, and the event is published uniformly through the publisher. The publishEvent will finally call the multicastEvent (ApplicationEvent event) method, and the corresponding listener will make corresponding operations after the corresponding judgment. How to determine whether this listener is interested in this event? The listener we implemented beforehand is generic, and we can judge by the relationship between this generic and the incoming event type.
FactoryBean added Currently, all beans are generated by BeanFactory. We use the FactoryBean interface to identify this special Bean that produces Beans. Solving circular dependencies Circular dependency means that A depends on B while B depends on A. The solution is to separate instantiation and initialization. If we only consider the general case, two-level cache is actually enough. Code Optimization Implementing simple AOP If you start with the orthodox AOP, a bunch of concepts will follow, including pointcuts and notifications. Let's first look at what AOP does So the core of AOP is dynamic proxy. Let's take Cglib as an example to see how to use dynamic proxy.
This is the core function of dynamic proxy, and also the core function of AOP. The ultimate goal of AOP is code 5, that is, to generate a proxy object and hand it over to IOC for management. In order to achieve this goal, the AOP framework needs to do what code 1-4 needs to do. Code 1 and 2 are combined to form JoinPoint, 3 is called Advice, and these two are combined to form Advisor. Can we write them all in one or several classes without distinguishing them? Of course, Spring 0.9 does this, but now, this division method has been adopted. This project also adopts this classification. Let's start with the connection point. How to determine where to implement functional enhancement is nothing more than two levels: class and method; Let's first define the two interfaces ClassFilter and MethodMacther
These two interfaces must be used in combination, so we use PointCut to combine them
The interface only defines abstract functions, which must be implemented in detail. By default, we use Java regular expressions to match method names, thereby constructing JdkRegexMethodMatcher
In Spring, MethodMatcher is not directly inherited. Considering the different syntax of regular expressions, an additional layer of abstraction is made, but it is omitted here. JdkRegexMethodMatcher also implements the PointCut class, which means that the cut point is now ready. Let’s look at Advice Since there are many extensibility points to consider, the inheritance levels have also increased.
Now that the Advice is defined, we leave the specific implementation to the user. The next step is to integrate it into Advisor
The Advisor functionality has been defined. Let's implement this interface
RegexMethodPointcutAdvisor integrates PointCut and Advice. Through it, we can determine where to make enhancements. The current advisor can complete the function of checking whether a class needs to be proxied, but if this class needs to be proxied, the advisor cannot save the corresponding information of this class So we need a class to combine the advisor with the corresponding proxy class, which is AdvisedSupport
The TargetSource in the above class attributes is the class that actually holds the proxy object information. Now that everything is ready, we just need to use Cglib to create new classes using the information we already have.
Comparing this code with the original code using cglib, you will find that the process is almost identical. However, as a framework, it should be as convenient as possible for users. So we need a Creator to do all this. It is responsible for combining Advice and PointCut into Advisor, assembling Advisor and TargetSource into AdvisedSupport, and then handing AdvisedSupport to Cglib dynamic proxy to generate proxy objects. Users only need to write Advice and pointcut expressions. Functional Demonstration Property injection Basic types Reference types Circular dependencies Container awareness Difficulties and solutions First of all, the problem is design |
<<: Pull or Push? How to choose a monitoring system?
As Matter’s foundational technology, Wi-Fi can he...
China's broadband speed used to be disappoint...
“While the discussion and hype around 5G has focu...
[[420883]] This article is reprinted from the WeC...
With the advancement of the "Healthy China 2...
RAKsmart is carrying out the "New Year's...
RackNerd has released a special package for the 6...
In the blink of an eye, we have entered 2018. In ...
I have talked about service mesh, API gateway and...
On the morning of December 8, at the 2016 GNTC Gl...
[51CTO.com original article] Recently, the 2017 H...
On May 13, according to IDC's Global Semi-ann...
5G messaging, which is seen by the industry as We...
1. OSI Reference Model 1. Origin of OSI OSI (Open...
Korea Telecom is using GIGA Wire 2.0 technology t...