一、静态方法
在Java里,可以定义一个不需要创建对象的方法,这种方法就是静态方法。要实现这样的效果,只需要在类中定义的方法前加上static关键字。例如:
一般情况下,工具类里面的方法都会定义为静态方法,例如下面这个排序工具类
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Vector;
public class SortUtils {
/**
* 对数组从小到大排序
* @param array
*/
public static void insertSort(int[] array){
for (int i = 1; i < array.length; i++) {
int currentVal = array[i]; //设置哨兵
int position = i; //即将插入的位置
for(int j=i-1; j>=0; j--){
if(array[j]>currentVal){
array[j+1] = array[j]; //如果前一个元素比后一个元素大,就往后移
position = j;
}else{
break;
}
}
array[position] = currentVal; //插入元素
}
}
/**
* 希尔排序
* 从小到大排序
* @param array
*/
public static void shellSort(int[] array){
int dk = array.length/2;
while(dk>=1){
shellInsertSort(array,dk);
dk=dk/2;
}
}
/**
* 根据增量对数组从小到大排序
* @param array
* @param dk 增量
*/
public static void shellInsertSort(int[] array,int dk){
for (int i = dk; i < array.length; i++) {
int currentVal = array[i]; //设置哨兵
int position = i; //即将插入的位置
for(int j=i-dk; j>=0; j=j-dk){
if(array[j]>currentVal){
array[j+dk] = array[j]; //如果前一个元素比后一个元素大,就往后移
position = j;
}else{
break;
}
}
array[position] = currentVal; //插入元素
}
}
//........
}
使用静态方法时,注意以下几点:
- 在静态方法里只能直接调用同类中其他的静态成员(包括变量和方法),而不能直接访问类中的非静态成员。这是因为,对于非静态的方法和变量,需要先创建类的实例对象后才可使用,而静态方法在使用前不用创建任何对象。
- 静态方法不能以任何方式引用this和super关键字,因为静态方法在使用前不用创建任何实例对象,当静态方法调用时,this所引用的对象根本没有产生。
- 在同一个类中,非静态方法里面可以直接调用同类中的静态成员(包括变量和方法)。而且可以通过this来调用。
例如:
import java.io.IOException;
import java.util.Scanner;
public class Test {
public static void main(String[] args) {
new Test().set();
}
public static void get(){
System.out.println("static");
}
public void set(){
this.get();
}
}
输入结果:static
- 在其他类中,既可以通过类名直接调用静态成员,也可以通过对象调用静态成员,但是二者操作的始终是同一个。
class HasStatic {
public static int x = 100;
}
public abstract class Test{
public static void main(String args[]) {
HasStatic hs1 = new HasStatic();
hs1.x++;
HasStatic hs2 = new HasStatic();
hs2.x++;
hs1 = new HasStatic();
hs1.x++;
HasStatic.x--;
System.out.println("x=" + HasStatic.x);
}
}
输出结果:x=102
二.静态变量
静态变量是属于整个类的变量而不是属于某个对象的。
使用静态变量时,注意以下几点:
- 不能把方法体内的变量声明为静态。
fun(){
static int i=0;//非法。
}
三.静态代码块
一个类可以使用不包含在任何方法体中的静态代码块,当类被载入时,静态代码块被执行,且只被执行一次,静态块常用来执行类属性的初始化。
1.类装载步骤
在Java中,类装载器把一个类装入Java虚拟机中,要经过三个步骤来完成:装载、链接和初始化,其中链接又可以分成校验、准备和解析三步,除了解析外,其它步骤是严格按照顺序完成的,各个步骤的主要工作如下:
装载:查找和导入类或接口的二进制数据;
链接:执行下面的校验、准备和解析步骤,其中解析步骤是可以选择的;
校验:检查导入类或接口的二进制数据的正确性;
准备:给类的静态变量分配并初始化存储空间;
解析:将符号引用转成直接引用;
初始化:激活类的静态变量的初始化Java代码和静态Java代码块。
初始化类中属性是静态代码块的常用用途,但只执行一次。
2.静态代码块的初始化顺序
public abstract double hyperbolicCosine();
public final static native int w();
class Parent {
static String name = "hello";
{
System.out.println("parent block");
}
static {
System.out.println("parent static block");
}
public Parent() {
System.out.println("parent constructor");
}
}
class Child extends Parent {
static String childName = "hello";
{
System.out.println("child block");
}
static {
System.out.println("child static block");
}
public Child() {
System.out.println("child constructor");
}
}
public class StaticIniBlockOrderTest {
public static void main(String[] args) {
new Child();//语句(*)
}
}
输出结果:
parent static block
child static block
parent block
parent constructor
child block
child constructor
分析:
当执行new Child()时,它首先去看父类里面有没有静态代码块,如果有,它先去执行父类里面静态代码块里面的内容,当父类的静态代码块里面的内容执行完毕之后,接着去执行子类(自己这个类)里面的静态代码块,当子类的静态代码块执行完毕之后,它接着又去看父类有没有非静态代码块,如果有就执行父类的非静态代码块,父类的非静态代码块执行完毕,接着执行父类的构造方法;父类的构造方法执行完毕之后,它接着去看子类有没有非静态代码块,如果有就执行子类的非静态代码块。子类的非静态代码块执行完毕再去执行子类的构造方法,这个就是一个对象的初始化顺序。
注意:
- 子类的构造方法,不管这个构造方法带不带参数,默认的它都会先去寻找父类的不带参数的构造方法。如果父类没有不带参数的构造方法,那么子类必须用supper关键子来调用父类带参数的构造方法,否则编译不能通过。
- 在同一个类中,静态代码块可以有多处
3.静态代码块与静态方法的区别
在类加载时,两者都会加载,但是静态代码块会立即执行,并且只执行这一次,而静态方法只用当调用时才会执行,而且每次调用时都会执行一次。