[TOC]
1. static关键字
1.1 类变量、类方法
    /**
     * static关键字
     * 类变量不用实例化,直接类名.属性名就可以使用,是类的一部分,被所有这个类的实例化所共享,也可以叫做静态变量
     * 如果想让一个类的所有实例共享数据,就用类变量!
     * 
     * 实列变量,只是实例化后才能使用,属于实例化对象的一部分,不能公用
     * 
     * 使用范围
     * 可用static修饰属性、方法、代码块、内部类
     * 被修饰后的成员具备以下特点
     * 1.随着类的加载而加载(类加载后,静态方法或属性就可以使用了,类名.方法名)
     * 2.优先于对象存在,不用new就可以使用
     * 3.修饰的成员被所有对象所共享
     * 4.访问权限允许时,可不创建对象,直接被类调用
     * 
     * 注意:
     * 类变量,这种可以被所有实例化对象所共享的属性,使用起来要慎重,因为只要一改,所有类都能得到变化
     * 
     * 类方法:作为工具类的情况较多
     * static方法内部不能有this(也不能有super),因为this是针对对象的,static是类的,重载方法的方法需要同时为static或非static
     */
1.2 工具类
静态方法
//判断字符串是否为空或null
public class Utils {
    public static boolean isEmpty(String s) {
        boolean flag = false;
        if (s ==null && s.equals("")){
            flag = true;
        }
        return flag;
    }
    
}
//调用工具类
Utils.isEmpty("test")
1.3 单例 (Singleton)设计模式
懒汉式单例设计模式
public class Single1 {
    private Single1() {}
    /**
     * 懒汉式实例:第一次创建实例前为null,第一次创建实例后,创建,后面的都是同一个实例
     */
    //私有的类变量,为null;
    private static Single1 s1 = null;
    
    //公共的方法,如果s1为null,就创建实例,后面的就不用创建了
    public static Single1 getInstance() {
        if(s1 == null) {
            s1 = new Single1();
        }
        return s1;
    }
}
饿汉式单例设计模式
public class Single {
    /**
     * 单例设计模式:
     * 设计模式:就是在我们实际编程过程中,逐渐总结出的一些解决问题的套路
     * 单例设计模式:只有一个实例(实例化对象)在整个软件系统运行过程中,这个类只被实例化一次,以后无论在哪都只调用这一个实例
     */
    //饿汉式
    //私有的构造方法
    private Single() {
        
    }
    //私有的类变量,创建实例
    private static Single s = new Single();
    
    //公共的方法,返回私有的类变量 
    public static Single getInstance() {
        return s;
    }
}
懒汉式和饿汉式的区别
        /**
         * 懒汉式和饿汉式的区别:
         * 就是什么时候new这个对象,懒汉式,是在第一次有人调用getlnstance方法时类new对象,
         * 以后再有人调用getinstance方法直接就返回第一次new好的对象
         * 
         * 饿汉式,是在类加载之后,还没有人调用的时候,就new好一个对象,以后无论谁来调用
         * getinstance方法都直接返回直接new好的那个对象
         */
1.4 main方法
        /**
         * main方法
         * public static void main(String[] args) {}
         * 由于java虚拟机需要调用类的main()方法,所以该方法的访问权限必须是public,
         * 又因为java虚拟机在执行main()方法时不必创建对象,所以该方法必须是static的,
         * 该方法接收一个String类型的数组参数,
         * 该数组中保存执行java命令时传递给所运行的类的参数
         */
2. 类的成员之初始化块
2.1 非静态代码块
//非静态代码块
    {
        System.out.println("这是非静态代码块中的方法1");
    }
    {
        System.out.println("这是非静态代码块中的方法2");
    }
2.2静态代码块
    //静态代码块
    //复杂的属性,实例属性
        static TestPer tp= new TestPer();
    static {
        //这里只能使用静态的属性和方法
        age = 2;
        showAge();
        /**
         * 这程序的运行过程中,非静态代码块每次new对象都有重新执行
         * 静态代码块只执行一次
         */
        System.out.println("这执行的是静态代码块");
        //复杂的属性,可以通过代码块来初始化调用
        tp.age = 1;
        tp.name = "张三";
    }
2.3 匿名内部类
        //匿名内部类
        Person p = new Person() {//这就是一个Person的匿名子类,
            /**
             * 构建了一个没有类名的Person的子类,也就是匿名的Person子类
             * 这两种类没有类名,就不能显示的new的方法创建对象,如果要是还要在构造器中初始化
             * 属性就没有办法了,这样情况就要用代码块{}来初始化工作
             */
            {
                //
                super.name = "李四";
            }
            //从写方法
            @Override
            public void test() {
                System.out.println("=========");
            }
        };//分号结尾
        System.out.println(p.name);
        p.test();
