首頁 > Java > java教程 > SpringBoot的底層原理是什麼

SpringBoot的底層原理是什麼

王林
發布: 2023-05-11 12:40:06
轉載
855 人瀏覽過

手寫springboot

在日常開發中只需要引入下面的依賴就可以開發Servlet進行存取了。

<dependency>
	<groupId>org.springframework.boot</groupId>
	<artifactId>spring-boot-starter-web</artifactId>
</dependency>
登入後複製

那這是怎麼做到的呢?今天就來一探究竟

先新建一個maven專案rick-spring-boot,並建立兩個子專案分別是spring-boot和user,其中spring-boot專案就是模擬手寫一個簡單springboot,user就是用來測試手寫的spring-boot的。

SpringBoot的底層原理是什麼

user專案-測試工程

user專案包含pom.xml、UserController和UserService

<dependencies>
    <dependency>
      <groupId>com.rick.spring.boot</groupId>
      <artifactId>spring-boot</artifactId>
      <version>1.0-SNAPSHOT</version>
    </dependency>
  </dependencies>
登入後複製
@RestController
public class UserController {
    @Autowired
    private UserService userService;
    @GetMapping("/user")
    public String getUser() {
        return userService.getUser();
    }
}
@Service
public class UserService {
    public String getUser() {
        return "rick";
    }
}
登入後複製

以及user專案的啟動類別RickApplication ,而RickSpringApplication.run()是需要手寫的啟動類別以及@RickSpringBootApplication註解,都是需要在spring-boot專案實現。

import com.rick.spring.boot.RickSpringApplication;
import com.rick.spring.boot.RickSpringBootApplication;
@RickSpringBootApplication
public class RickApplication {
    public static void main(String[] args) {
        RickSpringApplication.run(RickApplication.class);
    }
}
登入後複製

Springboot專案

首先來看RickSpringApplication.run(RickApplication.class)方法需要做的事情:

(1)建立spring容器,並將傳入的class註冊到spring容器中

(2)啟動web服務,如tomcat,用來處理請求,並透過DispatchServlet將請求分發到Servlet進行處理。

public class RickSpringApplication {
    public static void run(Class clz) {
        AnnotationConfigWebApplicationContext context = new AnnotationConfigWebApplicationContext();
        context.register(clz);
        context.refresh();
        start(context);
    }
    public static void start(WebApplicationContext applicationContext) {
        System.out.println("start tomcat");
        Tomcat tomcat = new Tomcat();
        Server server = tomcat.getServer();
        Service service = server.findService("Tomcat");
        Connector connector = new Connector();
        connector.setPort(8081);
        Engine engine = new StandardEngine();
        engine.setDefaultHost("localhost");
        Host host = new StandardHost();
        host.setName("localhost");
        String contextPath = "";
        Context context = new StandardContext();
        context.setPath(contextPath);
        context.addLifecycleListener(new Tomcat.FixContextListener());
        host.addChild(context);
        engine.addChild(host);
        service.setContainer(engine);
        service.addConnector(connector);
        tomcat.addServlet(contextPath, "dispatcher", new DispatcherServlet(applicationContext));
        context.addServletMappingDecoded("/*", "dispatcher");
        try {
            tomcat.start();
        } catch (LifecycleException e) {
            e.printStackTrace();
        }
    }
}
登入後複製

RickApplication是被@RickSpringBootApplication註解修飾的,從以下程式碼可以看出RickApplication是配置類,在被註冊到spring容器後,spring就會解析這個類別。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Configuration
@ComponentScan
public @interface RickSpringBootApplication {
}
登入後複製

啟動user專案RickApplication的main方法,

SpringBoot的底層原理是什麼

存取UserController

SpringBoot的底層原理是什麼

##至此簡單的spring-boot專案就整合完成了。

自動設定

實作tomcat和jetty的切換

在使用springboot時,如果我們不想使用tomcat作為請求處理服務,而是jetty或其他的web服務,通常只需要將相關的tomcat依賴進行排除,然後引入jetty的依賴就可以了,這就是springboot的自動組裝的機制。接下來看看是如何實現的

定義一個WebServer介面和兩個實作類別(tomcat和jetty),並寫好啟動tomcat和jetty服務的程式碼

