1、 为什么两个(1927年)时间相减得到一个奇怪的结果?
(3623个赞)
如果执行下面的程序,程序解析两个间隔1秒的日期字符串并比较:
01 | public static void main(String[] throws ParseException |
02 | SimpleDateFormat new SimpleDateFormat( "yyyy-MM-dd ); |
03 | String "1927-12-31 ; |
04 | String "1927-12-31 ; |
05 | Date |
06 | Date |
07 | long ld3 1000 ; |
08 | long ld4 1000 ; |
09 | System.out.println(ld4-ld3); |
10 | } |
输出是:
1 | 353 |
为什么 ld4-ld3 不是1(因为我希望这两个时间差是一秒),而是353?
如果将日期字符串各加一秒:
1 | String "1927-12-31 ; |
2 | String "1927-12-31 ; |
ld4-ld3 的结果是1.
1 | sun.util.calendar.ZoneInfo[ id = "Asia/Shanghai" , |
2 | offset=28800000,dstSavings=0, |
3 | useDaylight= false , |
4 | transitions=19, |
5 | lastRule=null] |
6 |
7 | Locale(Locale.getDefault()): |
解决方案
这是上海时区,在12月31日有一个变化。
查阅这个网址来了解上海在1927年时区变化的细节。基本上在1927年年底的午夜,始终会回拨5分52秒。所以“1927-12-31
23:54:08”实际上发生了两次,看起来Java解析了后一次的时间作为当地的日期和时间导致了差异。
Java一直是值传递。不幸的是,他们决定把指针叫做引用,因此新人总是被搞晕。因为这些引用也是通过值传递的。
3、一个关于Java += 操作符的问题
(2223赞)
直到今天我认为这个例子:
1 | i |
只是一个简写的:
1 | i |
但如果这样做:
1 | int i 5 ; |
2 | long j 8 ; |
然而 i = i + j; 没法编译,而 i += j; 就可以编译。
这意味着i += j;
实际上是i
的简写么?
= (type of i) (i + j)
解决方案
总有人问这类问题,JLS里有答案。参见 §15.26.2复合赋值运算符。摘录:
一个例子,引自 §15.26.2
1 | short x 3 ; |
2 | x 4.6 ; |
x的结果等于7,因为它等价于:
1 | short x 3 ; |
2 | x short )(x 4.6 ); |
换句话说,你的假设是正确的。
Java中 HashMap
和 Hashtable的不同是什么?
非多线程应用中使用哪个更有效率?
解决方案
Java 中 HashMap 和 HashTable 有几个不同点:
Hashtable
是同步的,然而HashMap
不是。
这使得HashMap更适合非多线程应用,因为非同步对象通常执行效率优于同步对象。- Hashtable 不允许 null 值和键。HashMap允许有一个 null 键和人一个 NULL 值。
- HashMap的一个子类是
LinkedHashMap。所以,如果想预知迭代顺序(默认的插入顺序),只需将HashMap转换成一个LinkedHashMap。用Hashtable就不会这么简单。
因为同步对你来说不是个问题,我推荐使用HashMap。如果同步成为问题,你可能还要看看ConcurrentHashMap。
如果你有一个 java.io.InputStream 对象,如处理这个对象并生成一个字符串?
假定我有一个 InputStream
对象,它包含文本数据,我希望将它转化成一个字符串(例如,这样我可以将流的内容写到一个log文件中)。
InputStream
转化成 String 最简单方法是什么?
解决方案
使用 Apache commons IOUtils库来拷贝InputStream到StringWriter是一种
不错的方式,类似这样:
1 | StringWriter new StringWriter(); |
2 | IOUtils.copy(inputStream, |
3 | String |
甚至
1 | // |
2 | // |
3 | String |
或者,如果不想混合Stream和Writer,可以使用 ByteArrayOutputStream。
在Swing中,密码字段有一个getPassword()(返回 char数组)方法而不是通常的getText()(返回String)方法。同样的,我遇到过一个建议,不要使用 String 来处理密码。
为什么String涉及到密码时,它就成了一个安全威胁?感觉使用char数组不太方便。
解决方案
String是不可变的。这意味着一旦创建了字符串,如果另一个进程可以进行内存转储,在GC发生前,(除了反射)没有方法可以清除字符串数据。
使用数组操作完之后,可以显式地清除数据:可以给数组赋任何值,密码也不会存在系统中,甚至垃圾回收之前也是如此。
所以,是的,这是一个安全问题 – 但是即使使用了char数组,仅仅缩小了了攻击者有机会获得密码的窗口,它值针对制定的攻击类型。
7、遍历HashMap的最佳方法
(1504个赞)
遍历HashMap中元素的最佳方法是什么?
解决方案
这样遍历entrySet:
1 | public static void printMap(Map |
2 | Iterator |
3 | while (it.hasNext()) |
4 | Map.Entry |
5 | System.out.println(pair.getKey() " + |
6 | it.remove(); // |
7 | } |
8 | } |
更多请查阅Map。
8、(如何)从数组创建ArrayList
我有一个数组,初始化如下:
1 | Element[] new Element( 1 ), new Element( 2 ), new Element( 3 )}; |
我希望将这个数组转化成一个ArrayList类的对象。
解决方案
1 | new ArrayList<Element>(Arrays.asList(array)) |
(1478个赞)
我有过一个面试,被问到如何产生一个Java内存泄露。不用说,我感到相当傻,甚至如何产生一个的线索都没有。
那么怎么才能产生一个内存泄露呢?
解决方案
在纯Java中,有一个很好的方式可以产生真正的内存泄露(通过执行代码使对象不可访问但仍存在于内存中):
- 应用产生一个长时间运行的线程(或者使用一个线程池加速泄露)。
- 线程通过一个(可选的自定义)类加载器加载一个类。
- 该类分配大内存(例如,new byte[1000000]),赋值给一个强引用存储在静态字段中,再将它自身的引用存储到ThreadLocal中。分配额外的内存是可选的(泄露类实例就够了),但是这样将加速泄露工作。
- 线程清除所有自定义类的或者类加载器载入的引用。
- 重复上面步骤。
这样是有效的,因为ThreadLocal持有对象的引用,对象持有类的引用,接着类持有类加载器的引用。反过来,类加载器持有所有已加载类的引用。这会使泄露变得更加严重,因为很多JVM实现的类和类加载都直接从持久带(permgen)分配内存,因而不会被GC回收。
(1422个赞)
标准的解决方式(Java1.7 之前)如下:
1 | import java.util.Random; |
2 |
3 | public static int randInt( int min, int max) |
4 |
5 | Random |
6 | int randomNum 1 ) |
7 |
8 | return randomNum; |
9 | } |
请查看相关的JavaDoc。在实践中,java.util.Random 类总是优于java.lang.Math.random()。