通用程序设计

最小化local variables的范围
  • 最强有力的方式是:在该local variables第一次使用的地方declare it
  • 优先使用for loops, 而不是while loops
  • keep methods small and focused,一个方法只执行一个activity
for 循环和while循环相比,有哪些好处?
  • 能最小化local variables的范围
for (Iterator<Element> i = c.iterator(); i.hasNext(); ) {
    Element e = i.next();
    ... // Do something with e and i
}
...

// Compile-time error - cannot find symbol i
for (Iterator<Element> i2 = c2.iterator(); i.hasNext(); ) {
    Element e2 = i2.next();
    ... // Do something with e2 and i2
} 
Iterator<Element> i = c.iterator();
while (i.hasNext()) {
    doSomething(i.next());
}
...
Iterator<Element> i2 = c2.iterator();
while (i.hasNext()) {             // BUG!
    doSomethingElse(i2.next());
}
  • for循环更简洁,能增强可读性
for (int i = 0, n = expensiveComputation(); i < n; i++) {
    ... // Do something with i;
}
优先使用for-each loops,而不是传统的for loops
  • 在对collections and arrays进行迭代时,优先使用for-each loops
  • 如果想使用for-each loops迭代any object,只需要该object实现Iterable接口
懂得使用the libraries
  • 不要自造轮子,代价太大
// Common but deeply flawed!
static Random rnd = new Random();
static int random(int n) {
    return Math.abs(rnd.nextInt()) % n;
}
// just use
Random.nextInt(int)
// for Java 7
use ThreadLocalRandom, not Random
// for fork join pools and parallel streams
use SplittableRandom 
  • 通过使用standard library,你可以使用写这些库的专家的知识以及已经使用过的这些库的那些人的经验
  • 通过使用standard library,如果有flaws被发现,那么它们会在 the next release 中fixed掉
  • 通过使用standard library,你可以专注于自己的应用,而不用关心某些不太紧要的细节
  • standard library,随着时间的推移,性能会越来越高,而且不需要你维护
  • standard library,随着时间的推移,功能越来越丰富,健全
  • 通过使用standard library,你的代码会更易读,可维护性更强,因为你的代码属于mainstream
  • 为了便利的使用standard library,你需要关注 every major release,不断学习这些新添加的特性
  • 每一个程序员都应该熟悉 java.lang, java.util, java.io, 以及它们的 subpackages
  • 如果Java platform libraries 不能满足你,接下来应该先去查找高质量的第三方库,比如:Guava,如果你实在找不到满足你需要的库,再去自己编码实现它
如果需要exact answer,请不要使用float 和 double类型
  • float 和 double类型尤其不适合货币计算
  • 应该使用BigDecimal, int, long进行货币计算
  • 如果数量级不超过9位十进制数,则用int;如果不超过18位,则用long;如果超过18位或者含有小数,则用BigDecimal
说说Java的两种类型系统
  • primitives 和 reference types
  • primitives 包括 比如 int, double, and boolean;reference types 包括比如String 和 List
  • primitive type对应的reference type称为boxed primitive
说说primitives 和 boxed primitives的区别
  • primitives仅仅拥有values,boxed primitives 除此以外,还拥有不同于values的identities
  • boxed primitives有一个nonfunctional value,即null
  • primitives 更省时间,更省空间
  • 对boxed primitives,使用== operator,一般总是错误的,因为该操作符会比较identities,最后造成不期望的结果
// error! 由于==符号,(new Integer(42), new Integer(42))比较,会返回1
Comparator<Integer> naturalOrder =
    (i, j) -> (i < j) ? -1 : (i == j ? 0 : 1);
//right way
Comparator<Integer> naturalOrder = (iBoxed, jBoxed) -> {
“int i = iBoxed, j = jBoxed; // Auto-unboxing
    return i < j ? -1 : (i == j ? 0 : 1);
};
  • 在一个操作中,混合使用 primitives 和 boxed primitives,会使boxed primitive 自动拆箱,下面的代码会报NullPointerException的错误