public interface WebServer {
    void start();
}
public class JettyServer implements WebServer{
    @Override
    public void start() {
        System.out.println("start jetty");
    }
}
登入後複製
public class TomcatServer implements WebServer, ApplicationContextAware {
    private WebApplicationContext applicationContext;
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = (WebApplicationContext) applicationContext;
    }
    @Override
    public void start() {
        System.out.println("start tomcat");
        ...
        tomcat.addServlet(contextPath, "dispatcher", new DispatcherServlet(applicationContext));
        context.addServletMappingDecoded("/*", "dispatcher");
        try {
            tomcat.start();
        } catch (LifecycleException e) {
            e.printStackTrace();
        }
    }
}
登入後複製

定義AutoConfiguration接口,用來標識需要自動裝配的類別。再定義一個WebServerAutoConfiguration類,它被表示為spring的一個配置類,最終我們需要導入這個類別由spring來解析它,隨後spring會解析@Bean註解的方法來載入Bean。注意這裡下面兩個方法也定義了@RickConditionalOnClass註解來決定是否需要解析這個bean,如果滿足條件則進行解析,即應用內存在Tomcat或Server的Class,會解析對應方法的Bean,

public interface AutoConfiguration {
}
@Configuration
public class WebServerAutoConfiguration implements AutoConfiguration {
    @Bean
    @RickConditionalOnClass("org.apache.catalina.startup.Tomcat")
    public TomcatServer tomcatServer() {
        return new TomcatServer();
    }
    @Bean
    @RickConditionalOnClass("org.eclipse.jetty.server.Server")
    public JettyServer jettyWebServer() {
        return new JettyServer();
    }
}
登入後複製

來看@RickConditionalOnClass註解:當spring解析被@RickConditionalOnClass註解的方法時,spring就知道它被@Conditional修飾,並會在解析時執行RickOnClassConditional的match()方法,來判斷是否滿足載入bean的條件。 match()會嘗試載入傳入的類別路徑名,如果應用程式內引入相關的jar則會載入成功回傳true,反之,回傳false。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
@Conditional(RickOnClassConditional.class)
public @interface RickConditionalOnClass {
    String value();
}
public class RickOnClassConditional implements Condition {
    @Override
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
        Map<String, Object> annotation = metadata.getAnnotationAttributes(RickConditionalOnClass.class.getName());
        try {
            context.getClassLoader().loadClass((String) annotation.get("value"));
        } catch (ClassNotFoundException e) {
            return false;
        }
        return true;
    }
}
登入後複製

引入WebServerAutoConfiguration,最簡單粗暴的方式就是透過@Import(WebServerAutoConfiguration.class)匯入該類別。但是spring-boot不可能這麼做,成千上萬的自動配置寫在程式碼裡肯定不好。 spring透過SPI機制,在resources目錄下建立如下目錄和檔案

SpringBoot的底層原理是什麼

#定義一個類別實作DeferredImportSelector接口,並實作selectImports(),透過JDK的ServiceLoader載入以上文件中的類。透過@Import(WebServerImportSelector.class)註解導入該類別spring在解析配置類別的時候就會執行selectImports(),從而將WebServerAutoConfiguration導入到spring容器中,spring就會解析這個配置類別。

public class WebServerImportSelector implements DeferredImportSelector {
    @Override
    public String[] selectImports(AnnotationMetadata metadata) {
        ServiceLoader<AutoConfiguration> load = ServiceLoader.load(AutoConfiguration.class);
        List<String> list = new ArrayList<>();
        for (AutoConfiguration loader : load) {
            list.add(loader.getClass().getName());
        }
        return list.toArray(new String[list.size()]);
    }
}
登入後複製

至此,springboot就做到了只需要修改user工程的maven依賴就能切換tomcat和jetty服務了

<dependency>
      <groupId>com.rick.spring.boot</groupId>
      <artifactId>spring-boot</artifactId>
      <version>1.0-SNAPSHOT</version>
      <exclusions>
        <exclusion>
          <groupId>org.apache.tomcat.embed</groupId>
          <artifactId>tomcat-embed-core</artifactId>
        </exclusion>
      </exclusions>
    </dependency>
    <dependency>
      <groupId>org.eclipse.jetty</groupId>
      <artifactId>jetty-server</artifactId>
      <version>9.4.43.v20210629</version>
    </dependency>
登入後複製
重啟user專案

SpringBoot的底層原理是什麼 #

以上是SpringBoot的底層原理是什麼的詳細內容。更多資訊請關注PHP中文網其他相關文章!

相關標籤:
來源:yisu.com
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
最新問題
java - springboot新手學習
來自於 1970-01-01 08:00:00
0
0
0
java - 新人用直接springmvc還是springboot
來自於 1970-01-01 08:00:00
0
0
0
java - springboot部署到tomcat url請求路徑問題
來自於 1970-01-01 08:00:00
0
0
0
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板