SpringBoot高级知识【原理分析、监控、项目部署】

  • Post author:
  • Post category:其他




1. 原理分析



1.1 自动配置



1.1.1

@Condition

  • Condition是在Spring 4.0增加的条件判断功能,通过这个可以功能可以实现选择性的创建Bean操作。



思考

  1. SpringBoot是如何知道要创建哪个Bean的? 比如

    • SpringBoot是如何知道要创建RedisTemplate的?



案例实操

  • 在Spring 的IOC容器中有一个 User 的 Bean,现要求:

    • 导入Jedis坐标后,加载该Bean,没导入,则不加载。
  1. 前期准备
<dependency>
     <groupId>org.springframework.boot</groupId>
     <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
package com.ithema.springbootconditional.domain;

public class User {
}
  1. 定义User的配置类(用来返回与user相关的Bean)
//定义配置类,用来返回与user相关的Bean
package com.ithema.springbootconditional.config;

import com.ithema.springbootconditional.condition.ClassCondition;
import com.ithema.springbootconditional.domain.User;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Conditional;
import org.springframework.context.annotation.Configuration;

@Configuration
public class UserConfig {

    //用来返回与user相关的Bean
    @Bean
    @Conditional(ClassCondition.class)
    public User user(){
        return  new User();
    }
}

  1. 定义判断类
//定义条件类
package com.ithema.springbootconditional.condition;

import org.springframework.context.annotation.Condition;
import org.springframework.context.annotation.ConditionContext;
import org.springframework.core.type.AnnotatedTypeMetadata;

public class ClassCondition implements Condition {
    @Override
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
        //1.需求: 导入Jedis坐标后创建Bean
        //思路:判断redis.clients.jedis.Jedis.class文件是否存在
        boolean flag = true;
        try {
            Class<?> cls = Class.forName("redis.clients.jedis.Jedis");
        } catch (ClassNotFoundException e) {
            flag = false;
        }
        return flag;
    }
}

  1. 执行引导类
package com.ithema.springbootcondition;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;

@SpringBootApplication
public class SpringbootConditionApplication {

    public static void main(String[] args) {
        //启动SpringBoot的应用,返回Spring的IOC容器
        ConfigurableApplicationContext context = SpringApplication.run(SpringbootConditionApplication.class, args);

        //获取Bean,redisTemplate
        Object redisTemplate = context.getBean("redisTemplate");
        System.out.println(redisTemplate);
    }
}

  • 在Spring 的IOC容器中有一个 Student 的 Bean,现要求:

    • 将类的判断定义为动态的。判断哪个字节码文件存在可以动态指定。


      (高级)

package com.ithema.springbootconditional.domain;

public class Student {
}
//定义配置类,用来返回与user相关的Bean
package com.ithema.springbootconditional.config;

import com.ithema.springbootconditional.condition.ConditionOnClass;
import com.ithema.springbootconditional.domain.Student;
import org.springframework.context.annotation.Bean;

public class StudentConfig {
    @Bean
    @ConditionOnClass("redis.clients.jedis.Jedis")
    public Student student(){
        return  new Student();
    }
}

package com.ithema.springbootconditional.condition;
import java.lang.annotation.*;
import org.springframework.context.annotation.Conditional;

@Target({ElementType.TYPE, ElementType.METHOD})//表示该注解(ConditionOnClass)可以加在哪上面TYPE(类)/Method
@Retention(RetentionPolicy.RUNTIME)//注解生效的实际
@Documented//生成文档
@Conditional(StudentConditon.class)
public @interface ConditionOnClass {
    String[] value();
}

//定义条件类
package com.ithema.springbootconditional.condition;

import org.springframework.context.annotation.Condition;
import org.springframework.context.annotation.ConditionContext;
import org.springframework.core.type.AnnotatedTypeMetadata;

import java.util.Map;

public class StudentConditon implements Condition {
    @Override
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
        //2.需求: 导入通过注解属性值value指定坐标后创建Bean
        //获取注解属性值:value
        String s = ConditionOnClass.class.getName();
        System.out.println(s);
        Map<String, Object> map = metadata.getAnnotationAttributes("s");
        System.out.println(map);  //{value=[redis.clients.jedis.Jedis]}

