做者:里奥ii
连接:https://zhuanlan.zhihu.com/p/67595187
来源:知乎
著做权归做者全部。商业转载请联系做者得到受权,非商业转载请注明出处。
html
01java
Array转ArrayList算法
当须要把Array转成ArrayList的时候,开发人员常常这样作:数组
Arrays.asList会返回一个ArrayList,可是要特别注意,这个ArrayList是Arrays类的静态内部类,并非java.util.ArrayList类。java.util.Arrays.ArrayList类实现了set,get,contains方法,可是并无实现增长元素的方法(事实上是能够调用add方法,可是没有具体实现,仅仅抛出UnsupportedOperationException异常),所以它的大小也是固定不变的。为了建立一个真正的java.util.ArrayList,你应该这样作:安全
ArrayList的构造方法能够接收一个Collection类型。数据结构
而java.util.Arrays.ArrayList已经实现了该接口。
ide
02性能
判断一个数组是否包含某个值ui
开发人员常常这样作:设计
以上代码能够正常工做,可是没有必要将其转换成set集合,将一个List转成Set须要额外的时间,其实咱们能够简单的使用以下方法便可:
或者
第一种方法可读性更强。
03
在循环内部删除List中的一个元素
考虑以下代码,在迭代期间删除元素:
结果打印:[b, d]
在上面这个方法中有一系列的问题,当一个元素被删除的时候,list大小减少,而后原先索引指向了其它元素。因此若是你想在循环里经过索引来删除多个元素,将不会正确工做。
你也许知道使用迭代器是在循环里删除元素的正确方式,或许你也知道foreach循环跟迭代器很相似,但事实状况却不是这样,以下代码:
将抛出ConcurrentModificationException异常。
然而接下来的代码倒是OK的:
next方法须要在remove方法以前被调用,在foreach循环里,编译器会在删除元素操做后调用next方法,这致使了ConcurrentModificationException异常。
04
HashTable与HashMap
从算法的角度来说,HashTable是一种数据结构名称。可是在Java中,这种数据结构叫作HashMap。HashTable与HashMap的一个主要的区别是HashTable是同步的,因此,一般来讲,你会使用HashMap,而不是Hashtable。
5
05
使用集合原始类型(raw type)
在Java中,原始类型(raw type)和***通配符类型很容易让人混淆。举个Set的例子,Set是原始类型,而Set<?>是***通配符类型。
请看以下代码,add方法使用了一个原始类型的List做为入参:
运行以上代码将会抛出异常:
Exception in thread "main" java.lang.ClassCastException: java.lang.Integer cannot be cast to java.lang.String at ...
使用原始类型集合很是危险,由于它跳过了泛型类型检查,是不安全的。另外,Set, Set<?>, 和Set<Object>这三个有很大的不一样。
06
访问级别
开发人员常用public修饰类字段,虽然这很容易让别人直接经过引用获取该字段的值,但这是一个很差的设计。根据经验,应该尽量的下降成员属性的访问级别。
07
ArrayList和LinkedList
为何开发人员常用ArrayList和LinkedList,殊不知道他们之间的区别,由于它们看起来很像。然而它们之间有着巨大的性能差别。简单的说,若是有大量的增长删除操做而且没有不少的随机访问元素的操做,应该首选LinkedList。不然反之。
08
可变与不可变
不可变对象有不少优势,如简单、安全等。可是对于每一个不一样的值都须要一个单独的对象,太多的对象会引发大量垃圾回收,所以在选择可变与不可变的时候,须要有一个平衡。
一般,可变对象用于避免产生大量的中间对象,一个经典的例子是大量字符串的拼接。若是你使用一个不可变对象,将会立刻产生大量符合垃圾回收标准的对象,这浪费了CPU大量的时间和精力。使用可变对象是正确的解决方案(StringBuilder);
另外,在有些其它状况下也是须要使用可变对象。例如往一个方法传入一个可变对象,而后收集多种结果,而不须要写太多的语法。另外一个例子是排序和过滤:固然,你能够写一个方法来接收原始的集合,而且返回一个排好序的集合,可是那样对于大的集合就太浪费了。
09
父类和子类的构造方法
之因此出现这个编译错误,是由于父类的默认构造方法未定义。在Java中,若是一个类没有定义构造方法,编译器会默认插入一个无参数的构造方法;可是若是一个构造方法在父类中已定义,在这种状况,编译器是不会自动插入一个默认的无参构造方法,这正是以上demo的状况;
对于子类来讲,不论是无参构造方法仍是有参构造方法,都会默认调用父类的无参构造方法;当编译器尝试在子类中往这两个构造方法插入super方法时,由于父类没有一个默认的无参构造方法,因此编译器报错;
要修复这个错误,很简单:
一、在父类手动定义一个无参构造方法:
二、移除父类中自定义的构造方法
三、在子类中本身写上父类构造方法的调用;如super(value);
做者:风同样的码农
原文连接: Java开发人员最常犯的10个错误 - 风同样的码农 - 博客园