分析:
JAVA重载解析过程分为2步:
(1)根据实参选取所有可应用的。
(2)根据形参选取最精确的。(并没有使用实参)
比如:
如果fun(Object o)和fun(double[]b)都是可应用的,则double[]的每个对象都是Object类型的,因此double[]比Object更精确。
public class PuzzleDemo46{ private PuzzleDemo46(Object o){ System.out.println("Object"); } private PuzzleDemo46(double[]darr){ System.out.println("double array"); } public static void main(String args[]){ new PuzzleDemo46(null); } }
每个静态的属性都在声明他的类和所有的子类中保持一个副本。
class Cat{ private static int count = 0; public Cat(){ } public void meow(){ count++; } public static int getCount(){ return count; } } class Dog{ private static int count = 0; public Dog(){ } public void woof(){ count++; } public static int getCount(){ return count; } } public class PuzzleDemo47{ public static void main(String args[]){ Cat c[] = {new Cat(),new Cat()}; Dog d[] = {new Dog(),new Dog(),new Dog()}; for(Cat e1:c){ e1.meow(); } for(Dog d1:d){ d1.woof(); } System.out.println("Cat:" + Cat.getCount()); System.out.println("Dog:" + Dog.getCount()); } }
静态方法调用不是动态的,因此如果调用一个静态方法,则在编译器执行期已经确定。
比如:Person p = new Student(); p.say(); 如果say是静态方法,则调用的是Person类的静态方法。
比如:(Person)null.say();调用的是Person.say(),null不起任何作用。
总结:调用静态方法时,必须使用类来调用,而不是用类的实例进行调用。
class Dog{ public static void bark(){ System.out.println("woof "); } } class Basenji extends Dog{ public static void bark(){ } } public class PuzzleDemo48{ public static void main(String args[]){ Dog.bark(); Basenji.bark(); } }示例2:
public class PuzzleDemo54{ public static void main(String args[]){ ((PuzzleDemo54)null).greet(); greet(); } public static void greet(){ System.out.println("Hello world!"); } }
类初始化顺序:
(1)把静态字段设置为默认值。
(2)静态字段初始器按照其在源代码中出现顺序执行,比如static属性、final static属性、static语句块等,因此会出现调用还没有被初始器初始化的静态字段,尽量把属性都放在类的前面。
(3)在执行类的方法前,必须对类进行初始化。
积极初始化和延迟初始化:不要同时使用两者。
当在程序每次执行都会用到时,则使用积极初始化。
当不是每次执行都会用到时,则使用延迟初始化。
结论:类的初始化顺序显得重要时,请特别当心!
比较两个程序:
import java.util.*; public class PuzzleDemo49_2{ private static final PuzzleDemo49_2 INSTANCE = new PuzzleDemo49_2(); private final int beltSize; private final int CURRENT_YEAR = Calendar.getInstance().get(Calendar.YEAR); private PuzzleDemo49_2(){ beltSize = CURRENT_YEAR-1930; } public int beltSize(){ return beltSize; } public static void main(String args[]){ System.out.println("Elvis wears a size " + INSTANCE.beltSize() + "belt."); } } //CURRENT_YEAR is final ,so not static
import java.util.*; public class PuzzleDemo49{ private static final PuzzleDemo49 INSTANCE = new PuzzleDemo49(); private final int beltSize; private static final int CURRENT_YEAR = Calendar.getInstance().get(Calendar.YEAR); private PuzzleDemo49(){ beltSize = CURRENT_YEAR-1930; } public int beltSize(){ return beltSize; } public static void main(String args[]){ System.out.println("Elvis wears a size " + INSTANCE.beltSize() + "belt."); } }
class Cache{ private static boolean initialized = false; static{ initializeIfNecessary(); } private static int sum; public static int getSum(){ initializeIfNecessary(); return sum; } private static synchronized void initializeIfNecessary(){ if(!initialized){ for(int i=0;i<100;i++){ sum +=i; } initialized = true; } } } public class PuzzleDemo52{ public static void main(String args[]){ System.out.println(Cache.getSum()); } }
(1)如果instanceof的左操作数是null,则返回false;
(2)如果两个操作数的类型都是类,则instanceof的左右操作数一定是子类或父类关系;
public class PuzzleDemo50{ public static void main(String args[]){ System.out.println("null instanceof String:"+(null instanceof String)); System.out.println("int instanceof String:"+(new Integer(1) instanceof String)); //不可转换 } }
public class PuzzleDemo51 extends Point{ private final String color; PuzzleDemo51(int x,int y,String color){ super(x,y); this.color = color; } protected String makeName(){ return super.makeName()+":"+color; } public static void main(String args[]){ System.out.println(new PuzzleDemo51(4,2,"Puzzle")); } } class Point{ private final int x,y; private String name; Point(int x,int y){ this.x = x; this.y = y; } protected String makeName(){ return "["+x+","+y+"]"; } public final String toString(){ name = this.makeName();//延迟初始化 return name; } }
JAVA语言规范:不允许一个局部变量声明语句作为一条语句在for、while、do循环中重复执行。
因此一个局部变量声明语句只能出现在一个语句块中。
public class PuzzleDemo55{ public static void main(String args[]){ for(int i=0;i<100;i++){ Creature creature = new Creature(); } System.out.println(Creature.numCreated()); } } class Creature{ private static long numCreated = 0; public Creature(){ synchronized(Creature.class){ numCreated++; } } public static long numCreated(){ return numCreated; } }