使用工具类(一)

开发过程中的效率提升工具

为了更愉快的写代码,开发过程中常常需要一些功能类来提高工作效率,记录一下用到过的比较好的开发效率工具

1、java StopWatch统计时间

       有时候记录一段代码的执行时间,最常见的方法是打印当前的时间与执行后的差值,确定是使用很麻烦并且很不直观。spring-framework提供了一个stopWatch类可以做类似任务的执行时间控制。StopWath是apache commons lang3包下的一个任务执行时间监视器。

主要方法:

start();     //开始计时

split();     //设置split点

getSplitTime();  //获取从start 到 最后一次split的时间

reset();     //重置计时

suspend();     //暂停计时, 直到调用resume()后才恢复计时

resume();      //恢复计时

stop();      //停止计时

getTime();    //统计从start到现在的计时

例子:

package com.example.stopwatch;

import org.springframework.util.StopWatch;

public class StopWatchTest {

    private void test() throws InterruptedException {

        StopWatch sw = new StopWatch();

        sw.start("起床");

        Thread.sleep(1000);

        sw.stop();

        sw.start("洗漱");

        Thread.sleep(2000);

        sw.stop();

        sw.start("锁门");

        Thread.sleep(500);

        sw.stop();

        System.out.println(sw.prettyPrint());

        System.out.println(sw.getTotalTimeMillis());

        System.out.println(sw.getLastTaskName());

        System.out.println(sw.getLastTaskInfo());

        System.out.println(sw.getTaskCount());

    }

    public static void main(String []argv) throws InterruptedException {

        TestStopWatch testStopWatch = new TestStopWatch();

        testStopWatch.test();

    }

}

结果如下


实验结果


2、Swagger Api

可以无需手写,自动生成接口的API文档的工具

1、Maven依赖

<dependency>

    <groupId>io.springfox</groupId>

    <artifactId>springfox-swagger2</artifactId>

    <version>2.2.2</version>

</dependency>

<dependency>

    <groupId>io.springfox</groupId>

    <artifactId>springfox-swagger-ui</artifactId>

    <version>2.2.2</version>

</dependency>

2、配置类

/**

* Swagger2配置类

* 在与spring boot集成时,放在与Application.java同级的目录下。

* 通过@Configuration注解,让Spring来加载该类配置。

* 再通过@EnableSwagger2注解来启用Swagger2。

*/

@Configuration

@EnableSwagger2

