IDEA 导入 spring 源码


前言

我们在学习 spring 源码的时候,有时候是需要在阅读源码的时候添加一些代码注释,或者是对源码的一些感悟,如果没有将 spring 的源码导入到 IDEA 中,单纯是通过 Maven 去阅读 jar 的形式是无法添加一些注释信息的,即便可以通过一些比如像 Private Notes 这样的插件做到在源代码中添加注释,但还是有局限性的,首先这个插件不允许你更改源码的行数,所以你的私人注释只能写在一行里,其次导入源代码,你可以更改源代码比如说打印一些东西,都是更有助于去理解源码的。

以下内容是我第一次使用 IDEA 去导入 Spring 源码的一个过程,踩了很多坑,我也记录下我所遇见的问题,以及解决的方案,希望对你有所帮助。


spring 官网:https://spring.io/

一、下载源码

我们可以在 github 或者 gitee 上都能下载到 spring 的源码

github 上下载:

github 上搜索 spring-fremawork 就能找到:https://github.com/spring-projects/spring-framework

在这里插入图片描述

gitee 上下载:

gitee 上搜索 spring-fremawork 就能找到:https://gitee.com/mirrors/spring-framework?_from=gitee_search

在这里插入图片描述

虽然说使用可以 git clone 下载源码,但是不太推荐(本人亲自踩过坑,好多问题),这里最好还是直接下载 ZIP 压缩包

我在电脑上创建了 /source-code 这样一个文件夹用于管理我的源代码,然后将下载好的源代码解压缩到该文件夹下,这里我学习的版本是 spring-framework-5.2.x

在这里插入图片描述


二、安装 Gradle


1. 下载 Gradle

Gradle 是一个构建工具,它类似于 Meven,Spring 就是用 Gradle 进行编译的,所以我们还是下载个 Gradle 会比较好。
在这里插入图片描述

下载 Gradle 我们一定要找到跟你下载的 Spring 源码兼容的版本,在你下载好的源码中的 \spring-framework-5.2.x\gradle\wrapper\gradle-wrapper.properties 文件中可以查找到应该下载哪个版本的 Gradle

在这里插入图片描述

所以按照 gradle-wrapper.properties 文件的指定我应该下载 gradle-5.6.4-bin.zip 这个版本的 Gradle

去到官网上下载 https://services.gradle.org/distributions/

在这里插入图片描述

我创建了一个 /gradle 的文件夹存放刚刚下载好的 Gradle,再进行解压

Gradle 和 Meven 一样都有个仓库,这里为了方便,我把 Gradle 的仓库 .gradle 也创建在这个目录下了

在这里插入图片描述


2. 配置环境变量

鼠标右键 我的电脑(此电脑) - 属性 - 高级系统设置 再选择 环境变量

添加链接描述

点击 环境变量 进来大概是这样子的,点击 新建

添加链接描述

如下图所示进行配置,配置完成后,点击确定保存配置

配置 GRADLE_HOME

变量名:GRADLE_HOME
变量值:D:\gradle\gradle-5.6.4-bin\gradle-5.6.4(Gradle 的安装目录,也就是 bin 文件夹所在的目录)

在这里插入图片描述

配置 GRADLE_USER_HOME

变量名:GRADLE_USER_HOME
变量值:D:\gradle\.gradle(Gradle 仓库目录)

在这里插入图片描述

同时还需要添加 Path 的 配置,选择 Path ,点击 编辑

添加链接描述

新建环境变量:

  • %GRADLE_HOME%\bin
  • %GRADLE_USER_HOME%

在这里插入图片描述

确定 保存

现在 cmd 打开命令窗口,输入 gradle -v 命令检测 Gradle 是否安装成功

在这里插入图片描述

能够看到 版本信息 就 ok 了


三、导入前准备

为了加速 gradle 下载依赖包的速度,我们需要在 build.gradlesettings.gradle 这两个文件中添加国内的镜像地址。

在这里插入图片描述

build.gradle 文件

在这里插入图片描述

maven { url "https://maven.aliyun.com/nexus/content/groups/public/"}
maven { url "https://maven.aliyun.com/nexus/content/repositories/jcenter"}
maven { url "https://repo.springsource.org/plugins-release"}

settings.gradle 文件

在这里插入图片描述

maven{ url "https://maven.aliyun.com/nexus/content/groups/public/"}