        String[] value = (String[]) map.get("value");

        boolean flag = true;
        try {
            for (String className : value) {
                Class.forName(className);
            }
        } catch (ClassNotFoundException e) {
            flag = false;
        }
        return flag;
    }
}



小结

  1. 自定义条件:


    1. 定义条件类

      :自定义类

      实现Condition接口

      ,重写matches方法,在matches方法中进行逻辑判断,返回 boolean值。matches方法两个参数:

      • context:上下文对象,可以获取属性值,获取类加载器,获取BeanFactory等。.
      • metadata:元数据对象,用于获取注解属性。

        2.

        判断条件

        :在初始化Bean时,使用@conditional(条件类.class)注解
  2. SpringBoot提供的常用条件注解:

    • @ Conditional:根据条件,决定类是否加载到Spring Ioc容器中,在SpringBoot中有大量的运用
    • @ConditionalonProperty:判断配置文件中是否有对应属性和值才初始化Bean.
    • @ConditionalOnClass:判断环境中是否有对应字节码文件才初始化Bean
    • @ConditionalOnMissingBean:判断环境中没有对应Bean才初始化Bean

  3. @ConditionalonProperty



    案例演示

//使用springboot提供的ConditionalonProperty注解 实现条件判断
@Bean("user2")
@ConditionalOnProperty(name = "itcast",havingValue = "heima")
public User getUser2(){
    return new User();
}
# 在yml或这propreties中配置相关内容
# itoldlu=oldlu
itoldlu: oldlu



1.1.2

切换内置web服务器

  • SpringBoot的web环境中默认使用tomcat作为内置服务器,其实SpringBoot提供了

    4中内置服务器

    供我们选择,我们可以很方便的进行切换。

    • tomcat
    • Jetty
    • Netty
    • Undertow

在这里插入图片描述

<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-web</artifactId>
    <!--排除tomcat依赖-->
    <exclusions>
         <exclusion>
             <artifactId>spring-boot-starter-tomcat</artifactId>
             <groupId>org.springframework.boot</groupId>
         </exclusion>
    </exclusions>
</dependency>

<!--引入jetty的依赖-->
<dependency>
   <artifactId>spring-boot-starter-jetty</artifactId>
   <groupId>org.springframework.boot</groupId>
</dependency>



1.1.3

@Enable*注解



思考

  1. SpringBoot工程是否可以直接获取jar包中定义的Bean?

    • 回答:当然不可以
  2. 那么我们应该怎么获取呢?

    • 请看下面讲解



案例讲解

  1. 前提准备

    • 创建springboot-enable模块
//springboot-enable引导类,我们在这里获取springboot-enable-other的user Bean。
package com.ithema.springbootenableother;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class SpringbootEnableApplication {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(SpringbootEnableApplication.class, args);
        //springboot不能直接获取其他模块的Bean,原因为@SpringBootApplication中的注解@ComponentScan
        Object user = context.getBean("user");
        System.out.println(user);
    }

}
  1. 前提准备2

    • 创建springboot-enable-other模块,该模块主要提供Bean的定义,无实际操作。
//domian包下创建一个User类
package com.ithema.domain;

public class User {
}
//config目录下创建一个User的配置类
package com.ithema.config;

import com.ithema.domain.User;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class UserConfig {
    @Bean
    public User user(){
        return new User();
    }
}
  1. 前期准备3

    • 让springboot-enable模块去依赖与springboot-enbale-other模块
<dependency>
     <groupId>com.ithema</groupId>
     <artifactId>springboot-enable-other</artifactId>
     <version>0.0.1-SNAPSHOT</version>
</dependency>



解决办法


  • 方法1

package com.ithema.springbootenable;

import com.ithema.config.EnableUser;
import com.ithema.config.UserConfig;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;

/**
 * @ComponentScan 扫描范围:当前引导类所在包及其子包
 *
 * com.ithema.springbootenable
 * com.ithema.config
 * 方法 1.解决springboot不能直接获取其他模块的Bean,可以使用@ComponentScan扫描com.ithema.config包
 */
