# Java 技术雷达:工具篇

Java 相关工具

gradle

项目网址

Overview

gradle 是一款基于JVM 的自动化构建工具。

gradle.build文件是gradle 的配置文件,�其使用groovy 来构建DSL 脚本。

在gradle 中, 有两个基本概念:

  • 项目是构建产物(如jar 包)或实施产物(将应用程序部署到生成环境),一个项目包含若干个任务。
  • 任务是指不可分割的最小工作单元,任务用来执行真正的构建工作。

在gradle 中,所有的特性都是由gradle 插件提供的, gradle 本身只提供了DSL 语言。

Usage

  • 应用gradle 插件:

    apply plugin: 'java'
    
  • 定义变量:

    project.ext {
        guavaVersion = '18.0'
        ...
    }
    
  • 定义依赖:

    project.dependencies {
        compile(...)
        runtime(...)
        testCompile(...)
    }
    
  • 定义任务:

    task uberjar(type: Jar, dependsOn: jar) {...}
    

Tips

  • 由于gradle 的各个版本间存在兼容性问题,使用wrapper 让每个项目的构建环境独立。

    在build.gradle 文件中使用以下的配置来指定本项目使用的gradle 半版本信息:

    task wrapper(type: Wrapper) {
        gradleVersion = '2.5'
    }
    
  • gradle的项目构建过程分为三个步骤:初始化, 配置和执行任务。使用org.gradle.daemon=true 来启动后台进程来在构建过程中跳过初始化步骤,从而加速gradle 的构建执行。

  • 将项目拆分为多个子项目,可以利用gradle 提供的并行构建能力。

    • 在root 项目中的build.gradle中定义公共变量,任务和依赖。
    • 在root 项目中的settings.gradle中定义包含的子项目。
    • 在各个子项目的build.gradle中定义各项目所需的变量,任务和依赖。
  • 从cvs 中新获取的项目后,在命令行中依次执行以下的命令:

    >> gradle wrapper
    >> ./gradlew clean idea
    >> ./gradlew clean check
    

    如果check任务通过,说明项目已经在本地成功构建。

Nexus

项目网址

Overview

nexus是一款支持多种流行组件(如Maven,Docker 等)的仓库管理软件。

在nexus 中,有两个核心的概念:

  • component:一种资源,例如library或者framework。作为应用程序的一部分,component 可以在运行时,集成,单元测试执行,或者在构建过程中被引用。

    build.gradle中的表现为:

    project.dependencies {
        compile(...)
        runtime(...)
        testCompile(...)
    }
    

    component 通常又被称为artifact, package, bundle, achive 等。

  • repository:为了方便components 的使用,repository 聚合了components 集合,并在internet 上提供service。

    • 组件通常有不同的格式,例如maven repository 依赖于特殊的目录结构和XML 格式的元数据, 并通过plain HTTP 命令和附带的XML 文件来对component 进行交互。

Usage

  • 指定从自建的nexus 仓库中获取依赖:

    buildscript {
        repositories {
            jcenter()
            maven {
                url "your-nexus-repository-url"
            }
        }
        dependencies {
            classpath 'your-dependencies'
        }
    }
    
  • gradle upload命令的上传配置:

    project.uploadArchives {
        repositories {
            mavenDeployer {
                repository(url: "your-nexus-repository-url") {
                    authentication(userName: "", password: "")
                }
                pom.version = "${project.version}"
                pom.artifactId = "${project.name}"
                pom.groupId = "${project.group}"
            }
        }
    }
    

Tips

  • 本地缓存导致的依赖不及时刷新新版本:

    configurations.all {
      resolutionStrategy.cacheChangingModulesFor 0, 'hours'
    }
    
* 根据版本号来分别存储snapshot 和release 组件:   

