final方法
《java编程思想》中讲到final方法时提到,使用final方法原因有两个,
一、锁定方法。防止任何继承类修改、覆盖
二、效率。在java早期实现中,如果将一个方法指明为final,就是同意编译器将针对该方法的调用都转化为内嵌调用。
如果是内嵌调用,虚拟机不再执行正常的方法调用(参数压栈,跳转到方法处执行,再调回,处理栈参数,处理返回值),而是直接将方法展开,以方法体重的实际代码替代原来的方法调用。这样减少了方法调用的开销。
内联函数
对函数(方法)进行某种处理,在程序运行的时候将函数(方法)展开,转换为代码。要问这样做的意义就不得不提到方法(Java中没有函数,以下用方法表示)的运行原理。
内联函数特点:
(1)提升效率。如上说明。
(2)占更多内存空间。编译器直接将内联函数扩展开,调用多复制品就多,因此更占用内存。
(3)java中不需额外关注,jvm会自动进行优化。
举例:
int max (int a, int b){
if (a > b)
return a;
else
return b;
}
void main() {
.....
a = max (x, y); // 内联,等价于 "a = (x > y ? x : y);" 直接扩展开了,不再调用方法
.....
}
调用方法的原理
调用某个方法实际上将程序执行顺序转移到该方法所存放在内存中某个地址,将方法的程序内容执行完后,再返回到转去执行该方法前的地方。这种转移操作要求在转去前要保护现场并记忆执行的地址,转回后先要恢复现场,并按原来保存地址继续执行。也就是通常说的压栈和出栈。因此,函数调用要有一定的时间和空间方面的开销。那么对于那些函数体代码不是很大,又频繁调用的函数来说,这个时间和空间的消耗会很大。
因此对于这种内容较短却又反复使用的方法我们可以通过使用内联函数来提升运行效率。
最后
final关键字只是告诉编译器,在编译的时候考虑性能的提升,可以将final函数视为内联函数。但最后编译器会怎么处理,编译器会分析将final函数处理为内联和不处理为内联的性能比较了。(和垃圾处理机制类似,程序员只有建议权而没有决定权)