java的基本类型
byte ,short,int ,long,float,double,char,boolean
Java的关键字
abstract:抽象类,不能实例,修饰方法和类,修饰类:publicabstractclass Test_abstract。 且可以有可以实现的方法在里面。且可以 用extends进行拓展实现。可重写任何非final方法。可以使用implements,但是只能是接口。
interface: 只能是接口集合类,不能增加方法体,不能使用implements,不能使用extends.一个类,可以实现多个接口。
final:可用于类和方法,表示该类不能扩展,该方法不能重写。
native, 用于方法,调用本地的方法,这个本地方法是java以外语言写的,例如c,c++. 0.1 java声明一个natie方法,然后编译,2.用javah生成.h文件,3,用.cpp文件实现native方法,4.将.cpp文件编译成动态链接库文件,5,java用system.loadLibrary(),加载,然后调用native方法。
synchronized:用于方法和代码块,如果应用静态方法,整个类被锁定,如果应用实例方法, 该实例被锁定,如果应用于对象或数组,该对象被锁定。
transient,用于类成员变量,表示他不是对象序列化的一部分,当序列化的时候,不序列化这个标示过的变量和属性。
volatile,表示这个变量可被多个线程修改,也表示他看到的值是最新的,一直存放内存中,也确保所有线程读到的值是相同的。类似轻度的synchroized.
instanceof, 用来确定这个对象所属哪个类。
const, 修饰的对象不能更新,类似final。
String类
==比较的是地址,equals比较的是值。
intern()方法:相当于会现在字符串池找,找不到,然后再将对象同步到常量池, 相对于JVM开销,会比较小。
1.7以后,常量池被放入到堆空间中,不会再复制一份放入字符串池。而是将引用地址放入过去,
1.6是常量池放在方法区。new的对象都在堆,引用的值都在方法区。
在JDK6.0及之前版本,字符串常量池是放在PermGen区(也就是方法区)中,所以常量池会受到PermGen内存大小的限制。
在JDK7.0版本,常量池的内存在Java堆上进行分配,意味着常量池不受固定大小的限制了。至于为什么移到堆内,大概是由于方法区的内存空间太小了。
JDK8中,虚拟机团队移除了永久代PermGen。
在HotSpot VM里实现的string pool功能的是一个StringTable类,它是一个Hash表,默认值大小长度是1009;这个StringTable在每个HotSpot VM的实例只有一份,被所有的类共享。字符串常量由一个一个字符组成,放在了StringTable上。
在JDK6.0中,StringTable的长度是固定的,长度就是1009,因此如果放入String Pool中的String非常多,就会造成hash冲突,导致链表过长,当调用String#intern()时会需要到链表上一个一个找,从而导致性能大幅度下降;
在JDK7.0中,StringTable的长度可以通过参数指定:-XX:StringTableSize=XXXX
StringBuffer 和 StringBuilder 类
StringBuilder是线程不安全,底层是数组copy。没有加synchroized. 两者底层差不多
StringBuffer是线程安全的,底层同理abstractStigBuilder, 加了synchroized.
Object类
wait()
方法就是使当前线程等待该对象的锁,当前线程必须是该对象的拥有者,也就是具有该对象的锁。wait()方法一直等待,直到获得锁或者被中断。wait(longtimeout)设定一个超时间隔,如果在规定时间内没有获得锁就返回。
调用该方法后当前线程进入睡眠状态,直到以下事件发生。
(1)其他线程调用了该对象的notify方法。
(2)其他线程调用了该对象的notifyAll方法。
(3)其他线程调用了interrupt中断该线程。
(4)时间间隔到了。
此时该线程就可以被调度了,如果是被中断的话就抛出一个InterruptedException异常。
notify方法
该方法唤醒在该对象上等待的某个线程。
notifyAll方法
该方法唤醒在该对象上等待的所有线程。
Integer类
Integer在127以内, 100==100,其他不等,因为有一个内部私有类,IntegerCache,它缓存了从-128到127之间的所有的整数对象。例如 1000!=1000。