2.4 总结
- 初始化块(代码块)作用:对Java对象进行初始化
- 程序的执行顺序:声明成员变量的默认值-显式初始化-多个初始化块依次被执行(同级别下按先后顺序执行)-构造器再对成员进行赋值操作
- 一个类中初始化块若有修饰符,则只能被static修饰,称为静态代码块(static block ),当类被载入时,类属性的声明和静态代码块先后顺序被执行,且只被执行一次。
- 非静态代码块:没有static修饰的代码块 - 可以有输出语句。 - 可以对类的属性声明进行初始化操作。 - 可以调用静态和非静态的变量或方法。 - 若有多个非静态的代码块,那么按照从上到下的顺序依 次执行。 - 每次创建对象的时候,都会执行一次。且先于构造器执行
- 静态代码块:用static 修饰的代码块 - 可以有输出语句。 - 可以对类的属性声明进行初始化操作。 - 不可以对非静态的属性初始化。即:不可以调用非静态的属 性和方法。 - 若有多个静态的代码块,那么按照从上到下的顺序依次执行。 - 静态代码块的执行要先于非静态代码块。 - 静态代码块只执行一次
3. final关键字
在Java中声明类、属性和方法时,可使用关键字final来修饰,表示“最终”。 final标记的类不能被继承。提高安全性,提高程序的可读性 final标记的方法不能被子类重写 final标记的变量(成员变量或局部变量)即称为常量。名称大写,且只能被赋值一次。 static final:全局常量
4. 抽象类
/**
 * 抽象类:
 * 1.用abstract关键字来修饰一个类时,这个类叫做抽象类
 * 2.用abstract来修饰一个方法时,该方法叫做抽象方法
 * 3.抽象方法:只有方法的声明,没有方法的实现。以分号结束:abstract int abstractMethod( int a );
 * 4.含有抽象方法的类必须被声明为抽象类
 * 5.抽象类不能被实例化。抽象类是用来作为父类被继承的,抽象类的子类必须重写父类的抽象方法,并提供方法体。若没有重写全部的抽象方法,仍为抽象类
 * 6.不能用abstract修饰属性、私有方法、构造器、静态方法、final的方法
 * @author Zw
 *
 */
public abstract class Employee {
    String id;
    String name;
    double salary;
    //构造方法
    public Employee() {}
    //抽象方法
    public abstract void work();
}
/**
 * 公司员工类,继承Employee类
 * 获取和设置员工信息
 *
 * @author Zw
 *
 */
class CommonEmployee extends Employee{
    @Override
    public void work() {
        System.out.println("这是一个员工.");
    }
    public void setEmployeeInfo(String id, String name, double salary) {
        super.id = id;
        super.name = name;
        super.salary = salary;
    }
    public void getEmployeeInfo() {
        System.out.println(id+"\n"+name+"\n"+salary);
    }
}
/**
 * 领导类继承Employee
 * @author Zw
 *
 */
class Manager extends Employee{
    double bonus;
    @Override
    public void work() {
        System.out.println("这是一个领导.");        
    }
    public void setManager(String id, String name, double salary, double bonus) {
        super.id = id;
        super.name = name;
        super.salary = salary;
        this.bonus = bonus;//子类的属性用this
    }
    public void getManager() {
        System.out.println(id+"\n"+name+"\n"+salary+"\n"+bonus);
    }
}
5. 模板模式的设计
/**
 * 模板模式的设计:
 * 抽象类体现的就是一种模板模式的设计,抽象类作为多个子类的通用模板,子类在抽象类的基础上进行扩展、改造,但子类总体上会保留抽象类的行为方式。
 * 解决的问题:
 * 1. 当功能内部一部分实现是确定,一部分实现是不确定的。这时可以把不确定的部分暴露出去,让子类去实现。
 * 2. 编写一个抽象父类,父类提供了多个子类的通用方法,并把一个或多个方法留给其子类实现,就是一种模板模式。
 * @author Zw
 *
 */
public abstract class Template {
    public abstract void run();
    //提供子类调用的,计算run方法运行时间
    public final void runTime() {
        long start = System.currentTimeMillis();
        run();
        long end = System.currentTimeMillis();
        System.out.println(end - start);
    }
}
class testTemplate extends Template{
    @Override
    public void run() {
        int a = 1;
        for(int i=0; i<9999999;i++) {
            a += i;
        }        
        
    }
}
//模板设计模式,main中调用
        Template tp = new testTemplate();
        tp.runTime();//7毫秒
