The main character we’re going to talk about today is the big dog in design patterns, and it’s called the template pattern. I believe that some of you have seen this design pattern before but do not understand why it is so important, and I believe that this will make you learn something new.

The template pattern, as it is called in many blogs, defines the skeleton of an algorithm in an operation, and by deferring some steps to subclasses, the template method can redefine specific steps of an algorithm without changing the structure of that algorithm. In our real programming, we usually use an abstract class to expose the methods/templates that define the methods that execute it. Its subclasses can override method implementations as needed, but the calls will be made in the manner defined in the abstract class.

Let’s start with a very simple example. Let’s take cooking in life as an example. It is generally divided into three steps: shopping, washing vegetables, and finally cooking.

public abstract class Cook {

    protected abstract List<Food> buy();

    protected void wash(List<Food> vegetables){
        vegetables.forEach(Food::washed);
    }

    protected abstract List<CookedFood> cook(List<Food> vegetables);

    // Template method
    public final void cook(){
        // Descendant can override the buy method, i.e. you can buy whatever you want
        List<Food> vegetables = buy();

        // Wash gives the default implementation, that is, it is generally used to wash the vegetables, of course you can rewrite it later
        wash(vegetables);

        // The final step is the cooking processList<CookedFood> cookedFoods = cook(vegetables); }}Copy the code

Then if we wanted to eat fish, we could write:

public class CookFish extends Cook{
    @Override
    protected List<Food> buy() {
        List<Food> result = new ArrayList<>();
        // Sorry, I'm a big eater. I need a shark
        result.add(new Shark());
        result.add(new Fish());
        return result;
    }

    @Override
    protected void wash(List<Food> foods){
        System.out.println("No, it tastes better when you scoop it out.");
    }

    @Override
    protected List<CookedFood> cook(List<Food> vegetables) {
        return vegetables.stream().map(CookedFood::new) .collect(Collectors.toList()); }}Copy the code

Food is the parent of Fish and Shark. Food has a washed method, and other classes are empty. One of the CookedFood member attributes is Food. If we wanted to Cook something else like lobster, we would define a descendant of Cook, then subclass Food ourselves, and rewrite the buying, washing, and cooking methods to suit our own needs.

Think it’s easy? It’s like you can master it in a few minutes. Yes, it’s that simple 😂, but the problem is that you have to play 6 to really know how to do it, not just the 5 minute demo above, which really doesn’t work. Let’s take a look at how it works in Spring.

We are of the third article have mentioned AbstractApplicationContext refresh method, a lot of people read the spring source of from here all the time, this method is also a template method, let’s look at the code:

        @Override
	public void refresh() throws BeansException, IllegalStateException {
		synchronized (this.startupShutdownMonitor) {
			// 1. Preparation before loading
			prepareRefresh();
			// 2. Get a new beanFactory instance
			ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();

			// 3. Initialize beanFactory and set various Settings for it
			prepareBeanFactory(beanFactory);

			try {
				/ / 4. Allowing for the beanFactory do some stuff in the front of the change, can be add a BeanFactoryPostProcessors this, also can give the increase
				// BeanDefinition can also be added with interfaces that the beanFactory will ignore during autowerization, and beans that can be used in specific scenarios.
				For example, some descendants add new scope beans and so on. But the point is, all of this is based on a concrete scenario, so in this abstract class,
				// This method is empty (not abstract because it doesn't force subclasses to implement it)
				postProcessBeanFactory(beanFactory);

				/ / 5. Trigger call all BeanFactoryPostProcessors a what behind (speak)
				invokeBeanFactoryPostProcessors(beanFactory);

				// register all beanPostProcessors (more on this later)
				registerBeanPostProcessors(beanFactory);

				// 7. Initialize objects that support internationalization
				initMessageSource();

				// 8. Initialize the event broadcaster
				initApplicationEventMulticaster();

				// 9. Initialize other special beans
				onRefresh();

				// 10. Register listeners
				registerListeners();

				// 11. Instantiate bean (BOSS difficulty code)
				finishBeanFactoryInitialization(beanFactory);

				// Last step: publish corresponding event.
				finishRefresh();
			}

			catch (BeansException ex) {
				logger.warn("Exception encountered during context initialization - cancelling refresh attempt", ex);

				// Destroy already created singletons to avoid dangling resources.
				destroyBeans();

				// Reset 'active' flag.
				cancelRefresh(ex);

				// Propagate exception to caller.
				throwex; }}}Copy the code

I’m only going to talk about step 2 here, because the first three articles are really just about 1 and 2. Let’s look at the code for step 2:

The reason for releasing all the code is just to let everyone mix a face familiar, don't worry, put out of my back will say 😊Copy the code

In fact, there are only two lines of code in the second place, and we focus on the method refreshBeanFactory in the first line

        /**
	 * Tell the subclass to refresh the internal bean factory.
	 * @return the fresh BeanFactory instance
	 * @see #refreshBeanFactory()
	 * @see #getBeanFactory()
	 */
	protected ConfigurableListableBeanFactory obtainFreshBeanFactory() {
		refreshBeanFactory();
		ConfigurableListableBeanFactory beanFactory = getBeanFactory();
		if (logger.isDebugEnabled()) {
			logger.debug("Bean factory for " + getDisplayName() + ":" + beanFactory);
		}
		return beanFactory;
	}
Copy the code

The subclass must implement this method in order to do the actual configuration loading, and this method must be triggered by Refresh before any other initialization work. Before we look at the implementation of this method is used in our XML configuration in this case the corresponding AbstrctApplicationContext subclasses (namely ClassPathXmlApplicationContext abstract superclass AbstractXmlApplicati OnContext), and to make things clear, I’ll take you through the previous code:

	@Override
	protected final void refreshBeanFactory() throws BeansException {
		if (hasBeanFactory()) {
			destroyBeans();
			closeBeanFactory();
		}
		try {
			DefaultListableBeanFactory beanFactory = createBeanFactory();
			beanFactory.setSerializationId(getId());
			customizeBeanFactory(beanFactory);
			// Load the BeanDefinition
			loadBeanDefinitions(beanFactory);
			synchronized (this.beanFactoryMonitor) {
				this.beanFactory = beanFactory; }}catch (IOException ex) {
			throw new ApplicationContextException("I/O error parsing bean definition source for "+ getDisplayName(), ex); }}Copy the code

This is the method of above refreBeanFactory AbstractRefreshAbleApplicationContext implementation, which marked comments inside the code loadBeanDefinitons is an abstract method, and the abstract methods of several implementation, A is the implementation of AbstractXmlApplicationContext we mentioned earlier, it is in this method to complete the loading of XML parsing, and the other is an implementation of AnnotationConfigWebApplicationContext for its implementation, In this implementation, we complete the process of loading BeanDefiniton into the class that marked our common annotations.

And then return to the refresh method in AbstractApplicationContext refreshBeanFactory this abstract method, it is not difficult to find that, different subclass implements this method according to the different scene, We then get the BeanDefinition based on the different configuration, and we’re ready to instantiate the bean. There are other methods in the Refresh method that have similar capabilities, but I won’t go through them here, otherwise it will be a long article.

We think about the article that a very simple demo in the beginning, indeed, that little demo anyone can easy to understand, but we think about it and want to use this template pattern premise is you must be well for your business process are very understanding, a process which is small the process of the abstract, which requires the default implementation, And what is need to empty implementation, these are all doing template method (business process is actually in the abstract you) need to consider the problems when (template pattern is more about this process), if at the beginning of the design is the design doing well, template pattern brings the infinite extension of benefits, but if not good, So much so that after using the template method for a while, it can be very painful to change.

The last sentence is also a voice-over, my own experience is: this model is very good combination of process and object oriented, abstract business is playing with process, use subclasses to implement is playing with object oriented, if the description is not appropriate, you should take a look.Copy the code