mavenDeployer {
          repository(url: "your-nexus-repository-url" +
                  "${project.version.endsWith('-SNAPSHOT') ? 'snapshots' : 'releases'}") {...
​```

checkstyle

项目网址

Overview

checkstyle 是一款Java 代码风格的检查工具。

它是高度可配置化的,通过配置可以支持任意的代码风格(例如Sun Code ConvensionsGoogle Java Style)。

checkstyle 的配置文件位于project/config/checkstyle/sun_checks.xml

通过修改该配置文件来统一项目的代码风格。例如,出于源码阅读的考虑,想要限制代码行的长度为100 个字符,则修改以下的配置:

<module name="LineLength">
     <property name="max" value="100"/>
</module>

Usage

在项目的build.gradle文件中,通过以下的配置使用checkstyle:

apply plugin: 'checkstyle'

checkstyle {
    configFile = file("$project.projectDir/config/checkstyle/sun_checks.xml")
    toolVersion = '6.7'
}

// 是否对测试代码进行代码风格的检查。
checkstyleTest {
    enabled = false
}

Tips

  • 不可变形参需要加上final.
  • 将数字定义为有意义的static final常量(否则,会被认为是magic number)。
  • 在代码提交前, 使用IDE 提供的format 快捷键, 过滤掉一些低级的风格问题(如无用的引用, 代码对其等等)。

findbugs

项目网址

Overview

findbugs 是一款静态检查Java 代码中bug 的工具。

在项目开发过程中,执行gradle check 命令时,findbugs 会在coverageCheck 通过后执行。

如果代码中含有工具认为可能会引发Bug 的写法,那么会在/yourProject/build/reports/findbugs目录中的生成一个main.xml文件来记录。

Usage

如果在check过程中,出现了Execution failed for task ':findbugsMain'.,那么打开main.xml 文件并找到以下的节点:

  <BugInstance type="" priority="" rank="" abbrev="" category="">

根据type 字段,在网站中可以找到对应的bug 的描述。

根据Class, Field, SouceLine 字段, 可以在Java 源代码中找到bug 对应的源码位置。

Tips

  • 方法忽略了返回值(这常见于对不可变对象进行调用方法,而误以为不可变对象会被更新)。
  • 忘记资源释放(方法可能没有关闭stream)。
  • switch-case 语句块中缺少break关键字。

jacoco

项目网址

Overview

jacoco 是一款检查代码单元测试覆盖率的工具。

jacoco 的配置文件位于config/scripts/coverage.gradle

如果代码的覆盖率没有达到指定的标准,那么会在yourproject/build/report/jacoco 目录下生成index.html 和相关的结果文件,打开该文件,可以看到图形化显示的没有覆盖到的代码所在的文件和代码位置。

Usage

  • 在gradle 项目中,在该配置文件中定义jacoco 在gradle 的扩展和任务:
project.extensions.create('coverage', CoverageExtension)
task coverageCheck(dependsOn: test) << {...}
  • 在实际使用中,我们定义了excludePackagesexcludeClasses 对象来让jacoco 跳过一些包和类的覆盖率检查。
coverage.excludePackages.each() {
    exclude name: "${it.replaceAll('\\.', '/') + '/*'}"
}
coverage.excludeClasses.each() {
    exclude name: "${it.replaceAll('\\.', '/') + '.class'}"
}
  • 然后在build.gradle中使用以下的配置来在gradle check 任务中启用jacoco 任务。
check.dependsOn "coverageCheck"

Tips

  • 尽量使用TDD 的开发方式。
  • 在实际开发过程中,除了一些无法模拟的Exception 之外,尽量不要使用exclude 来跳过单元测试覆盖率的检查。

flyway

项目网址

Overview

flyway 是一款数据库migration 工具。

相比于手动的sql 脚本的数据库数据版本管理,数据库migration 能够带来以下的优势:

  • 从零开始重建一个数据库。
  • 清晰地获知当期数据库状态。
  • 从当前版本到任意的一个新状态。

flyway 使用schema_version 数据表管理数据库的数据版本。

Usage

  • 在build.gradle 中进行以下的配置:

    apply plugin: "org.flywaydb.flyway"
    
    flyway {
      url = 'your-database-schema-url'
      user = ''
      password = ''
    }
    
  • 在以下的位置中放置数据库migration 脚本:
    yourProject/src/main/resources/db.migration

  • 数据库migration 脚本的命名格式:
    V1__Create_demo_table.sql。 需要注意的是版本号后面是两个下划线。

Tips

  • 在测试中自动化地进行flywayCleanflywayMigrate

    test.dependsOn "flywayClean"
    test.dependsOn "flywayMigrate"
    
  • 数据schema 的更新原则是只增不删,例如要修改某个既存表的结构,那么在一个新建的migration 脚本中进行,而不用去修改该寄存表的创建脚本。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容

  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,801评论 6 342
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,647评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,028评论 25 707
  • 1,简介 1)gradle基于JVM的构建工具,需要安装JDK或者JRE。2)全面支持已有的maven和ivy仓库...
    沐兮_d64c阅读 1,120评论 0 3
  • 早起是一件可以令人开心的事,但早起到医院检查身体可以让一天开心的心情都烟消云散,尤其是还要排着长长的队伍等待...
    Sophie七小朵阅读 259评论 0 0