publicclass Swagger2 {


/**

    * 创建API应用

    * apiInfo() 增加API相关信息

    * 通过select()函数返回一个ApiSelectorBuilder实例,用来控制哪些接口暴露给Swagger来展现,

    * 本例采用指定扫描的包路径来定义指定要建立API的目录。

    *

    * @return

    */

@Bean

public Docket createRestApi() {

returnnewDocket(DocumentationType.SWAGGER_2)

                .apiInfo(apiInfo())

                .select()

.apis(RequestHandlerSelectors.basePackage("com.swaggerTest.controller"))

                .paths(PathSelectors.any())

                .build();

    }


/**

    * 创建该API的基本信息(这些基本信息会展现在文档页面中)

    * 访问地址:http://项目实际地址/swagger-ui.html

    * @return

    */

private ApiInfo apiInfo() {

returnnewApiInfoBuilder()

.title("Spring Boot中使用Swagger2构建RESTful APIs")

.description("更多请关注http://www.baidu.com")

.termsOfServiceUrl("http://www.baidu.com")

.contact("sunf")

.version("1.0")

                .build();

    }

如上代码所示,通过createRestApi函数创建Docket的Bean之后,apiInfo()用来创建该Api的基本信息(这些基本信息会展现在文档页面中)。

3、使用方法介绍

Swagger使用的注解及其说明:

@Api:用在类上,说明该类的作用。

@ApiOperation:注解来给API增加方法说明。

@ApiImplicitParams : 用在方法上包含一组参数说明。

@ApiImplicitParam:用来注解来给方法入参增加说明。

@ApiResponses:用于表示一组响应

@ApiResponse:用在@ApiResponses中,一般用于表达一个错误的响应信息

code:数字,例如400

message:信息,例如"请求参数没填好"

response:抛出异常的类

@ApiModel:描述一个Model的信息(一般用在请求参数无法使用@ApiImplicitParam注解进行描述的时候)

@ApiModelProperty:描述一个model的属性


/**

* 一个用来测试swagger注解的控制器

* 注意@ApiImplicitParam的使用会影响程序运行,如果使用不当可能造成控制器收不到消息

*

*@authorSUNF

*/

@Controller

@RequestMapping("/say")

@Api(value ="SayController|一个用来测试swagger注解的控制器")

publicclass SayController {


@ResponseBody

@RequestMapping(value ="/getUserName", method= RequestMethod.GET)

@ApiOperation(value="根据用户编号获取用户姓名", notes="test: 仅1和2有正确返回")

@ApiImplicitParam(paramType="query", name ="userNumber", value ="用户编号", required =true, dataType ="Integer")

public String getUserName(@RequestParam Integer userNumber){

if(userNumber ==1){

return"张三丰";

        }

elseif(userNumber ==2){

return"慕容复";

        }

else{

return"未知";

        }

    }


@ResponseBody

@RequestMapping("/updatePassword")

@ApiOperation(value="修改用户密码", notes="根据用户id修改密码")

@ApiImplicitParams({

@ApiImplicitParam(paramType="query", name ="userId", value ="用户ID", required =true, dataType ="Integer"),

@ApiImplicitParam(paramType="query", name ="password", value ="旧密码", required =true, dataType ="String"),

@ApiImplicitParam(paramType="query", name ="newPassword", value ="新密码", required =true, dataType ="String")

    })

public String updatePassword(@RequestParam(value="userId") Integer userId, @RequestParam(value="password") String password,

@RequestParam(value="newPassword") String newPassword){

if(userId <=0|| userId >2){

return"未知的用户";

      }

if(StringUtils.isEmpty(password) || StringUtils.isEmpty(newPassword)){

return"密码不能为空";

      }

if(password.equals(newPassword)){

return"新旧密码不能相同";

      }

return"密码修改成功!";

    }

完成上述代码添加上,启动Spring Boot程序,访问:http://localhost:8080/swagger-ui.html

其他注意事项:

1、paramType会直接影响程序的运行期,如果paramType与方法参数获取使用的注解不一致,会直接影响到参数的接收。

2、Conntroller中定义的方法必须在@RequestMapper中显示的指定RequestMethod类型,否则SawggerUi会默认为全类型皆可访问, API列表中会生成多条项目。

工具类

所有的工具类,优先采用 apache commons 系列;

apache commons未提供的工具可以自行扩展,也应该是在其基础之上进行扩展,而不是另外实现

编码及加解密方法规范

应用中资源统一采用 UTF-8 编码,统一采用Spring提供的CharacterEncodingFilter来实现。

常见的加解密统一采用 apache commons-codec 库所提供的方法,具体 API 及文档参见:https://commons.apache.org/proper/commons-codec

关于金额类型的说明

应用中涉及到金额时,在系统、数据库设计的时候通常有两种做法:

1. Decimal 格式:(也是较通用的格式),数据库和Java中都有对应的类型支持;

优点:直观,在使用的过程中也无需转换。精度方面也支持得非常好,可以随时动态调整;

缺点:API 使用相对复杂一点,对应的 Java 对象为 BigDecimal ,做四则运算的时候都是对象方法调用;

2. Int 格式:通常按照一定的小数点留存数量乘以相应的倍数,将金额转换成分(美分);

比如: 1.88 元 = 188 分

优点:数据类型简单,做四则运算时也比较快捷方便;

缺点:精度无法动态变换,在用户输入/展示与数据存储之间需要进行数据转换;

注:由于真正做运算的场景不多, 相关 API 也不会频繁的调用,所以根据以上的比较,规定本中心所有金额相关的数据类型,如无特殊原因需要说明外,统一采用 Decimal 数据格式;

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

推荐阅读更多精彩内容