Effective java笔记(七),通用程序设计
45、将局部变量的作用域最小化
将局部变量的作用域最小化,可以增强代码的可读性和可维护性,并降低出错的可能性。
Java允许在任何可以出现语句的地方声明变量(C语言中局部变量要在代码块开头声明),要使局部变量的作用域最小化,最好的方法是在第一次使用它的地方声明。局部变量的作用域从它被声明的点开始扩展,一直到外围块的结束处。
如果在循环终止之后不再需要循环变量的内容,for循环就优于while循环。for循环中变量的作用域范围更小,可以避免一些复制、粘贴错误,并且for循环更简短、可读性更强。如:
for(Element e : c) {
doSomething(e);
}
Iterator<Element> i = c.iterator();
while(i.hasNext()) {
doSomething(i.next());
}
若循环测试中涉及方法调用,并且每次迭代都返回相同的结果。应使用下面的方法避免每次迭代中执行冗余计算。
for(int i=0, n=upper(); i<n; i++) {
doSomething(i);
}
46、for-each循环优先于传统的for循环
for-each循环,通过完全隐藏迭代器或索引变量,避免了调用时混乱和出错的可能。
例如:打印一对骰子的所有可能情况
enum Face { ONE, TWO, THREE, FOUR, FIVE, SIX }
....
Collection<Face> faces = Arrays.asList(Face.values());
for(Iterator<Face> i=faces.iterator(); i.hasNext(); ) {
for(Iterator<Face> j=faces.iterator(); j.hasNext(); ) {
System.out.println(i.next() + " " + j.next());
}
}
这个程序不会抛出异常,而是打印6种组合(ONE ONE 到 SIX SIX),而不是36种组合。要修正这个错误,必须在外部循环的作用域中添加一个临时变量来保存外部元素。如:
for(Iterator<Face> i=faces.iterator(); i.hasNext(); ) {
Face temp = i.next();
for(Iterator<Face> j=faces.iterator(); j.hasNext(); ) {
System.out.println(temp + " " + j.next());
}
}
若使用嵌套的for-each循环,这个错误就可以完全避免。如:
for(Face face1 : faces) {
for(Face face2 : faces) {
System.out.println(face1 + " " + face2);
}
}
for-each循环不仅可以遍历集合和数组,还可以遍历任何实现了Iterable接口的对象。但有三种常见的情况无法使用for-each循环:
- 过滤,遍历集合或数组并删除选定的元素,需要使用显式的迭代器。
- 转换,遍历集合或数组并替换选定的元素。
- 平行迭代,需要平行的遍历多个集合或数组(骰子打印6种组合的情况)
47、了解和使用类库
使用标准类库的好处:
- 可以充分利用他人的使用经验
- 不必浪费时间在一些与工作不相关的问题上
- 性能会随着时间的推移而不断提高
- 可以使自己的代码融入主流
java程序员应该熟练掌握和使用java.lang,java.util,java.io包中的内容。
一句话,不要重新发明轮子。
48、如果需要精确的答案,请避免使用float和double
float和double类型在执行二进制浮点运算时,不能得到完全精确的结果,它们不应该被用于需要精确结果的场合。例如:0.4 + 0.2
输出结果为0.6000000000000001
。 原因可以参考这篇博客 代码之谜(五)- 浮点数(谁偷了你的精度?)
解决这个问题的办法是使用BigDecimal进行计算,或转化为int、long类型(自己处理小数点)。如:
BigDecimal result = new BigDecimal("0.2").add(new BigDecimal("0.4"));
对于任何需要精确答案的计算任务,不要使用float或double。
49、基本类型优先于装箱基本类型
java中每个基本类型(int、double)都有一个对应的引用类型(Integer、Double),称作装箱基本类型。
基本类型和装箱基本类型的区别:
- 基本类型是值,而装箱基本类型是对象
- 装箱基本类型有非功能值null
- 基本类型更节省时间和空间
编程时应注意下面几种常见的错误:
//1. ==操作
public int compare(Integer first, Integer second) {
return first < second ? -1 :(first == second ? 0 : 1); //compare(42,42)结果返回1
}
//2. null
Integer i;
if(i == 42) { //报空指针异常,i初始值为null
...
}
//3. 无意识的装箱
Long sum = 0L; //无意识的装箱,性能严重下降
for(long i=0; i<Integer.MAX_VALUE; i++ ) {
sum += i;
}
必须使用装箱基本类型的情况:
- 泛型中的参数化类型
- 进行反射的方法调用时
50、如果其他类型更适合,则尽量避免使用字符串
如果可以使用更加合适的数据类型,或者可以编写更适当的数据类型,就应该避免用字符串来表示对象。若使用不当,字符串会比其他类型更笨拙、速度更慢、也更容易出错。不要用字符串来代替基本类型、枚举类型和聚集类型。
51、当心字符串连接的性能
不要使用字符串连接操作符+
来合并多个字符串,应该使用StringBuilder的append方法。第一种方法的时间复杂度为O(n2),第二种方法的时间复杂度为O(n)。
52、通过接口引用对象
若有合适的接口类型存在,那么对于参数、返回值、变量和域,应该使用接口而不是类进行声明。这将使程序更加灵活。
53、接口优先于反射机制
反射机制提供了访问编译时未知的类的能力,对于复杂的系统编程任务,它是必要的。但它也有一些缺点,比如:丧失了编译时类型的检查、代码冗长、性能较低等。若有可能应该仅仅使用反射机制来实例化对象,而访问对象则使用编译时已知的接口或超类。
54、谨慎的使用本地方法
Java Native Interface(JNI) 允许java程序可以调用本地方法—native method,本地程序设计语言(如C或C++)编写的特殊方法。
本地方法的主要用途有:访问遗留代码库、提高系统性能、访问注册表和文件锁等。本地方法是不安全的、难于调试并且不可自由移植。应该尽量少使用或不使用本地方法。
55、谨慎的进行优化
- 不要因为性能而牺牲合理的结构。要努力编写好的程序而不是快的程序。
- 不要进行优化,特别是不成熟的优化。
不要费力去编写快速的程序——应该努力编写好的程序。在设计API、线路层协议和永久数据格式的时候,一定要考虑性能的因素。若系统不够快,可使用性能剖析工具找到问题的根源,并设法优化相关的部分。再多的低层优化也无法弥补算法的选择不当,所以选择一个好的算法是性能优化的根本。
56、遵循普通接受的命名惯例
java的命名惯例包含在《Java编程规范》(the java language specification)中。
泛型参数类型:T表示任意的类型,E表示集合的元素类型,K和V表示映射的键和值类型,X表示异常。任何类型的序列使用T1、T2、T3。
Effective java笔记(七),通用程序设计的更多相关文章
- [Effective Java]第八章 通用程序设计
声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...
- Effective java笔记5--通用程序设计
一.将局部变量的作用域最小化 本条目与前面(使类和成员的可访问能力最小化)本质上是类似的.将局部变量的作用域最小化,可以增加代码的可读性和可维护性,并降低出错的可能性. 使一个局部变量的作用 ...
- Effective Java笔记一 创建和销毁对象
Effective Java笔记一 创建和销毁对象 第1条 考虑用静态工厂方法代替构造器 第2条 遇到多个构造器参数时要考虑用构建器 第3条 用私有构造器或者枚举类型强化Singleton属性 第4条 ...
- Effective java笔记(二),所有对象的通用方法
Object类的所有非final方法(equals.hashCode.toString.clone.finalize)都要遵守通用约定(general contract),否则其它依赖于这些约定的类( ...
- effective java笔记之单例模式与序列化
单例模式:"一个类有且仅有一个实例,并且自行实例化向整个系统提供." 单例模式实现方式有多种,例如懒汉模式(等用到时候再实例化),饿汉模式(类加载时就实例化)等,这里用饿汉模式方法 ...
- effective java笔记之java服务提供者框架
博主是一名苦逼的大四实习生,现在java从业人员越来越多,面对的竞争越来越大,还没走出校园,就TM可能面临失业,而且对那些增删改查的业务毫无兴趣,于是决定提升自己,在实习期间的时间还是很充裕的,期间自 ...
- Effective Java 读书笔记之七 通用程序设计
一.将局部变量的作用域最小化 1.在第一次使用变量的地方声明 2.几乎每个变量的声明都应该包含一个初始化表达式:try-catch语句是一个例外 3.使方法小而集中是一个好的策略 二.for-each ...
- Effective java笔记6--异常
充分发挥异常的优点,可以提高一个程序的可读性.可靠性和可维护性.如果使用不当的话,它们也会带来负面影响. 一.只针对不正常的条件才使用异常 先看一段代码: //Horrible abuse of ex ...
- [Effective Java]第七章 方法
声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...
随机推荐
- Microsoft开源跨平台的序列化库——Bond
上个月Microsoft开源了Bond,一个跨平台的模式化数据处理框架.Bond支持跨语言的序列化/反序列化,支持强大的泛型机制能够对数据进行有效地处理.该框架在Microsoft公司内部的高扩展服务 ...
- 玩转Asp.net MVC 的八个扩展点
MVC模型以低耦合.可重用.可维护性高等众多优点已逐渐代替了WebForm模型.能够灵活使用MVC提供的扩展点可以达到事半功倍的效果,另一方面Asp.net MVC优秀的设计和高质量的代码也值得我们去 ...
- Hibernate的三种状态及对象生命周期
理解Hibernate的三种状态,更利于理解Hibernate的运行机制,这些可以让你在开发中对疑点问题的定位产生关键性的帮助. 三种状态 临时状态(Transient):在通过new关键字, ...
- Ubuntu安装Python2.7,nodejs
安装Python2.7 sudo add-apt-repository ppa:fkrull/deadsnakes-python2.7sudo apt-get update sudo apt-get ...
- 《Entity Framework 6 Recipes》中文翻译系列 (31) ------ 第六章 继承与建模高级应用之自引用关联
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 6-4 使用TPH建模自引用关系 问题 你有一张自引用的表,它代表数据库上不同类型 ...
- ASP.NET Web API自身对CORS的支持: CORS授权检验的实施
通过<EnableCorsAttribute特性背后的故事>我们知道:由CorsPolicyProvider提供的CorsPolicy表示目标Action采用的资源授权策略,ASP.NET ...
- vmware 虚拟机通信拿不到 inet addr 的解决办法
我在虚拟机上安装完红帽之后,使用ifconfig命令来看网卡的IP,但是,输入命令之后,eht0里面只有 inet6 addr 而没有 inet addr,不多说,上图. 解决办法如下:打开 虚拟机设 ...
- Spring声明式事务管理
一.Spring 的声明式事务管理概述 1.Spring 的声明式事务管理在底层是建立在 AOP 的基础之上的.其本质是对方法前后进行拦截,然后在目标方法开始之前创建或者加入一个事务,在执行完目标方法 ...
- 轻量级前端MVVM框架avalon - 整体架构
官网提供架构图 单看这个图呢,还木有说明,感觉有点蛋疼,作者的抽象度太高了,还好在前面已经大概分析过了执行流程 如图 左边是View视图,我们就理解html结构,换句话就是说用户能看到的界面,渲染页面 ...
- OpenCascade Draw Test Harness
OpenCascade Draw Test Harness eryar@163.com Abstract. Draw is a command interpreter based on Tcl/Tk ...