`
michael8335
  • 浏览: 187583 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

由一段代码发表一点想法

阅读更多
今天在公司发现了一段很怪异的代码,为此还跟公司员工争执了一下,但由于自身是新员工,我只有无奈的屈服了,心里确实不爽,在这里发表一下自己的看法。先看一下代码,代码已经我已经简化了,只有两个类,一个是action层,另一个是Service层,具体如下:
Action:
package com.yf.test;

import java.util.List;

public class Action {
	public void excute(){
		List list=null;
		list=Service.doSomething();
		if(list.isEmpty()){
			/*
			 *此处省略其他操作
			 */
		}
	}
}

Service
package com.yf.test;

import java.util.ArrayList;
import java.util.List;

public class Service {
	public static List doSomething() {
		List list = new ArrayList();
		if(1==2){
			/*
			 * 此处省略对list的其他操作
			 *  list=.....
			 */			
		}
		return list;
	}
}

当我看到if(list.isEmpty()){时,发现潜在产生空指针异常的可能,然后就给同事说了一下,这里应该先判断List是否为null,然后他让我看Service层,说Service不会返回null,我看了一下,确实不会返回为null,但是关于这段代码,我觉得写的实在太烂,为什么烂,主要有一下几个原因:
1、在Service层,每次调用doSomething方法时,都实例化一个List,虚拟机都会在堆中为这个list开辟内存,这无疑实在浪费内存和虚拟机的,而且这个list只有在if条件成立时,才需要,如果if不成立,虚拟机还得在方法调用结束后,回收这块内存,这难道不是没事找事吗??
2、在Action层,action不对返回的list做非null判断,这也是一种很恶心的做法,首先,根据面向对象的封装性,Service层中的实现对Action而言,应该是不可见的,Action层应该对其返回值的可能情况做判断,即list!=null必须在Action做,如果后续Service层单独抽出,以API提供Jar包的形式,即我们无法知道里面的具体细节,这时,Action层还得做非空判断。因此,本人觉得,这段代码应该做如下重构
Action:
package com.yf.test;

import java.util.List;

public class Action {
	public void excute(){
		List list=null;
		list=Service.doSomething();
		if(list!=null&&list.isEmpty()){
			/*
			 *此处省略其他操作
			 */
		}
	}
}

Service
package com.yf.test;

import java.util.ArrayList;
import java.util.List;

public class Service {
	public static List doSomething() {
		List list=null;
		if(1==2){
			list= new ArrayList();
			/*
			 * 此处省略对list的其他操作
			 *  list=.....
			 */			
		}
		return list;
	}
}

或许并不是每个人都认同我这种做法,不过我个人觉得这样比较合理,软件设计的时候要讲究层次,各层应该干得事情,就应该在所在层做好,而不是有其下层来保证,这种强依赖下层保证是一种很恶心的做法,如果后续下层代码逻辑变更,还得去上层看看对其的影响,这就很无耻了!!!
21
13
分享到:
评论
38 楼 hlylove 2012-10-15  
syx278250658 写道
final List<String> someList = Service.doSomething();    
if(CollectionUtils.isEmpty(someList)){    
   .  
   .  
   . 


不管service如何,action这样判断是应该的


我也是这样的
37 楼 iceblooded 2012-10-15  
看下面的评论 学习了
36 楼 gwei 2012-10-15  
拜读了,个人认为像这种返回结果集对象的服务方法最好不要返回null,返回null无意义。至于什么内存开销太较真了吧,这算什么,在远程的情况下说不定都不在一个环境中。程序健壮性应该得到保证,Action还是应该判断NP吧,至于方式还是按项目组规范。拙见拙见。
35 楼 beiyeren 2012-10-15  
骨之灵魂 写道
flashing 写道
sqtds 写道
opnmzxcvb 写道
kidneyball 写道
1. 容器类型的底层方法应该尽量返回空集,避免返回null。从而避免顶层方法做大量的null判断。具体可参考《Effective Java》。

2. 一个以容器为返回类型的方法,返回空集通常是不常见的异常分支,为一个不常见的分支考虑类创建和垃圾回收层面上的性能问题而导致调用者做大量的额外NP判断并不值得。这类优化应该在有profiling数据支持的情况下再做。

3. “具体细节不可见”的问题,写点Javadoc就能解决。调用者需要参考接口文档知道返回值的具体情况,是很正常的事情。后续维护时要保证接口协议不变,也是很正常的事情。

4. 某些团队都会形成共识,只要是返回容器类型的方法就不返回null。只要大家都遵守这个协议,不会出问题而且可以免去大量无谓的NP判断。博主作为一个新人,应该先了解团队中的开发规范,而不是和现有规则较劲。

说的对,  

看看effective java

yeah,我也想说 effective java

看看effective java


分析的很到位
34 楼 congjl2002 2012-10-15  
michael8335 写道
young_suse 写道
楼主的担心也很合理,每次在方法开头用List list = new ArrayList();确实可能造成不必要的开销。
但从代码规范的角度来讲,声明为集合(List或数组)返回类型的函数,通常不要返回null,宁可返回size为0的空集合。
要达到既减少内存开销,又规范代码,应该使用List list = Collections.emptyList()来取代List list=null;并且按照kidneyball说的,用javadoc来说明,当条件不满足时,返回size为0的List,而不会返回null。

由于我编码时间不长,所以估计是我自身理解确实有问题,不过通过这篇文章,获得了一些知识,看来大家的意见还是倾向于不返回null,那就按照大家的来吧,反正我只是就这段代码发表一下自己的看法,谢谢大家的意见,后期会买一本《Effective Java》,在设计方面继续积累

你们员工可以坚持不实用Null说明他们水平不错,你有机会多向他们学习,不要总想去质疑他们的代码,毕竟在编程里有这样一个规定“习惯大于配置”,编程时间短,有很多行规你还不清楚,因此要虚心,但也不是就全盘接受,你有这个分析问题的想法很好,只是要注意方法
33 楼 lianglaiyang 2012-10-15  
forcer521 写道
jorneyR 写道
在职场中,不要轻易去表现出质疑别人的东西,就算那东西再差,自己不赞同,也尽量不要表现出来,人心难测,除非你有决策权的时候。


很有经验啊,呵呵。
我们程序员很多时候就是缺乏这种思维能力。
很多时候,做的对并不能得到大家的认可的。

那是自己一厢情愿的认为自己对,其实最终可能会发现自己错的离谱,是那么可笑。。。
32 楼 骨之灵魂 2012-10-15  
flashing 写道
sqtds 写道
opnmzxcvb 写道
kidneyball 写道
1. 容器类型的底层方法应该尽量返回空集,避免返回null。从而避免顶层方法做大量的null判断。具体可参考《Effective Java》。

2. 一个以容器为返回类型的方法,返回空集通常是不常见的异常分支,为一个不常见的分支考虑类创建和垃圾回收层面上的性能问题而导致调用者做大量的额外NP判断并不值得。这类优化应该在有profiling数据支持的情况下再做。

3. “具体细节不可见”的问题,写点Javadoc就能解决。调用者需要参考接口文档知道返回值的具体情况,是很正常的事情。后续维护时要保证接口协议不变,也是很正常的事情。

4. 某些团队都会形成共识,只要是返回容器类型的方法就不返回null。只要大家都遵守这个协议,不会出问题而且可以免去大量无谓的NP判断。博主作为一个新人,应该先了解团队中的开发规范,而不是和现有规则较劲。

说的对,  

看看effective java

yeah,我也想说 effective java

看看effective java
31 楼 lianglaiyang 2012-10-15  
参考大部分开源框架API,返回集合时都不会返回null这一种情况!当今什么时代了,如果一般应用程序的话那一点内存根本可以忽略不计
30 楼 fwmlove 2012-10-15  
我怎么觉得Service层的重构跟没重构一样,你就算NUll不NULL也好,这个List在Service层里面又不是成员变量,只要是同个线程内,你每次调用这个方法,原来的那个就会可以被回收啦。不知道我理解有没有问题。
29 楼 vvggsky 2012-10-15  
防御式编程 未尝不可
28 楼 zuoshu 2012-10-15  
skzr.org 写道
尽量不返回null——说实在的就算返回了null,你又能做什么了。
——你只能得到一大堆的if (xxx == null) ...,如果直接返回empty集合,代码更加简洁易懂

绝大多数情况(99%)empty集合更加合适。
过早的优化——万恶之源啊。。。

既然意识到只能保证99%,为何不优化到100%,程序的健壮性何在!
27 楼 kyfxbl 2012-10-15  
我只能说楼主乐于思考的精神还是很好的,应该保持
26 楼 skzr.org 2012-10-15  
尽量不返回null——说实在的就算返回了null,你又能做什么了。
——你只能得到一大堆的if (xxx == null) ...,如果直接返回empty集合,代码更加简洁易懂

绝大多数情况(99%)empty集合更加合适。
过早的优化——万恶之源啊。。。
25 楼 syx278250658 2012-10-15  
final List<String> someList = Service.doSomething();    
if(CollectionUtils.isEmpty(someList)){    
   .  
   .  
   . 


不管service如何,action这样判断是应该的

24 楼 jinnianshilongnian 2012-10-15  
rensanning 写道
jinnianshilongnian 写道
当然还有一种叫做契约式编程:

在service返回时 加:
Assert.assertNotNull(list);  来强制约束返回值不为null 

契约的缺点就是靠程序员意识,写不写很难控制

spring对契约式编程的支持:
http://jinnianshilongnian.iteye.com/blog/1495594

产品代码中一般很少见JDK Assertions,多见于调试或者测试。运行时默认是关闭状态,即使开启,它所抛出的AssertionError异常扩展自Error而不是Exception,会扰乱系统的异常处理。普遍的框架中对采用IllegalArgumentException,比如Spring的org.springframework.util.Assert

不是java自带的assert断言  而是自己扩展的一套 正如你说的如Spring的
23 楼 rensanning 2012-10-15  
jinnianshilongnian 写道
当然还有一种叫做契约式编程:

在service返回时 加:
Assert.assertNotNull(list);  来强制约束返回值不为null 

契约的缺点就是靠程序员意识,写不写很难控制

spring对契约式编程的支持:
http://jinnianshilongnian.iteye.com/blog/1495594

产品代码中一般很少见JDK Assertions,多见于调试或者测试。运行时默认是关闭状态,即使开启,它所抛出的AssertionError异常扩展自Error而不是Exception,会扰乱系统的异常处理。普遍的框架中对采用IllegalArgumentException,比如Spring的org.springframework.util.Assert
22 楼 jinnianshilongnian 2012-10-15  
当然还有一种叫做契约式编程:

在service返回时 加:
Assert.assertNotNull(list);  来强制约束返回值不为null 

契约的缺点就是靠程序员意识,写不写很难控制

spring对契约式编程的支持:
http://jinnianshilongnian.iteye.com/blog/1495594
21 楼 jinnianshilongnian 2012-10-15  
1、在Service层,每次调用doSomething方法时,都实例化一个List,虚拟机都会在堆中为这个list开辟内存,这无疑实在浪费内存和虚拟机的,而且这个list只有在if条件成立时,才需要,如果if不成立,虚拟机还得在方法调用结束后,回收这块内存,这难道不是没事找事吗??

如果服务层 调用 持久层框架(如hibernate) 即使是空 也是返回一个集合的;
                 自己写业务  如果为null  可以返回Collections.EMPTY_LIST(单例 且不可变的) 
建议:不要过早优化 先保证代码可读和可维护



2、在Action层,action不对返回的list做非null判断,这也是一种很恶心的做法,首先,根据面向对象的封装性,Service层中的实现对Action而言,应该是不可见的,Action层应该对其返回值的可能情况做判断,即list!=null必须在Action做,如果后续Service层单独抽出,以API提供Jar包的形式,即我们无法知道里面的具体细节,这时,Action层还得做非空判断。因此,本人觉得,这段代码应该做如下重构

  可以通过如ListUtils.isEmpty(list) 来判断  然后把判断null 和 空的逻辑封装起来;
  另外:如StringUtils等这种工具应该是自己工具箱必备的;

  建议:掌握常用工具箱工具,减少做重复的事情。

20 楼 lazy_ 2012-10-15  
package com.yf.test;

import java.util.ArrayList;
import java.util.List;

public class Service {
	public static List doSomething() {
		List list=null;
		if(xxx){
			list= new ArrayList();
			/*
			 * 此处省略对list的其他操作
			 *  list=.....
			 */			
		}
                if(null == list){
                      return new ArrayList(0);
                }
		return list;
	}
}

19 楼 lazy_ 2012-10-15  
楼主的问题我也在烦恼。从实践的角度看,NULL好容易引起错误。也许就是这个原因,freemarker没有NULL。

1 我编码的时候,尽量不返回NULL。可以return一个new ArrayList(0)或者new HashMap(0),或者"",这就不用担心所谓内存消耗的问题。
2 我用别人写的类,或者好久之前的类,保守起见我肯定会做NULL判断。

引用
容器类型的底层方法应该尽量返回空集,避免返回null。从而避免顶层方法做大量的null判断。具体可参考《Effective Java》。

我没看过Effective Java,但是这句我非常赞成啊,能少写代码就少写代码,简单是王道。其实封装一下
class Util{
  //not null
  public static boolean NN(list){
       if(null != list && !list.empty() ){
              return true;
       }
       return false;
  }
}
也能达到目的。当然这样虽然较少了业务代码,但是会导致公用代码变多了。

相关推荐

    软著申请:源代码文件模板,每页五十行,一共六十页

    本资源是一个基于Java语言的源代码文件模板,总共六十页,每页五十行,主要涉及到嵌入式校园网网络质量监测系统的开发。以下是相关知识点的总结: 1. Java语言基础:该源代码文件模板使用Java语言编写,涉及到Java...

    G代码解释程序

    G代码由一系列字母、数字和符号组成,每个代码或指令代表一个特定的动作,如移动刀具、设定速度、开始切割等。 C++ Builder是一款由Embarcadero Technologies开发的集成开发环境(IDE),它基于C++语言并提供了RAD...

    软著源代码整理工具

    标题中的“软著源代码整理工具”指的是一个专门用于整理软件著作权申请所需的源代码的工具。在IT行业中,软件著作权是保护开发者权益的重要法律手段,它要求申请者提交源代码以证明其原创性。源代码整理是这个过程中...

    软著代码整理工具,可以实现一键提取文件中的代码并且自动删去空行和注释,便于进行软著申请

    为了简化这一过程,"软著代码整理工具"应运而生,它能够一键提取代码、删除空行和注释,使代码更符合申请要求。 1. **一键提取文件中的代码**:此工具的核心功能之一是能够快速从各种类型的代码文件中提取出有效的...

    组织机构代码生成器

    在中国,组织机构代码由9位数字组成,包括一个校验码,确保每个代码的唯一性。生成器会按照这些规则生成符合标准的代码。 2. **自动化**:该工具应具备自动化功能,自动填充必要的信息,如机构名称、注册地等,减少...

    JavaScript代码生成器

    JavaScript代码生成器就是一种能够帮助开发者快速生成JavaScript代码的工具,它可以极大提高开发效率,减少手动编写重复性代码的工作量。 JavaScript代码生成器通常具备以下功能: 1. 自动化模板:这些工具提供了...

    android 经典代码例子

    在Android开发领域,经典代码例子是开发者学习和提升技能的重要资源。这些例子涵盖了各种关键功能和组件的实现,有助于深入理解Android应用的工作原理。在这个压缩包中,我们可能找到了多个有关Android编程的示例...

    editplus如何实现整段缩进

    EditPlus是一款功能强大的文本编辑器,尤其在编程领域广受欢迎,因为它提供了许多便捷的代码编辑功能,其中之一就是整段缩进。整段缩进在编程中扮演着至关重要的角色,它能够极大地提升代码的可读性和维护性。下面将...

    igh ethercat 台达电机控制代码

    本文将详细解析"igh ethercat 台达电机控制代码"中的核心概念和技术要点,以及如何利用这段代码来驱动符合DS402协议的台达电机。 EtherCAT 主站是EtherCAT网络中的核心部分,它负责协调整个网络的操作,向从站设备...

    美食网站源代码

    【美食网站源代码】是一个专为展示美食相关资讯和功能的网站开发的源代码集合,是互联网上搜集并整理的资源,供开发者下载和使用。这个源代码可能包含了一个完整的美食主题网站的所有必要组成部分,例如前端用户界面...

    编译原理中间代码生成实验报告——完整版

    在编译原理中,中间代码生成是编译过程的关键步骤之一,它位于词法分析、语法分析之后,目标代码生成之前。本实验报告主要围绕算术表达式的中间代码生成展开,旨在让学生深入理解算术表达式的语法分析原理,并能实现...

    恶意代码分析实战

    本书是一本内容全面的恶意代码分析技术指南,其内容兼顾理论,重在实践,从不同方面为读者讲解恶意代码分析的实用技术方法。, 本书分为21章,覆盖恶意代码行为、恶意代码静态分析方法、恶意代码动态分析方法、恶意...

    微信小程序游戏源代码

    7. **社交分享与邀请机制**:微信小程序的一大优势是社交功能,"跳一跳"的源代码可能会包含如何实现分享到朋友圈、微信群,以及邀请好友等功能,这是增强用户粘性和游戏传播的关键。 8. **性能优化**:源代码还会...

    vc++游戏(源代码)

    在本资源包中,我们收集了一系列使用VC++编写的经典游戏源代码,这些源代码提供了深入理解C++编程语言以及游戏开发基础的宝贵机会。通过分析和学习这些源码,开发者可以了解到如何利用C++来实现游戏逻辑、图形界面、...

    电子签章系统源代码

    "电子签章系统源代码" 电子签章系统源代码是指实现电子签章功能的源代码,它可以和WORD、EXCEL等办公软件集成,实现电子签章的功能。 1. 标题:电子签章系统源代码 电子签章系统源代码是指实现电子签章功能的源...

    8 位 CPU vhdl实现(含全部源代码)

    你可以写出一段17位的指令代码,并放入ROM区,该CPU即可自动运行出结果。压缩包里是源代码和我们当时的设计要求。本源代码的最后调试时在地址0--17是放入的斐波纳契数字(Fibonacci Numbers)指令。通过modelsim仿真...

    Android第二行代码(郭霖全书源代码)

    《Android第二行代码》是郭霖撰写的一本深入浅出的Android编程入门书籍,它旨在帮助初学者快速理解和掌握Android应用开发。这本书的全书源代码提供了丰富的实践示例,帮助读者通过动手实践来巩固理论知识。以下是...

    完整的matlab遗传算法代码

    遗传算法是一种基于生物进化原理的优化方法,广泛应用于解决复杂问题的全局寻优。MATLAB作为一款强大的数学计算...通过学习和实践这段代码,不仅可以掌握遗传算法的基本原理,还能提升在MATLAB中解决实际问题的能力。

    代码注释率统计工具下载

    代码注释是编程实践中至关重要的一个环节,它有助于提高代码的可读性和可维护性。注释率是指源代码中的注释行数占总代码行数的比例,通常用来衡量代码的文档质量。`linecount`是一款用于统计代码注释率的工具,其...

    用C语言实现的画线代码(画任意曲线)

    4. **代码实现**: 一个简单的例子是创建一个窗口,然后在窗口的Paint消息处理程序中绘制线条。首先,创建一个消息循环,注册窗口类,然后在窗口回调函数中处理WM_PAINT消息,调用GDI函数来画线。 5. **优化与扩展...

Global site tag (gtag.js) - Google Analytics