6. 接口
6.1 什么时候使用接口
- 抽象类是对于一类事物的高度抽象,其中既有属性也有方法; 
- 接口是对方法的抽象,也就是一系列动作的抽象 
- 当需要对一类事物抽象的时候,应该使用抽象类,好形成一个法类 
- 当需要对一系列的动作(方法)抽象,就使用接口,需要使用这些动作的类去实现相应的接口 
6.2 接口的特点
- 用interface来定义。 
- 接口中的所有成员变量都默认是由public static final修饰的。 
- 接口中的所有方法都默认是由public abstract修饰的。 
- 接口没有构造器。 
- 接口采用多层继承机制。 
- 实现接口的类中必须提供接口中所有方法的具体实现内容,方可实例化。否则,仍为抽象类 
- 接口的主要用途就是被实现类实现 
- 与继承关系类似,接口与实现类之间存在多态性 
- 定义Java类的语法格式:先写extends,后写implements 
- 如果实现接口的类中没有实现接口中的全部方法,必须将此类定义为抽象类 
6.3 具体实例
需求: teach类继承person类,并实现cook、sing接口

代码
 * @author Zw
 * person 父类
 */
public class Person1 {
    String name;
    int age;
    int sex;
    public void showInfo() {
        
    }
    
}
/**
 * Cook接口
 * @author Zw
 */
public interface Cook {
    void cooking();
}
/**
 * Sing接口
 * @author Zw
 */
public interface Sing {
    void singing();
}
public class TeachCT extends Person1 implements Cook,Sing {//先继承后实现
    String score;//老师自己的属性,科目
    
    @Override
    public void showInfo() {
        System.out.println("这是一个会唱歌做菜的老师的信息");
        System.out.println(super.name+"\n"+super.age+"\n"+super.sex+"\n"+this.score);
    }
    public void setInfo() {
        super.name = "李四";
        super.age = 11;
        super.sex = 1;
        this.score = "物理";
    }
    @Override
    public void singing() {//重写singing
        System.out.println(super.name+"老师擅长美声唱法!");
        
    }
    @Override
    public void cooking() {//重写cooking
        System.out.println(super.name+"老师擅长炒菜!");
        
    }
}
//调用
//接口
        TeachCT  tc = new TeachCT();
        tc.setInfo();
        tc.showInfo();
        tc.cooking();
        tc.singing();
        
        //接口对象的多态
        Cook tp1 = new TeachCT();//可以转为实现接口的类型
        tp1.cooking();//只能使用Cook类自己的属性和方法
7. 工厂模式
FactoryMethod模式是设计模式中应用最为广泛的模式,在面向对象的编程中,对象的创建工作非常简单,对象的创建时机却很重要。FactoryMethod解决的就是这个问题,它通过面向对象的手法,将所要创建的具体对象的创建工作延迟到了子类,从而提供了一种扩展的策略,较好的解决了这种紧耦合的关系。 实例
//父类接口
public interface BMW {
    //产品信息
    void showInfo();
}
//宝马3系产品,实现接口
class BMW1 implements BMW{
    @Override
    public void showInfo() {
        System.out.println("这是BMW3系产品");        
    }    
} 
//宝马5系产品
class BMW5 implements BMW{
    @Override
    public void showInfo() {
        System.out.println("这是BMW5系产品");        
    }    
} 
//宝马7系产品
class BMW7 implements BMW{
    @Override
    public void showInfo() {
        System.out.println("这是BMW7系产品");        
    }    
} 
//工厂模式
/**
 * 汽车生产工厂接口
 * 通过工厂把new对象给隔离,通过产品的接口可以接受不同实际产品的实现类,
 * 实现类名改变不影响其他合作开发人员的编程
 * @author Zw
 *
 */
public interface BMWFactory {
    BMW productBMW();
}
/**
 * 实现具体车型的生产工厂
 * @author Zw
 *
 */
class BMW3Factory implements BMWFactory{
    @Override
    public BMW productBMW() {
        System.out.println("这是生产宝马3系车");
        return new BMW1();//return 具体的产品
    }    
}
class BMW5Factory implements BMWFactory{
    @Override
    public BMW productBMW() {
        System.out.println("这是生产宝马3系车");
        return new BMW5();
    }    
}
class BMW7Factory implements BMWFactory{
    @Override
    public BMW productBMW() {
        System.out.println("这是生产宝马3系车");
        return new BMW7();
    }    
}
//调用
public class Test2 {
    public static void main(String[] args) {
        /**
         * 调用工厂方法
         */
        BMW b3 = new BMW3Factory().productBMW();//new 具体车型的产品().方法()
        b3.showInfo();
        
        BMW b5 = new BMW5Factory().productBMW();
        b5.showInfo();
        
        BMW b7 = new BMW7Factory().productBMW();
        b7.showInfo();
    }
}
 
  
  
 