OBJECT
clone();
一、Java中有两种方式创建对象:1、使用new操作符创建对象;2、使用clone方法复制对象,并且clone要比创建一个新的对象的效率要高。
二、clone分为浅拷贝和深拷贝。浅拷贝指仅仅拷贝对象本身,如果对象中包含引用的对象,则不会被拷贝,深拷贝则会将引用对象一起拷贝。
三、要实现clone需要实现cloneable接口。如果没有实现接口,直接clone则会抛出CloneNotSupportedException异常。
hashcode();
如何判别在集合中是否已经存在该对象了?一般我们会用到equals()方法,,如果采用equals方法去逐一比较,效率必然是一个问题。此时,我们就需要用到hasncode()方法。实际上在HashMap的具体实现中会用一个table保存已经存进去的对象的hashcode值,如果table中没有该hashcode值,它就可以直接存进去,不用再进行任何比较了;如果存在该hashcode值, 就调用它的equals方法与新元素进行比较,相同的话就不存了,不相同就散列其它的地址,所以这里存在一个冲突解决的问题,这样一来实际调用equals方法的次数就大大降低了,说通俗一点:Java中的hashCode方法就是根据一定的规则将与对象相关的信息(比如对象的存储地址,对象的字段等)映射成一个数值,这个数值称作为散列值。
不同的对象可能会产生相同的hashcode值,如果两个对象的hashcode值不等,则必定是两个不同的对象,所以,一般来说可以用hashcode()判断两个对象不相等,如果要判断两个对象相等需要使用到equal()方法。
notify()
notify方法只唤醒一个等待(对象的)线程并使该线程开始执行。所以如果有多个线程等待一个对象,这个方法只会唤醒其中一个线程,选择哪个线程取决于操作系统对多线程管理的实现。notifyAll 会唤醒所有等待(对象的)线程,尽管哪一个线程将会第一个处理取决于操作系统的实现。如果当前情况下有多个线程需要被唤醒,推荐使用notifyAll 方法。比如在生产者-消费者里面的使用,每次都需要唤醒所有的消费者或是生产者,以判断程序是否可以继续往下执行
String
string.intern();
jdk7 版本对 intern 操作和常量池都做了一定的修改。主要包括2点:
1、将String常量池从Perm区移动到了Java Heap区
2、String#intern方法时,如果存在堆中的对象,会直接保存对象的引用,而不会重新创建对象。
在 Jdk6 以及以前的版本中,字符串的常量池是放在堆的Perm区的,Perm区是一个类静态的区域,主要存储一些加载类的信息,常量池,方法片段等内容,默认大小只有4m,一旦常量池中大量使用 intern 是会直接产生java.lang.OutOfMemoryError:PermGen space错误的
以下的程序作为理解:
String s =new String("1");
s.intern();
String s2 ="1";
System.out.println(s == s2);
String s3 =new String("1") +new String("1");
s3.intern();
String s4 ="11";
System.out.println(s3 == s4);
在jdk1.6以前结果为 false,false
在 jdk7 的版本中,false,true
AbstractStringBuilder
最大容量为Integer.MAX_VALUE-8
StringBuffer和StringBuilder
两个默认的初始容量均为为16,前者是线程安全的因为它的方法均添加了锁,
Thread
最小的权重是1,默认的权重是5,最大的权重为10
在thread类中,sleep,yield,join方法
(1)最简单的区别是,wait方法依赖于同步,而sleep方法可以直接调用。而更深层次的区别在于sleep方法只是暂时让出CPU的执行权,并不释放锁。而wait方法则需要释放锁。
(2)yield方法的作用是暂停当前线程,以便其他线程有机会执行,不过不能指定暂停的时间,并且也不能保证当前线程马上停止。yield方法只是将Running状态转变为Runnable状态。
(3)join方法的作用是父线程等待子线程执行完成后再执行,换句话说就是将异步执行的线程合并为同步的线程
线程
wait方法是一个本地方法,其底层是通过一个叫做监视器锁的对象来完成的。
如何获取monitor对象所有权?Java中只能通过Synchronized关键字来完成
1)调用wait方法后,线程是会释放对monitor对象的所有权的。
2)一个通过wait方法阻塞的线程,必须同时满足以下两个条件才能被真正执行:
线程需要被唤醒(超时唤醒或调用notify/notifyll)。
线程唤醒后需要竞争到锁(monitor)