@SpringBootApplication
@ComponentScan("com.ithema.config")  //com.ithema.domain.User@585ac855
public class SpringbootEnableApplication {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(SpringbootEnableApplication.class, args);
   
        Object user = context.getBean("user");
        System.out.println(user);
    }

}


  • 方法2

package com.ithema.springbootenable;

import com.ithema.config.EnableUser;
import com.ithema.config.UserConfig;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;

/**
 * @ComponentScan 扫描范围:当前引导类所在包及其子包
 *
 * com.ithema.springbootenable
 * com.ithema.config
 * 方法 1.解决springboot不能直接获取其他模块的Bean,可以使用@ComponentScan扫描com.ithema.config包
 * 方法 2.可以使用@Import注解,加载类。这些类都会被Spring创建,并放入IOC容器
 */
@SpringBootApplication
@Import(UserConfig.class)  //com.ithema.domain.User@7c2a69b4
public class SpringbootEnableApplication {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(SpringbootEnableApplication.class, args);
        //springboot不能直接获取其他模块的Bean,原因为@SpringBootApplication中的注解@ComponentScan
        Object user = context.getBean("user");
        System.out.println(user);
    }

}

  • 方法3

//提前在springboot-enable-other模块中提供一个@EnableUser供使用
package com.ithema.config;

import org.springframework.context.annotation.Import;

import java.lang.annotation.*;

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Import(UserConfig.class)
public @interface EnableUser {
}
package com.ithema.springbootenable;

import com.ithema.config.EnableUser;
import com.ithema.config.UserConfig;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;

/**
 * @ComponentScan 扫描范围:当前引导类所在包及其子包
 *
 * com.ithema.springbootenable
 * com.ithema.config
 * 方法 1.解决springboot不能直接获取其他模块的Bean,可以使用@ComponentScan扫描com.ithema.config包
 * 方法 2.可以使用@Import注解,加载类。这些类都会被Spring创建,并放入IOC容器
 * 方法 3.可以对Import注解进行封装。
 */
@SpringBootApplication
@EnableUser  //com.ithema.domain.User@3eba57a7
public class SpringbootEnableApplication {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(SpringbootEnableApplication.class, args);
        //springboot不能直接获取其他模块的Bean,原因为@SpringBootApplication中的注解@ComponentScan
        Object user = context.getBean("user");
        System.out.println(user);
    }

}



解析:

方法3就是我们接下来讲的SpringBoot为我们提供的

@Enable*注解


  • SpringBoot中提供了很多Enable开头的注解,这些注解都是用于动态启用某些功能的。而其底层原理是使用@lmport注解导入一些配置类,实现Bean的动态加载。



1.1.4

@lmport注解


@Enable*

底层依赖于@lmport注解导入一些类,使用@lmport导入的类会被Spring

加载到IOC容器

中。而@Import

提供4中用法

  • 导入Bean
  • 导入配置类
  • 导入lmportSelector实现类。一般用于加载配置文件中的类
  • 导入lmportBeanDefinitionRegistrar实现类。



用法 1:导入Bean

//用法1:导入Bean
@SpringBootApplication
@Import(User.class)
public class SpringbootEnableApplication {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(SpringbootEnableApplication.class, args);
        
        User user = context.getBean(User.class);
        System.out.println(user);//com.ithema.domain.User@41477a6d

        Map<String,User> map =  context.getBeansOfType(User.class);
        System.out.println(map);//获取Bean的名称,全路径:{com.ithema.domain.User=com.ithema.domain.User@41477a6d}
    }
}



用法 2:导入配置类

//用法2:导入配置类,它可以同时把多个Bean 加载到IOC容器中
package com.ithema.domain;

public class Role {
}

package com.ithema.config;

import com.ithema.domain.Role;
import com.ithema.domain.User;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

//@Configuration  使用方法2可以不加该注解
public class UserConfig {

    @Bean
    public User user() {
        return new User();
    }

    @Bean
    public Role role() {
        return new Role();
    }
}

