小马哥 Java 分布式架构训练营 一期二期

491 阅读7分钟

小马哥 Java 分布式架构训练营

代码注释,必看↓

// download: 3w ukoou com

Java 分布式架构训练营 - 理解 Spring 模块化设计

构建工具

首先是构建工具的选择

现在主流的构建工具基本上就是GradleMaven

一般最早开始学后端的同学都是习惯用Maven

而像我这种从Android转后端的还是Gradle比较好上手

构建工具的话其实两者都可以,大家可以根据自己的习惯来选择

不过我还是建议大家可以尝试一下Gradle

我后面的示例也是基于Gradle来构建

项目结构

那我们现在就开始创建项目吧

我们要创建的项目叫juejin,包含三个模块:juejin-user(用户)juejin-pin(沸点)juejin-message(消息)

结构1(不推荐)

Long long ago,这个世界上存在这样一种项目结构

项目结构1.jpg

每个模块是一个单独的项目,然后当时有10多个模块,而且还是我负责搭建的

现在想想,我真的是。。。想立马穿越回去,给自己来一个大嘴巴子

我给大家说说这种方式的优劣

缺点:

  • 多模块同时打开很卡

    • 每一个模块都是一个项目,模块多的情况下没办法同时打开,以一般公司的电脑配置,会很卡
    • 只能要改哪个模块开哪个模块项目,有新的要开就把已经打开的关掉一个
    • 有的时候刚关掉一个模块,突然想到有个地方漏改了,又把之前关掉的再打开,效率直接-50%
  • 版本管理麻烦

    • git管理也是多个项目,拉分支切分支也很麻烦
    • 有的时候一个模块需要修改就单独拉了一个分支,最后有部分模块有某个分支,有部分模块没有
    • 如果没有人专门做整理记录,这些分支大概率最后直接乱成好几副耳机线,就算有记录管理起来也是极其麻烦

优点:

  1. 无!

如果大家现在的项目是这种方式的话,emmmm,耗子尾汁吧

不过如果模块数不多,只有3-4个,这种方式也算能接受

或是需要做代码权限,如不让所有人都能看到所有代码而特意分开的话,这种是属于特殊情况了

但我还是建议大家,只要不是公司硬性规定,还是不要用这种方式来搭建

结构2(推荐)

有了前车之鉴之后,我肯定要优化一下项目的结构

我们先创建一个项目,名称叫juejin

项目结构2创建.jpg

大家可以根据自己的习惯或是公司的规定,选择JavaKotlinGradleMaven

这里不用选择Spring Initializer,因为最外面的项目只是一个目录

点击Create之后就会生成一个项目

项目结构2项目.jpg

项目已经建好了,接下来我们来建模块

通过File => New => Module来创建一个模块

在创建业务模块之前,我建议大家可以建一个basic模块,用于提供全局的配置和依赖,当然名字大家可以按照自己的习惯或是经过讨论来决定,basic只是我习惯的名称

项目结构2模块basic.jpg

同样的,点击Create就会在juejin的目录下生成一个juejin-basic模块

项目结构2模块basic.jpg

用相同的操作,我们把juejin-userjuejin-pinjuejin-message这几个模块也都创建出来

项目结构2功能模块.jpg

这里大家可以自行选择用不用Spring Initializer

  • 如果不用Spring Initializer则需要手动修改build.gradle补插件和依赖
  • 如果使用Spring Initializer则需要手动删除一些无用的文件和依赖

我这里没有选择Spring Initializer

如果大家使用了Spring Initializer可以对照着我的示例进行添加删除和修改就行了

我们的项目结构基本就是这样了,相比于之前每个模块都是一个项目的方式,这样搭建所有的模块一目了然而且方便管理

小马哥分布式架构训练营 - 深入了解Spring-boot-starter常用依赖模块

Spring-boot的2大优点:

1.基于Spring框架的“约定优先于配置(COC)”理念以及最佳实践之路。
2.针对日常企业应用研发各种场景的Spring-boot-starter自动配置依赖模块,且“开箱即用”(约定spring-boot-starter- 作为命名前缀,都位于org.springframenwork.boot包或者命名空间下)。

应用日志和spring-boot-starter-logging

常见的日志系统大致有:java.util默认提供的日志支持,log4j,log4j2,commons logging,下面的spring-boot-starter-logging也是其中的一种。

maven依赖:

**

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-logging</artifactId>
        </dependency>

springBoot将使用logback作为应用日志的框架,程序启动时,由org.springframework.boot.logging-Logging-Application-Lisetener根据情况初始化并使用。
如果要想改变springBoot提供的应用日志设定,可以通过以下原则:

  • 遵循logback的约定,在classpath中使用自己定制的logback.xml配置文件。
  • 在文件系统的任意一个位置提供自己的logback.xml配置文件,然后通过logging.config配置项指向这个配置文件然后引用它,例如在application.properties中指定如下的配置:

**

logging.config=/{some.path.you.defined}/any-logfile-name-I-like.log}

快速web应用开发与spring-boot-starter-web

maven依赖:

**

       <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

在当下项目运行mvn spring-boot:run就可以直接启用一个嵌套了tomcat的web应用。
如果没有提供任何服务的Cotroller,访问任何路径都会返回一个springBoot默认的错误页面(Whitelabel error page)。

嵌入式Web容器层面的约定和定制

spring-boot-starter-web默认使用嵌套式的Tomcat作为Web容器对外提供HTTP服务,默认端口8080对外监听和提供服务。
我们同样可以使用spring-boot-starter-jetty或者spring-boot-starter-undertow作为Web容器。
想改变默认的配置端口,可以在application.properties中指定:

**

server.port = 9000(the port number you want)

类似的配置还有:

**

server.address
server.ssl.*
server.tomcat.*

如果上诉仍然没有办法满足要求,springBoot支持对嵌入式的Web容器实例进行定制,可以通过向IoC容器中注册一个EmbeddedServletContainerCustomizer类型的组件来对嵌入式的Web容器进行定制

**

public class UnveilSpringEmbeddedTomcatCustomizer implements EmbeddedServletContainer{
        public void customize(ConfigurableEmbeddedServletContainer container){
            container.setPort(9999);
            container.setContextPath("C\hello");
                           ...
        }
    }

数据访问与spring-boot-starter-jdbc

maven依赖:

**

    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-jdbc</artifactId>
    </dependency>

默认情况下,当我们没有配置任何DataSource,SpringBoot会为我们自动配置一个DataSource,这种自动配置的方式一般适用于测试,开发还是自己配置一个DataSource的实例比较好。
如果我们的工程只依赖一个数据库,那么,使用DataSource自动配置模块提供的参数是最方便的:

**

spring.datasource.url=jdbc:mysql://{datasource host}:3306/{databaseName}
spring.datasource.username={database username}
spring.datasource.passwd={database passwd}

还会自动配置的有:JdbcTemplate DateSourceTransactionManager等,我们只要在使用的时候注入(@Autowired)就好了
此外,SpringBoot还支持的数据库有spring-boot-data-jpa spring-boot-data-mongodb

spring-boot-starter-aop应用及其使用场景

AOP:Aspect Oriented Programming,面向切面编程

maven依赖:

**

    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-aop</artifactId>
    </dependency>

spring-boot-starter-aop主要由2部分组成:
1.位于spring-boot-autoconfigure的org.sringframework.boot.autoconfigure.aop.AopAutoConfiguration提供的@Configuration配置类和相应的配置项,即下面的2个配置项:

**

spring.aop.auto=true
spring.aop.proxy-target-class=false

2.spring-boot-starter-aop模块提供了针对spring-aop aspectjrt 和aspectjweaver的依赖

应用安全与spring-boot-starter-security

spring-boot-starter-security 主要面向的是Web应用开发,配合spring-boot-starter-web,所以,对应的maven依赖如下:

**

       <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-security</artifactId>
        </dependency>

spring-boot-starter-security 默认会提供一个基于HTTP Basic认证的安全防护策略,默认用户为user,访问密码则在当前web应用启动的时候,打印到控制台,要想定制,则在配置文件如下进行配置:

**

security.user.name={username}
security.user.password={passwd}

除此之外,spring-boot-starter-security还会默认启动一些必要的Web安全防护,比如针对XSS CSRF等场针对Web应用的攻击,同时,也会将一些常见的静态资源路径排除在安全防护之外。

AuthenticationManager AccessDecisionManager AbstractSecurityInterceptor被称为Spring Security的基础铁三角,前2者负责制定规则, AbstractSecurityInterceptor负责执行。我们常见的Filter类就可以定制和扩展SpringSecurity的防护机制。

进一步定制spring-boot-starter-security

上诉使用SecurityProperties暴露的配置项,即以security.*对spring-boot-starter-security进行简单的配置,还可以通过给出一个继承了WebSecurityConfigurerAdapter的JavaConfig配置类对spring-boot-starter-security的行为进行更深级别的定制,例如:

  • 使用其他的AuthenticationManager实例
  • 对默认的HttpSecurity定义的资源访问的规则重新定义
  • 对默认提供的WebSecurity行为进行调整
    一般配合@Order(SecurityProperties.ACCESS_OVERRIDE_ORDER)进行标注。