JDK阅读笔记

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)

你可能感兴趣的:(JDK阅读笔记)