spring-framework\gradle\wrapper\gradle-wrapper.properties 文件中的 distributionUrl 修改成自己下载的 (PS:这一步可做可不做

在这里插入图片描述

distributionUrl=file\:///D\:/gradle/gradle-5.6.4-bin.zip

四、编译源码


1. 导入源码

打开 IDEA ,选择 File -> Open

在这里插入图片描述

选中 spring 源码所在目录

在这里插入图片描述

导入进来是这样子的

在这里插入图片描述

项目导进来之后先检查下 JDK 的配置,因为 Gradle JVM 是需要使用 JDK 11,所以务必配置 JDK 11,可以避免很多坑

在这里插入图片描述

在这里插入图片描述

然后再配置 Gradle

在这里插入图片描述

检查下 git 配置(编译的时候会自动去检测 git,所以需要检查下)

在这里插入图片描述

接下来就等项目自动构建,因为这个过程 Gradle 会下载一些 jar 包需要一些时间。

在这里插入图片描述

当你看到每一个项目 右下角 基本上都带着一个 蓝色方块,就表示 Spring 的源码导入成功了。

如果你在上述过程中遇到问题,不妨往下面看看 ~~


2. 我所遇见的问题

  1. fatal: not a git repository (or any of the parent directories): .git

在这里插入图片描述

解决这个问题很简单,进到这层目录下, 调起 Git Bash Here ,依次执行以下三行命令

git init
git add .
git commit -m "随便写点啥都行"

在这里插入图片描述

在去 IDEA 中重新构建一下就好了

  1. Gradle 版本导致 A problem occurred evaluating root project 'spring'. 等问题

在这里插入图片描述

之前我是按照 gradle.properties 这个文件下载的 gradle-5.3-bin.zip,我以为够了,没想到 java-test-fixtures 这些包下载不到,没办法,我只能根据 gradle-wrapper.properties 再下载一个 gradle-5.6.4-bin.zip 的版本

在这里插入图片描述

重新设置一下 IDEA 的 GreadleSpecified location,选择刚下的 Gradle gradle-5.6.4-bin.zip 的版本

在这里插入图片描述

所以还是要以 gradle-wrapper.properties 文件指定的 Gradle 版本为准会避免掉一些坑

  1. POM relocation to an other version number is not fully supported in Gradle : xml-apis:xml-apis:2.0.2 relocated to xml-apis:xml-apis:1.0.b2

在这里插入图片描述

我重新构建下就好了,所以也没啥好说的


五、测试


1. 创建 module

现在源码是顺利的导入进来了,那就在 spring 的项目中新建一个 module 来测试一下

右键项目根目录,New -> Module 创建一个 module

在这里插入图片描述

这个要选 Gradle,点击 Next

在这里插入图片描述

设置模块名,点击 Finish

在这里插入图片描述

就 ok 了

在这里插入图片描述


2. 编写测试代码

在这里插入图片描述

替换成

在这里插入图片描述

    testImplementation group: 'junit', name: 'junit', version: '4.13.1'
    testImplementation group: 'org.hamcrest', name: 'hamcrest-core', version: '1.3'
    compile(project(":spring-context"))

这里简单说明以下,Gradle 如果引用的是本项目中的模块,则使用 compile(project(":moduleName)),如果使用的是第三方的包,则使用 testImplementation group:xxxx ' 导入

测试结果:

可以看到是没什么问题的,没有报错,并且使用的类是本项目的,能够正确执行

在这里插入图片描述

测试代码如下:

UserController.calss

package com.mike.controller;

import com.mike.service.UserService;

public class UserController {

	private UserService userService;

	/**
	 * DI:set 方法注入
	 */
	public void setUserService(UserService userService) {
		this.userService = userService;
	}

	public String findAll() {
		return userService.findAll();
	}
}

UserService.java

package com.mike.service;

public interface UserService {

	String findAll();

}

UserServiceImpl.java

package com.mike.service.impl;

import com.mike.service.UserService;

public class UserServiceImpl implements UserService {

	@Override
	public String findAll() {
		return "返回所有用户信息";
	}

}

MainTest.java

package com.mike.test;

import com.mike.controller.UserController;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MainTest {

	@Test
	public void test() {
		// 获取 Spring 容器对象
		// 执行这行代码相当于启动了 Spring 容器,解析 spring.xml 文件,并且实例化所有的 bean 对象放到 spring 容器中
		ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:spring.xml");
		// 获取 UserController 对象
		UserController userController = applicationContext.getBean("userController", UserController.class);
		// 执行方法
		String allUsers = userController.findAll();
		System.out.println("allUsers = " + allUsers);
	}
}

spring.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
	   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	   xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">


	<!-- 使用 Spring IOC 和 ID 管理对象 -->
	<bean id="userService" class="com.mike.service.impl.UserServiceImpl"/>

	<bean id="userController" class="com.mike.controller.UserController">
		<property name="userService" ref="userService"/>
	</bean>

</beans>

如果你在上述过程中遇到问题,不妨往下面看看 ~~


3. 我所遇到的问题

  1. build.gradle 怎么配置第三方依赖

比如我写的测试模板中需要 junit,在 pom 文件中我们可以这样导入

        <!-- junit 单元测试 -->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.13.1</version>
            <scope>test</scope>
        </dependency>

但是在 build.gradle 文件中我们该怎么去写呢?

其实在 Maven 仓库 中就提供了 Gradle 的导入方式,直接复制过来就行了

在这里插入图片描述

  1. Kotlin: warnings found and -Weeror specified

在我写完测试代码之后运行 MainTest 类中的 test() 方法所报

在这里插入图片描述

这个问题出现的原因是 缺少 cglib 和 objenesis 包

解决:在这个目录下调起命令行窗口

在这里插入图片描述

输入以下两个指令:

gradle objenesisRepackJar
gradle cglibRepackJar

在这里插入图片描述

  1. java: 找不到符 符号:变量 CoroutinesUtils

解决上一个问题之后再运行 MainTest 类中的 test() 方法所报,这个问题可以说经常会遇到,每构建一次项目都有可能出现

在这里插入图片描述

这是因为 CoroutinesUtilskotlin 的一个工具类,Spring 源码包读取不到,所以需要手动添加 kotlin-coroutines-X.X.X.BUILD-SNAPSHOT.jarLibraries 中。

点击 File -> Project Structure -> Libraries -> + -> Java

在这里插入图片描述

然后选择 spring-framework/spring-core/kotlin-coroutines/build/libs/kotlin-coroutines-X.X.X.BUILD-SNAPSHOT.jar 点击 OK (如果你的 kotlin-coroutines 目录下没有 build 目录,跳转到问题3)

在这里插入图片描述

然后就会跳出一个选择 modules 的页面,选中 spring.spring-core.main 点击 OK

在这里插入图片描述

  1. kotlin-coroutines 目录下没有 build 目录

如果你和我一样并没有发现

在这里插入图片描述

点击右侧的 Gradle -> Tasks -> other -> complieKotlin

在这里插入图片描述

编译之后就有了

  1. compileXXXX 时或者在运行时报错 PropertyComparator<capture#1, 共 ?>无法转换为java.util.Comparator<? super capture#1, 共 ?> 等各种泛型使用问题

比如 compileKontlin 时

D:\source-code\spring-framework-5.2.x\spring-framework-5.2.x\spring-beans\src\main\java\org\springframework\beans\support\PropertyComparator.java:138: ����: �����ݵ�����: PropertyComparator<CAP#1>�޷�ת��ΪComparator<? super CAP#1>
			source.sort(new PropertyComparator<>(sortDefinition));
			            ^
  ����, CAP#1�������ͱ���:
    CAP#1��?�IJ�����չObject

在这里插入图片描述

或者是运行 main 方法、Test 方法 时

D:\source-code\spring-framework-5.2.x\spring-framework-5.2.x\spring-beans\src\main\java\org\springframework\beans\support\PropertyComparator.java:138:25
java: 不兼容的类型: org.springframework.beans.support.PropertyComparator<capture#1,?>无法转换为java.util.Comparator<? super capture#1,?>

在这里插入图片描述

这个问题感觉是跟泛型有关的坑,我就改了下源码

在这里插入图片描述

再编译之后这个问题就没有再出现了,不过又出现新的问题

D:\source-code\spring-framework-5.2.x\spring-framework-5.2.x\spring-messaging\src\main\java\org\springframework\messaging\handler\annotation\reactive\PayloadMethodArgumentResolver.java:236: ����: �����ݵ�����: lambda ���ʽ�еķ������ʹ���
							.onErrorResume(ex -> Flux.error(handleReadError(parameter, message, ex)));
							                               ^
    Flux<CAP#1>�޷�ת��ΪPublisher<? extends CAP#1>
  ����, CAP#1�������ͱ���:
    CAP#1��?�IJ�����չObject

在这里插入图片描述

也跟泛型有关,所以修改这边的代码不是个很好的解决办法,改了一个后续还会遇见很多个,治标不治本

这个问题出现的原因是 gradle jvm 的版本没有使用 jdk 11导致的,所以 Gradle JVM 务必换成 JDK 11

在这里插入图片描述

Project 的 JDK 也改成 11

在这里插入图片描述

再编译就没问题了

在这里插入图片描述


六、总结

以上就是我使用 IDEA 导入 Spring 源码的全过程,不得不说 IDEA 导入 Spring 源码真的有好多的坑啊,不知道多少学习源码的人在这里就被劝退了 ~~

其中还有些错误我没有列举出来,因为这是我优化过后总结出来的,所以另外些错误我没有再遇到过就不写在这上面了,在这个过程中觉得最大的阻碍就是对 Gradle 的不熟悉,很多问题都跟 Gradle 有关,好在最后都解决了,也希望这篇文章能够让你在学习源码的过程中有个好的开头。


参考博客:
《Spring》第一篇 IDEA导入Spring源码:https://blog.csdn.net/weixin_44167408/article/details/121769949
知乎:spring源码编译的过程和问题:https://zhuanlan.zhihu.com/p/378831634

  • 11
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值