//用法2:导入配置类,它可以同时把多个Bean 加载到IOC容器中
@SpringBootApplication
@Import(UserConfig.class)
public class SpringbootEnableApplication {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(SpringbootEnableApplication.class, args);
        
        User user = context.getBean(User.class);
        System.out.println(user);

        Role role = context.getBean(Role.class);
        System.out.println(role);
    }
}



用法 3:导入lmportSelector实现类。一般用于加载配置文件中的类

package com.ithema.config;

import org.springframework.context.annotation.ImportSelector;
import org.springframework.core.type.AnnotationMetadata;

public class MyImportSelector implements ImportSelector {
    @Override
    public String[] selectImports(AnnotationMetadata importingClassMetadata) {
        return new String[]{"com.ithema.domain.User", "com.ithema.domain.Role"};
    }
}

package com.ithema.springbootenable;

import com.ithema.config.EnableUser;
import com.ithema.config.MyImportSelector;
import com.ithema.config.UserConfig;
import com.ithema.domain.Role;
import com.ithema.domain.User;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;

import java.util.Map;

@SpringBootApplication
@Import(MyImportSelector.class)
public class SpringbootEnableApplication {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(SpringbootEnableApplication.class, args);
        
        User user = context.getBean(User.class);
        System.out.println(user);

        Role role = context.getBean(Role.class);
        System.out.println(role);
    }

}



用法4:导入lmportBeanDefinitionRegistrar实现类

//用法4
package com.ithema.config;

import com.ithema.domain.User;
import org.springframework.beans.factory.support.AbstractBeanDefinition;
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.support.BeanDefinitionRegistry;
import org.springframework.context.annotation.ImportBeanDefinitionRegistrar;
import org.springframework.core.type.AnnotationMetadata;

public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {
    @Override
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
        AbstractBeanDefinition beanDefinition = BeanDefinitionBuilder.rootBeanDefinition(User.class).getBeanDefinition();
        registry.registerBeanDefinition("user", beanDefinition);

    }
}
package com.ithema.springbootenable;

import com.ithema.config.EnableUser;
import com.ithema.config.MyImportBeanDefinitionRegistrar;
import com.ithema.config.MyImportSelector;
import com.ithema.config.UserConfig;
import com.ithema.domain.Role;
import com.ithema.domain.User;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Import;

import java.util.Map;

@SpringBootApplication
@Import(MyImportBeanDefinitionRegistrar.class)
public class SpringbootEnableApplication {

    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(SpringbootEnableApplication.class, args);


        User user = context.getBean(User.class);
        System.out.println(user); //com.ithema.domain.User@796d3c9f

        Object user2 = context.getBean("user");
        System.out.println(user2);//com.ithema.domain.User@796d3c9f  为什么对象一样?是单例哦
    }

}



1.1.5

@EnableAutoConfiguration注解

  • @EnableAutoConfiguration注解内部使用@Import (AutoConfigurationImportselector.class)来加载配置类。
  • 配置文件位置:META-INF/spring.factories,该配置文件中定义了大量的配置类,当SpringBoot应用启动时,会自动载这些配置类,初始化Bean
  • 并不是所有的Bean都会被初始化,在配置类中使用Condition来加载满足条件的Bean



1.1.6 小结:

案例实操

  1. 需求:

    • ​ 自定义redis-starter。要求当导入redis坐标时,SpringBoot自动创建Jedis的Bean。
  2. 实现步骤:

    • 创建redis-spring-boot-autoconfigure模块
    • 创建redis-spring-boot-starter模块,依赖redis-spring-boot-autoconfigure的模块
    • 在redis-spring-boot-autoconfigure模块中初始化Jedis 的Bean。并定义META-INF/spring.factories文件
    • 在测试模块中引入自定义的redis-starter依赖,测试获取Jedis 的Bean,操作redis。



1.2 监听机制



1.2.1 Java监听机制

SpringBoot的监听机制,其实是对

Java提供的事件监听机制的封装

Java中的事件监听机制定义了以下几个角色:

  1. 事件: Event,继承java.util.EventObject类的对象
  2. 事件源:Source,任意对象Object
  3. 监听器:Listener,实现java.util.EventListener 接口的对象



