目录
一、static成员(补)
1.1. static修饰成员方法
1.2. static成员变量初始化
二、代码块
2.1. 静态代码块和实例代码块
三、对象的打印
一、static成员(补)
1.1. static修饰成员方法
public class Linear {public static int count = 100;public int func(){Linear lin1 = new Linear();Linear lin2 = new Linear();Linear lin3 = new Linear();lin1.count++;lin2.count++;lin3.count++;Linear.count++;return count;}
}
public class Main {public static void main(String[] args) {Linear num = new Linear();System.out.println(num.func());}
}
老铁们猜一下打印的count是多少?这段代码具有迷惑性,老铁们可能猜会是101,而实际结果确实104。count因为被static修饰,就不属于对象了,所以count都是以同一个变量进行运算。class前的public是包访问权限,这个对象只能在同一个包中进行实例化访问。
而下面,博主将带大家进行更加直观的感受(接下来的代码可能会震碎你的三观):按常理推断我们给num赋值了一个null,按理说应该会抛出异常。可是我们一运行,照样会打印。这就是因为ret被static修饰,从而不属于任何对象。
public class demo {public static String ret = "bite";
}
public class Main {public static void main(String[] args) {demo num = null;System.out.println(num.ret);}
}
1.2. static成员变量初始化
第一种是就地初始化,在定义时直接给出初始值。
public class Student{private String name;private String gender;private int age;private double score;private static String classRoom = "107"; }
另一种是静态代码块初始化,接下来会讲到。
二、代码块
2.1. 静态代码块和实例代码块
以下是静态代码块与实例代码块的语法规则:
//静态代码块static{System.out.println("静态代码块被执行了");
}
//实例代码块
{System.out.println("实例代码块被执行了");
}
package demo;class Student{private String name;private int age;private static String ClassRoom;
//实例代码块
{System.out.println("实例代码块被执行了");
}
//静态代码块static{System.out.println("静态代码块被执行了");
}public Student(String name, int age) {this.name = name;this.age = age;System.out.println("构造Student(String name, int age)方法被执行了。。。");}public String getName() {return name;}public void setName(String name) {this.name = name;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}
}
public class StarWars {public static void main(String[] args) {Student stu = new Student("Sandman",16);}
}
以下为执行的结果:
从这里可以看到,执行的顺序是先执行静态代码块,再执行实例代码块,最后是构造方法。如果我们在new一个stu2的对象,再次执行,结果如下:静态代码块只被执行了一次。
如果我们在静态代码块里面,如果我们调用一个成员变量,则会产生报错。
三、对象的打印
Student stu1 = new Student("Sandman",16);
Student stu2 = new Student("Truck",18);System.out.println(stu1);
System.out.println(stu2);
打印的结果我们可以理解为地址,@的左边demo包底下的Student类,右边是地址。当我们stu1和stu2这两个引用变量接受参数时,它是怎么调用的呢?我们看下println的源码。
public void println(Object x) {String s = String.valueOf(x);if (getClass() == PrintStream.class) {// need to apply String.valueOf again since first invocation// might return nullwriteln(String.valueOf(s));} else {synchronized (this) {print(s);newLine();}}
}
当我们调用构造方法时,object类里的x会接收参数,接着传给valueOf里面;
//valueOf的源码public static String valueOf(Object obj) {return (obj == null) ? "null" : obj.toString();
}
如果obj不是一个null,则会执行后面的toString;
//toString的源码public String toString() {return getClass().getName() + "@" + Integer.toHexString(hashCode());
}
最终返回并打印出的结果。
如果我们把toString的源码拿到我们的程序里面,把返回值改成我们想要打印的:
public String toString() {return "Your name";
}