SpringBoot源码-Spring Boot启动时控制台为何会打印logo以及自定义banner.txt文件控制台打印

1.当我们启动一个SpringBoot项目的时候,入口程序就是main方法,而在main方法中就执行了一个run方法。

java 复制代码
@SpringBootApplication
public class StartApp {

	public static void main(String[] args) {
		//  test
		SpringApplication.run(StartApp.class);
	}
}
java 复制代码
	public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) {
		// 创建了一个SpringApplication对象,并调用其run方法
		// 1.先看下构造方法中的逻辑
		// 2.然后再看run方法的逻辑
		return new SpringApplication(primarySources).run(args);
	}

2.SpringApplication.run方法

java 复制代码
public static ConfigurableApplicationContext run(Class<?> primarySource, String... args) {
		// 调用重载的run方法,将传递的Class对象封装为了一个数组
		return run(new Class<?>[] { primarySource }, args);
	}

3.输出自定义banner.txt的方法就是:

java 复制代码
	// 输出的Banner信息
			Banner printedBanner = printBanner(environment);

在当前项目的资源目录下,定义了一个banner.txt内容如下,接下来讲解这个txt的内容是如何输出到控制台的。

进入printBanner这个方法:

Banner.Mode有三个枚举值:

java 复制代码
	enum Mode {

		/**
		 * 禁用打印横幅
		 */
		OFF,

		/**
		 * 打印横幅到System.out。
		 */
		CONSOLE,

		/**
		 * 将横幅打印到日志文件中。
		 */
		LOG

	}

this.bannerMode 是默认的

Banner.Mode bannerMode = Banner.Mode.CONSOLE;这里会输出到控制台

java 复制代码
	private Banner printBanner(ConfigurableEnvironment environment) {
		if (this.bannerMode == Banner.Mode.OFF) {
			return null;
		}
		ResourceLoader resourceLoader = (this.resourceLoader != null) ? this.resourceLoader
				: new DefaultResourceLoader(getClassLoader());
		SpringApplicationBannerPrinter bannerPrinter = new SpringApplicationBannerPrinter(resourceLoader, this.banner);
		if (this.bannerMode == Mode.LOG) {
			return bannerPrinter.print(environment, this.mainApplicationClass, logger);
		}
		return bannerPrinter.print(environment, this.mainApplicationClass, System.out);
	}

这里会进入最后一个return bannerPrinter.print(environment, this.mainApplicationClass, System.out);

java 复制代码
	Banner print(Environment environment, Class<?> sourceClass, PrintStream out) {
		Banner banner = getBanner(environment);
		banner.printBanner(environment, sourceClass, out);
		return new PrintedBanner(banner, sourceClass);
	}
复制代码
getBanner方法会选择图片或者是文字打印输出内容,这里是txt直接进入getTextBanner
java 复制代码
	private Banner getBanner(Environment environment) {
		Banners banners = new Banners();
		banners.addIfNotNull(getImageBanner(environment));
		banners.addIfNotNull(getTextBanner(environment));
		if (banners.hasAtLeastOneBanner()) {
			return banners;
		}
		if (this.fallbackBanner != null) {
			return this.fallbackBanner;
		}
		return DEFAULT_BANNER;
	}

getTextBanner实现细节如下:

这里没有配置BANNER_LOCATION_PROPERTY 所以location 会是banner.txt。

java 复制代码
static final String BANNER_LOCATION_PROPERTY = "spring.banner.location";
static final String DEFAULT_BANNER_LOCATION = "banner.txt";

	private Banner getTextBanner(Environment environment) {
		String location = environment.getProperty(BANNER_LOCATION_PROPERTY, DEFAULT_BANNER_LOCATION);
		Resource resource = this.resourceLoader.getResource(location);
		if (resource.exists()) {
			return new ResourceBanner(resource);
		}
		return null;
	}

getBanner有一个DEFAULT_BANNER,这个默认的内容如下:

java 复制代码
	private static final Banner DEFAULT_BANNER = new SpringBootBanner();

SpringBootBanner内容如下:

这个输出就是springBoot默认启动输出的logo,因为这里自定义了txt所以不会输出这个。

得到banner之后开始打印到控制台的逻辑了

banner.printBanner(environment, sourceClass, out);实现如下:

继续printBanner:

这里,自定义的banner就完成了打印功能。

相关推荐
devlei1 天前
从源码泄露看AI Agent未来:深度对比Claude Code原生实现与OpenClaw开源方案
android·前端·后端
pshdhx_albert1 天前
AI agent实现打字机效果
java·http·ai编程
沉鱼.441 天前
第十二届题目
java·前端·算法
努力的小郑1 天前
Canal 不难,难的是用好:从接入到治理
后端·mysql·性能优化
赫瑞1 天前
数据结构中的排列组合 —— Java实现
java·开发语言·数据结构
Victor3561 天前
MongoDB(87)如何使用GridFS?
后端
Victor3561 天前
MongoDB(88)如何进行数据迁移?
后端
小红的布丁1 天前
单线程 Redis 的高性能之道
redis·后端
GetcharZp1 天前
Go 语言只能写后端?这款 2D 游戏引擎刷新你的认知!
后端
周末也要写八哥1 天前
多进程和多线程的特点和区别
java·开发语言·jvm