1.2.2 SpringBoot监听机制

SpringBoot在项目启动时,会对几个监听器进行回调,我们可以实现这些监听器接口,在项目启动时完成—些操作。

  1. ApplicationContextInitializer、
  2. SpringApplicationRunListener、
  3. CommandLineRunner、
  4. ApplicationRunner



1.3 启动流程分析

在这里插入图片描述



2. 监控



2.1 监控概述

SpringBoot自带

监控功能Actuator

,可以帮助实现对程序内部运行情况监控,比如

  • 监控状况、

  • Bean加载情况


  • 配置属性

  • 日志信息等。

在这里插入图片描述



2.2 监控使用



2.2.1

使用步骤


第一步:

导入依赖坐标

<!--导入依赖坐标-->
<dependency>
	<groupld>org.springframework.boot</groupld>
	<artifactld>spring-boot-starter-actuator</artifactld>
</dependency>


第二步:

访问 http://localhost:8080/actuator



2.2.2

案例实操

可以在

application.properties

配置下面相关信息。若不配置则只能监控

info



health

的内容

# 注意:springboot2.7.2中,info端点默认是不启用,info开头的变量默认也是不启用的
info.name=zhangsan
info.age=23

# springboot2.7.2需要 启用配置里的info开头的变量
management.info.env.enabled=true

# 开启健康检查的完整信息
management.endpoint.health.show-details=always

# 将所有的监控endpoint暴露出来
management.endpoints.web.exposure.include=*



访问

在这里插入图片描述

在这里插入图片描述



2.3

SpringBoot Admin

  • Spring Boot Admin是一个开源社区项目,用于管理和监控SpringBoot应用程序。
  • Spring Boot Admin有两个角色,客户端

    (Client)

    和服务端

    (Server)

  • 应用程序作为

    Spring Boot Admin Client

    向为

    Spring Boot Admin Server

    注册

  • Spring Boot Admin Server



    UI界面



    Spring Boot Admin Client



    Actuator Endpoint

    上的一些监控信息。



2.3.1

操作步骤


1.admin-server:

  • 创建admin-server模块
  • 导入依赖坐标admin-starter-server


  • 引导类

    上启用监控功能@EnableAdminServer


2.admin-client:

  • 创建admin-client模块
  • 导入依赖坐标admin-starter-client

  • 配置相关信息

    : server地址等
  • 启动server和client服务,访问server



2.3.2

案例实操


1. 创建admin-server模块

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述


把服务器端中server.port改成9000,避免启动客户端和服务器两个模块时,端口冲突


在这里插入图片描述


2. 创建admin-client模块

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述


3. 检测

  • 启动服务端
  • 启动客户端
  • 访问

    localhost:9000

    查看结果

在这里插入图片描述


打开应用墙



执行客户端的uerfindAll方法:localhost:8080/user/findAll


在这里插入图片描述


查看相关信息,可以看到findAll被访问了两次


在这里插入图片描述



3. 项目部署

SpringBoot 项目开发完毕后,支持两种方式部署到服务器:

  • jar包(官方推荐)
  • war包



3.1

jar 包


右侧边栏找到Maven,找到要打包的项目。进行package

在这里插入图片描述


通过下面的地址,找到jar包,在目标文件的目录下,按(shlft+右键)执行java -jar命令

在这里插入图片描述

在这里插入图片描述



3.2

war 包

  1. 首先,我们需要对引导类进行一点点的修改

在这里插入图片描述

  1. 然后把打包方式改为

    war

    包。进行打包

在这里插入图片描述

当然,如果觉得名字太长,在打包的同时想给war包指定一个名字也是可以的。例如下图

在这里插入图片描述

  1. 找到

    war

    包,把他放到一个

    tomcat

    服务器的

    webapps

    文件夹下执行。

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述


可以看到项目已经启动,访问项目


  • localhost:8080/springboot/.....

    • 例如:上面的findAll方法:localhost:8080/springboot/user/findAll



版权声明:本文为qq_40926887原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。