这篇主要总结一下Java语法方面的技巧
Pubilic Class和Class
使用Pubilic class类名称声明一个类时,类名称必须和文件名称一致,否则程序将无法编译
JVM
每当使用Java命令执行一个类时,实际上都会启动一个JVM,每一个JVM实际上就是在操作系统中启动了一个进程,Java本身具备了垃圾收集机制,所以在Java运行时至少会启动两个线程,一个是main线程,另外一个是垃圾收集线程。
Java有参构造方法和无参构造方法
- 编写一个类时没有添加无参构造方法,那么编译器会自动添加无参构造方法;(如果自己添加构造函数,无论有参数或是没参数,默认构造函数都将无效)
- 编写时添加了有参构造方法而未添加无参构造方法,那么编译器只认有参构造方法而不会默认添加无参构造方法!
如果需要使用无参构造方法,一定要在类里面添加
对象的生命周期
String,StringBuffer,StringBuilder
- String是不可变的,String类是final修饰的,所以值是不变的(StirngBufer,StringBuilder可变)
- String,StringBuffer是线程安全的,StringBuilder是线程不安全的(StringBuffer的append操作用了synchronized)
- String对象串联的效率最慢,单线程下字符串的串联用StringBuilder,多线程下用StringBuffer
equals和hashCode
- 如果两个对象相同(equals方法返回true),那么它们的hashCode值一定要相同;
- 如果两个对象的hashCode相同,它们并不一定相同。
- 当然,你未必要按照要求去做,但是如果你违背了上述原则就会发现在使用容器时,相同的对象可以出现在Set集合中,同时增加新元素的效率会大大下降(对于使用哈希存储的系统,如果哈希码频繁的冲突将会造成存取性能急剧下降)。
volatile关键字
volatile关键字的两层语义
- 保证了不同线程对这个变量进行操作时的可见性,即一个线程修改了某个变量的值,这新值对其他线程来说是立即可见的。
- 禁止进行指令重排序。
内存模型
当程序在运行过程中,会将运算需要的数据从主存复制一份到CPU的高速缓存当中,那么CPU进行计算时就可以直接从它的高速缓存读取数据和向其中写入数据,当运算结束之后,再将高速缓存中的数据刷新到主存当中。
如果一个变量在多个CPU中都存在缓存(一般在多线程编程时才会出现),那么就可能存在缓存不一致的问题。
为了解决缓存不一致性问题,通常来说有以下2种解决方法:
- 通过在总线加LOCK#锁的方式(在锁住总线期间,其他CPU无法访问内存,导致效率低下)
- 通过缓存一致性协议(Intel 的MESI协议)
1 | 核心的思想是:当CPU写数据时,如果发现操作的变量是共享变量,即在其他CPU中也存在该 |
final关键字
final关键字可以用来修饰类、方法和变量(包括成员变量和局部变量)
- 修饰类
当用final修饰一个类时,表明这个类不能被继承。final类中的成员变量可以根据需要设为final,但是要注意final类中的所有成员方法都会被隐式地指定为final方法。 - 修饰方法
“使用final方法的原因有两个。第一个原因是把方法锁定,以防任何继承类修改它的含义;第二个原因是效率。在早期的Java实现版本中,会将final方法转为内嵌调用。但是如果方法过于庞大,可能看不到内嵌调用带来的任何性能提升。在最近的Java版本中,不需要使用final方法进行这些优化了。“
因此,如果只有在想明确禁止 该方法在子类中被覆盖的情况下才将方法设置为final的。
注:类的private方法会隐式地被指定为final方法
。 - 修饰变量
修饰变量是final用得最多的地方
对于一个final变量,如果是基本数据类型的变量,则其数值一旦在初始化之后便不能更改;如果是引用类型的变量,则在对其初始化之后便不能再让其指向另一个对象。
Static关键字
- static方法
“static方法就是没有this的方法。在static方法内部不能调用非静态方法,反过来是可以的。而且可以在没有创建任何对象的前提下,仅仅通过类本身来调用static方法。这实际上正是static方法的主要用途。”
一句话来描述就是:方便在没有创建对象的情况下来进行调用(方法/变量)。
- static变量
static变量也称作静态变量,静态变量和非静态变量的区别是:静态变量被所有的对象所共享,在内存中只有一个副本,它当且仅当在类初次加载时会被初始化。而非静态变量是对象所拥有的,在创建对象的时候被初始化,存在多个副本,各个对象拥有的副本互不影响。
static成员变量的初始化顺序按照定义的顺序进行初始化。
- static代码块
在类初次被加载的时候,会按照static块的顺序来执行每个static块,并且只会执行一次。
Extends implements
Extends可以理解为全盘继承了父类的功能。implements可以理解为为这个类附加一些额外的功能;interface定义一些方法,并没有实现,需要implements来实现才可用。extend可以继承一个接口,但仍是一个接口,也需要implements之后才可用。
对于class而言,Extends用于(单)继承一个类(class),而implements用于实现一个接口(interface)
Implements 接口实现
Implements是一个类实现一个接口用的关键字,它是用来实现接口中定义的抽象方法