多参数方法进阶

很多高级工程师还在写包含N个参数的方法、使用setter方法构造实例,其实这些方式都是过时并且有很大缺陷的,本篇将深入讲解这些问题及解决方法。

多参数方法的问题

相信很多人曾经都写过多参数的构造方法,就像下面示例的代码。

当想要创建一个给全部属性赋值的实例的时候,就会利用这个多参数的构造方法。但是,当类的属性特别多的时候,你还会这么写吗?

如果你写了一个包含N个参数的多参数构造方法,当创建实例的时候,是不是特别谨慎的在大脑里记着第几个参数是给哪个属性赋值的,生怕给弄错了,当然类型不对的时候,编译器还会友善的提醒赋值错了,但对那些类型相同的属性,如果不小心颠倒了两个参数的顺序,编译器也爱莫能助了,程序运行起来也不会报错,就是执行结果不是期望值。

如果有人写了一个包含N个参数的多参数构造方法并打包后提供给你使用,又如果你看不到方法的源代码或Javadoc,更不巧这些参数的名称很随意(例如arg1、arg2...argN等),你会不会恶狠狠的在背后慰问代码的作者。

这种多参数的构造方法,当参数的个数小于等于3的时候不是很糟糕,但是,随着参数个数越来越多的时候,这种构造实例的方法就会失去控制,变得难以使用。

看到这里,也许有人会说,当然不会写一个包含N个参数的多参数构造方法,可以使用默认的构造方法创建实例,然后使用setter方法给各个属性一一赋值,就像下面示例代码这样。

不得不说这种方式创建实例很容易,产生的代码也易读,同时也是使用最频繁的方式。但是,它有很严重的缺点,就是一旦提供了属性的setter方法,则代表你可以在任何地方、任何时候给某个或某些属性单独赋值,致使实例的构造被分散到各个角落,最终可能会导致连代码原作者都不知道在哪里给某个属性“又”赋了值,导致实例数据不一致,调试查找问题也很麻烦。

这里笔者分享一个曾经经历的,使用setter方法给属性赋值的坑。很早之前使用Hibernate进行CRUD操作,大家都知道Hibernate的get和load方法查询出的实体对象会被缓存,并且是处于持久态的,持久态的实体无论你显示调用update操作或不调用,只要它在一个事务中,当事务被提交的时候,它会比较缓存中实体数据和快照区中的实体数据,将变化主动更新到数据库中,坑就出现在这里。查询出来的实体对象被当做参数传递到了其它方法中,而此时为了某些操作的需要,调用了某个属性的setter方法赋值,但是并没有显示调用update方法,导致看上去每次执行完查询数据就被自动更新,查找原因起来不是很容易。这就是使用setter方法导致的赋值分散的问题。

相信大多数人都写过或使用过多参数的普通方法,就像下面示例的代码。

可能会有人觉得这没什么,不就5个参数嘛!没事,就这么写!对于这种觉悟,笔者只能说你试试10个参数或更多,保证你会精神崩溃,即使你能忍受,调用你方法的同事保证会在背后慰问你。

这种写法也同样有上面多参数构造方法的问题,参数多了方法就会失去控制,难以使用。

多参数方法指南

对于多参数的构造方法,使用Builder模式代替,就像下面的示例代码。

从实例代码可以看到,使用Builder模式构造实例根本不需要费尽心思的牢记参数列表,想给哪个属性赋值直接显示的给哪个属性赋值,再也不怕赋错值,代码清晰明了!注意看示例代码中,实际给bank实例属性赋值的地方,即下面的代码。

可以看到在Bank的构造方法里集中给各个属性赋值,并不会分散赋值。如果想再次给某个属性赋值,可以像示例代码这样。

对于普通方法,如果方法参数的个数大于三,将这些参数抽象成一个类,这些参数就是类的属性,然后在多参数的方法中,使用这个类代替这些参数,同时在这个参数类中提供Builder模式代码用于创建类的实例,就像下面的示例代码。

原文发布于微信公众号 - JavaQ(Java-Q)

原文发表时间:2017-12-27

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏菩提树下的杨过

ruby学习笔记(11)--symbol与hash参数

symbol是啥就不深入的讨论了,只简单说说symbol的好处 ruby内部对于每个对象,都会有一个数字id用来标识并区分,可以用xxx.object_id来查...

191100
来自专栏Java帮帮-微信公众号-技术文章全总结

Java基础-day01-基础题

1. 简述java语言,具有哪些特性? (1).java语言是简单的 java语言是和c++语言类似的,其次java中丢弃了c++中一些难理解的特性,比如运算符...

29340
来自专栏xingoo, 一个梦想做发明家的程序员

const指南

基本词义  意思就就是说利用const进行修饰的变量的值在程序的任意位置将不能再被修改,就如同常数一样使用!  使用方法 const int a=1;//这里定...

204100
来自专栏小白的技术客栈

Python内置数据结构之字典

今天给大家讲解Python内置数据结构:字典。字典的内容比较多,今天只是简单地介绍一下,明天会继续补充字典相关的内容。 关于Windows的环境安装及配置,小白...

32440
来自专栏前端说吧

【消灭代办】第一周 - 敏感词判断

  一堆字符串组成的数组,给你一个字符串,让你去查找这个字符串是否在这个数组当中?

13110
来自专栏北京马哥教育

Python程序员最常犯的十个错误,看完你自己都笑了

本文由马哥教育Python自动化实战班4期学员推荐,转载自简书,作者为EarlGrey,内容略经小编改编和加工,观点跟作者无关,最后感谢作者的辛苦贡献与付出。 ...

29040
来自专栏大数据

在Python中什么时候用Yield什么时候用Return

许多Python开发人员在代码中使用yield,而不考虑他们是否真的需要。这篇文章解释了你什么时候应该使用它。

39900
来自专栏微信公众号:Java团长

Java异常处理和设计

在程序设计中,进行异常处理是非常关键和重要的一部分。一个程序的异常处理框架的好坏直接影响到整个项目的代码质量以及后期维护成本和难度。试想一下,如果一个项目从头到...

15630
来自专栏前端下午茶

JS 利用高阶函数实现函数缓存(备忘模式)

高阶函数就是那种输入参数里面有一个或者多个函数,输出也是函数的函数,这个在js里面主要是利用闭包实现的,最简单的就是经常看到的在一个函数内部输出另一个函数,比如

82230
来自专栏C语言及其他语言

Python程序员最常犯的十个错误

来源:编程派 ? 不管是在学习还是工作过程中,人都会犯错。虽然Python的语法简单、灵活,但也一样存在一些不小的坑,一不小心,初学者和资深Python程序...

33670

扫码关注云+社区

领取腾讯云代金券