public class Unbelievable {
    static Integer i;

    public static void main(String[] args) {
        if (i == 42)  //NullPointerException
            System.out.println("Unbelievable");
    }
}
public static void main(String[] args) {
    Long sum = 0L;
    for (long i = 0; i < Integer.MAX_VALUE; i++) {
        sum += i; //重复的拆箱和装箱,影响性能
    }
    System.out.println(sum);
}
什么时候必须使用boxed primitives
  • 充当collections中的元素
  • 激活反射方法
当其他类型更适合时,请不要使用strings类型
  • Strings不适合替代其他的value types
  • Strings不适合替代enum types
  • Strings不适合替代aggregate types(复合类型),应该提供一个私有的静态成员类
// Inappropriate use of string as aggregate type
String compoundKey = className + "#" + i.next(); 
小心String concatenation operator (+)的性能
  • 不要使用String 的+操作符来 combine more than a few strings,应该使用StringBuilder 的 append 方法
使用interfaces来引用objects
  • 如果有适合的接口存在,则 parameters, return values, variables, 以及 fields 都应该由interface types宣称
// Good - uses interface as type
Set<Son> sonSet = new LinkedHashSet<>();
// Bad - uses class as type!
LinkedHashSet<Son> sonSet = new LinkedHashSet<>();
有哪些情况适合使用具体类来引用objects
  • value classes, 比如 String 和 BigInteger
  • class-based framework,比如 大多数 java.io 类,如OutputStream
  • 该interface中没有提供某些方法,比如PriorityQueue中有个comparator方法,该方法没有在Queue interface中提供
优先使用 interfaces 而不是 reflection
  • java.lang.reflect 提供获取一个类的 Constructor, Method, Field,你可以使用它们 construct instances, invoke methods, 以及 access fields of the class
reflection的缺点
  • 失去了编译期类型检查的所有好处
  • 执行reflective access的代码笨拙又冗长
  • 性能差
怎么结合使用 interfaces 和 reflection
  • 仅仅使用反射 create instances,使用interfaces和superclass access them
  • 小例子
// Reflective instantiation with interface access
public static void main(String[] args) {
    // Translate the class name into a Class object
    Class<? extends Set<String>> cl = null;
    try {
        cl = (Class<? extends  Set<String>>)  // Unchecked cast!
                Class.forName(args[0]);
    } catch (ClassNotFoundException e) {
        fatalError("Class not found.");
    }
    // Get the constructor
    Constructor<? extends Set<String>> cons = null;
    try {
        cons = cl.getDeclaredConstructor();
    } catch (NoSuchMethodException e) {
        fatalError("No parameterless constructor");
    }
    // Instantiate the set
    Set<String> s = null;
    try {
        s = cons.newInstance();
    } catch (IllegalAccessException e) {
        fatalError("Constructor not accessible");
    } catch (InstantiationException e) {
        fatalError("Class not instantiable.");
    } catch (InvocationTargetException e) {
        fatalError("Constructor threw " + e.getCause());
    } catch (ClassCastException e) {
        fatalError("Class doesn't implement Set");
    }
    // Exercise the set
    s.addAll(Arrays.asList(args).subList(1, args.length));
    System.out.println(s);
}
private static void fatalError(String msg) {
    System.err.println(msg);
    System.exit(1);
}
使用native methods,要谨慎
  • native methods,即使用C或C++写的方法,Java 允许使用Java Native Interface (JNI) 来调用native methods
  • native methods 使用的胶水语言难以阅读和书写,难以debug,可移植性差,易造成内存污染,不利于GC。所以应该尽可能少的使用native methods
代码优化,要谨慎
  • 过早的优化是万恶之源
  • 不要为了性能而牺牲好的程序结构
  • 致力于写好的代码,而不是快的代码
  • Good API design 会带来 Good performance
  • programs spend 90 percent of their time in 10 percent of their code,必要时,可以借助profiler或jmh来搜寻那10%
坚持公认的命名惯例
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351