Java超详细讲解设计模式之一的单例模式

目录
  • 单例模式
    • 1.单例模式的结构
    • 2.单例模式的实现
      • 2.1饿汉式
      • 2.2懒汉式
    • 3.单例模式的破坏
      • 3.1序列化和反序列化
      • 3.2反射

单例模式

单例模式顾名思义就是单一的实例,涉及到一个单一的类,该类负责创建自己的对象,同时确保只有一个对象被创建,并且提供一种可以访问这个对象的方式,可以直接访问,不需要实例化该类的对象。

单例模式的特点:

1.单例类只能有一个实例

2.这个实例必须由单例类自己创建

3.单例类需要提供给外界访问这个实例

单例模式的作用:

单例模式主要为了保证在Java应用程序中,一个类只有一个实例存在。

1.单例模式的结构

单例模式主要有以下角色:

  • 单例类

只能创建一个实例的类

  • 访问类

测试类,就是使用单例类的类

2.单例模式的实现

2.1饿汉式

饿汉式:类加载时创建该单实例类对象

1.饿汉式-方式1 静态成员变量

创建 饿汉式静态成员变量 单例类

public class Demo1 {

    /**
     *私有构造方法  让外界不能创建该类对象
     */
    private Demo1(){}

    /**
     * 在类中创建该本类对象 static是由于外界获取该类对象的方法getInstance()是 static
     * 这个对象instance就是静态成员变量
     */
    private static Demo1 instance = new Demo1();

    /**
     * 提供一个公共的访问方式,让外界可以获取该类的对象 static是因为外界不需要创建对象,直接通过类访问
     */
    public static Demo1 getInstance(){
        return instance;
    }
}

创建 饿汉式静态成员变量 测试类(访问类)

public class Test1 {
    public static void main(String[] args) {
      //创建demo1类的对象 这个时候就无法通过new创建了,因为demo1的构造方法是私有的
        Demo1 instance = Demo1.getInstance();

        Demo1 instance1 = Demo1.getInstance();

        //判断两个对象是否是同一个
        System.out.println(instance == instance1);

    }
}

输出true 表明是同一个对象,指向同一块内存地址,这样我们就保证了Demo1单例类只有一个对象被创建

2.饿汉式-方式2 静态代码块

创建 饿汉式静态代码块 单例类

public class Demo2 {
    //饿汉式单例类  静态代码块

    /**
     *私有构造方法  让外界不能创建该类对象
     */
    private Demo2(){}

    /**
     *  声明一个静态的成员变量instance但是不赋值(不创建对象)
     *  没有为instance赋值,默认为null
     */
    private static  Demo2 instance;

    /**
     * 在静态代码快中为instance赋值(创建对象)
     */
    static {
        instance = new Demo2();
    }
    /**
     * 提供一个公共的访问方式,让外界可以获取该类的对象 static是因为外界不需要创建对象,直接通过类访问
     */
    public static Demo2  getInstance(){
        return instance;
    }
}

创建 饿汉式静态代码块 测试类

public class Test2 {
    public static void main(String[] args) {
        Demo2 instance = Demo2.getInstance();

        Demo2 instance1 = Demo2.getInstance();

        System.out.println(instance == instance1);
    }
}

输出true 表明是同一个对象,指向同一块内存地址,这样我们就保证了Demo2单例类只有一个对象被创建

3.饿汉式-方式3(枚举方式)

枚举类实现单例模式是十分推荐的一种单例实现模式,由于枚举类型是线程安全的,并且只会加载一次,这是十分符合单例模式的特点的,枚举的写法很简单,而且枚举方式是所有单例实现中唯一一个不会被破环的单例实现模式

单例类

//枚举方式创建单例
public enum Singleton {
     INSTANCE;
}

测试类

public class Test1 {
    public static void main(String[] args) {
    Singleton instance = Singleton.INSTANCE;
    Singleton instance1 = Singleton.INSTANCE;

        System.out.println(instance == instance1);
        //输出 true

    }
}

注意:

​ 由于枚举方式是饿汉式,因此根据饿汉式的特点,枚举方式也会造成内存浪费,但是在不考虑内存问题下,枚举方式是首选,毕竟实现最简单了

2.2懒汉式

懒汉式:类加载时不会创建该单实例对象,首次使用该对象时才会创建

1.懒汉式-方式1 (线程不安全)

public class Demo3 {
    /**
     *私有构造方法  让外界不能创建该类对象
     */
    private Demo3(){}

    /**
     * 在类中创建该本类对象 static是由于外界获取该类对象的方法getInstance()是 static
     * 没有进行赋值(创建对象)
     */
    private static  Demo3 instance;

    /**
     * 提供一个公共的访问方式,让外界可以获取该类的对象 static是因为外界不需要创建对象,直接通过类访问
     */
    public static Demo3 getInstance(){
        //在首次使用该对象时创建,因此instance赋值也就是对象创建 就是在外界获取该单例类的方法getInstance()中创建
        instance = new Demo3();
        return instance;
    }

}
public class Test3 {
    public static void main(String[] args) {
        Demo3 instance = Demo3.getInstance();

        Demo3 instance1 = Demo3.getInstance();
        //判断两个对象是否是同一个
        System.out.println(instance == instance1);
    }
}

输出结果为false,表明我们创建懒汉式单例失败了。是因为我们在调用getInstance()时每次调用都会new一个实例对象,那么也就必然不可能相等了。

   // 如果instance为null,表明还没有创建该类的对象,那么就进行创建
        if(instance == null){
          instance = new Demo3();
        }
        //如果instance不为null,表明已经创建过该类的对象,根据单例类只能创建一个对象的特点,因此         //我们直接返回instance
        return instance;
    }

注意:

我们在测试是只是单线程,但是在实际应用中必须要考虑到多线程的问题。我们假设一种情况,线程1进入if判断然后还没来得及创建instance,这个时候线程1失去了cpu的执行权变为阻塞状态,线程2获取cpu执行权,然后进行if判断此时instance还是null,因此线程2为instance赋值创建了该单例对象,那么等到线程1再次获取cpu执行权,也进行了instance赋值创建了该单例对象,单例模式被破坏。

2.懒汉式-方式2 (线程安全)

我们可以通过加synchronized同步锁的方式保证单例模式在多线程下依旧有效

 public static synchronized Demo3 getInstance(){
        //在首次使用该对象时创建,因此instance赋值也就是对象创建 就是在外界获取该单例类的方法getInstance()中创建

        // 如果instance为null,表明还没有创建该类的对象,那么就进行创建

        if(instance == null){
          instance = new Demo3();
        }
        //如果instance不为null,表明已经创建过该类的对象,根据单例类只能创建一个对象的特点,因此我们直接返回instance
        return instance;
    }

注意:

虽然保证了线程安全问题,但是在getInstance()方法上添加了synchronized关键字,导致该方法执行效率很低(这是加锁的一个常见问题)。其实我们可以很容易发现,我们只是在判断instance时需要解决多线程的安全问题,而没必要在getInstance()上加锁

3.懒汉式-方式3(双重检查锁)

对于getInstance()方法来说,绝大部分的操作都是读操作,读操作是线程安全的,没必要让每个线程必须持有锁才能调用该方法,我们可以调整加锁的时机。

public class Demo4 {
    /**
     *私有构造方法  让外界不能创建该类对象
     */
    private Demo4(){}

    /**
     *
     * 没有进行赋值(创建对象) 只是声明了一个该类的变量
     */
    private static Demo4 instance;

    /**
     * 提供一个公共的访问方式,让外界可以获取该类的对象 static是因为外界不需要创建对象,直接通过类访问
     */
    public static  Demo4 getInstance(){

        // (第一次判断)如果instance为null,表明还没有创建该类的对象,那么就进行创建
        if(instance == null){
            synchronized (Demo4.class){
                //第二次判断 如果instance不为null
                if(instance == null){
                    instance = new Demo4();
                }
            }

        }

        //如果instance不为null,表明已经创建过该单例类的对象,不需要抢占锁,直接返回
        return instance;
    }

}

双重检查锁模式完美的解决了单例、性能、线程安全问题,但是只是这样还是有问题的…

JVM在创建对象时会进行优化和指令重排,在多线程下可能会发生空指针异常的问题,可以使用volatile关键字,volatile可以保证可见性和有序性。

 private static volatile Demo4  instance;

如果发生指令重排 2 和 3 的步骤颠倒,那么instance会指向一块虚无的内存(也有可能是有数据的一块内存)

完整代码

public class Demo4 {
    /**
     *私有构造方法  让外界不能创建该类对象
     */
    private Demo4(){}

    /**
     * volatile可以保证有序性
     * 没有进行赋值(创建对象) 只是声明了一个该类的变量
     */
    private static volatile Demo4  instance;

    /**
     * 提供一个公共的访问方式,让外界可以获取该类的对象 static是因为外界不需要创建对象,直接通过类访问
     */
    public static  Demo4 getInstance(){
        // (第一次判断)如果instance为null,表明还没有创建该类的对象,那么就进行创建
        if(instance == null){
            synchronized (Demo4.class){
                //第二次判断 如果instance不为null
                if(instance == null){
                    instance = new Demo4();
                }
            }
        }

        //如果instance不为null,表明已经创建过该单例类的对象,不需要抢占锁,直接返回
        return instance;
    }
}

4.懒汉式-4 (静态内部类)

静态内部类单例模式中实例由内部类创建,由于JVM在加载外部类的过程中,是不会加载静态内部类的,只有内部类的属性/方法被调用时才会被加载,并初始化其静态属性。静态属性由于被final修饰,保证只被实例化一次,并且严格保证实例化顺序。

创建单例类

public class Singleton {

    private Singleton(){}

    /**
     *定义一个静态内部类
     */
    private static  class SingletonHolder{
        //在静态内部类中创建外部类的对象
        private static final Singleton INSTANCE = new Singleton();
    }

    public static Singleton getInstance(){
        return SingletonHolder.INSTANCE;
    }
}

创建测试类

public class Test4 {
    public static void main(String[] args) {
        Singleton instance = Singleton.getInstance();

        Singleton instance1 = Singleton.getInstance();
        //判断两个对象是否是同一个
        System.out.println(instance == instance1);
    }
}

注意:

​ 第一次加载Singleton类时不会去初始化INSTANCE,只有在调用getInstance()方法时,JVM加载SingletonHolder并初始化INSTANCE,这样可以保证线程安全,并且Singleton类的唯一性

​ 静态内部类单例模式是一种开源项目比较常用的单例模式,在没有任何加锁的情况下保证多线程的安全,并且没有任何性能和空间上的浪费

3.单例模式的破坏

单例模式最重要的一个特点就是只能创建一个实例对象,那么如果能使单例类能创建多个就破坏了单例模式(除了枚举方式)破坏单例模式的方式有两种:

3.1序列化和反序列化

从以上创建单例模式的方式中任选一种(除枚举方式),例如静态内部类方式

//记得要实现Serializable序列化接口
public class Singleton implements Serializable {

    private Singleton(){}

    /**
     *定义一个静态内部类
     */
    private static  class SingletonHolder{
        //在静态内部类中创建外部类的对象
        private static final Singleton INSTANCE = new Singleton();
    }

    public static Singleton getInstance(){
        return SingletonHolder.INSTANCE;
    }
}

测试类

public class Test1 {

    public static void main(String[] args) throws IOException {
              writeObjectToFile();
    }

    /**
     * 向文件中写数据(对象)
     * @throws IOException
     */
    public static void writeObjectToFile() throws IOException {
        //1.获取singleton对象
        Singleton instance = Singleton.getInstance();
        //2.创建对象输出流对象
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("d:\\1.txt"));
        //3.写对象
        oos.writeObject(instance);
        //4.释放资源
        oos.close();

    }
}

在d盘根目录下出现一个文件1.txt由于数据是序列化后的 咱也看不懂

然后我们从这个文件中读取instance对象

public static void main(String[] args) throws Exception {
             // writeObjectToFile();
        readObjectFromFile();
        readObjectFromFile();
    }
    /**
     * 从文件中读数据(对象)
     * @throws Exception
     */
    public static  void readObjectFromFile() throws Exception {

        //1.创建对象输入流对象
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("d:\\1.txt"));
        //2.读对象
        Singleton instance = (Singleton) ois.readObject();
        System.out.println(instance);
        //3.释放资源
        ois.close();
    }

输出结果不相同,结论为:序列化破坏了单例模式,两次读的对象不一样了

com.xue.demo01.Singleton@2328c243
com.xue.demo01.Singleton@bebdb06

解决方案

在singleton中添加readResolve方法

  /**
     * 当进行反序列化时,会自动调用该方法,将该方法的返回值直接返回
     * @return
     */
    public Object readResolve(){
        return SingletonHolder.INSTANCE;
    }

重新进行写和读,发现两次读的结果是相同的,解决了序列化破坏单例模式的问题

为什么在singleton单例类中添加readResolve方法就可以解决序列化破坏单例的问题呢,我们在ObjectInputStream源码中在readOrdinaryObject方法中

 private Object readOrdinaryObject(boolean unshared)
        throws IOException{
//代码段
Object obj;
        try {
            //isInstantiable如果一个实现序列化的类在运行时被实例化就返回true
            //desc.newInstance()会通过反射调用无参构造创建一个新的对象
            obj = desc.isInstantiable() ? desc.newInstance() : null;
        } catch (Exception ex) {
            throw (IOException) new InvalidClassException(
                desc.forClass().getName(),
                "unable to create instance").initCause(ex);
        }

   //代码段

   if (obj != null &&
            handles.lookupException(passHandle) == null &&
       //hasReadResolveMethod 如果实现序列化接口的类中定义了readResolve方法就返回true
            desc.hasReadResolveMethod())
        {
       //通过反射的方式调用被反序列化类的readResolve方法
            Object rep = desc.invokeReadResolve(obj);
            if (unshared && rep.getClass().isArray()) {
                rep = cloneArray(rep);
            }

    //代码段
 }

3.2反射

从以上创建单例模式的方式中任选一种(除枚举方式),例如静态内部类方式

测试类

public class Test1 {

    public static void main(String[] args) throws Exception {

        //1.获取Singleton的字节码对象
        Class<Singleton> singletonClass = Singleton.class;

        //2.获取无参构造方法对象
        Constructor cons = singletonClass.getDeclaredConstructor();

        //3.取消访问检查
        cons.setAccessible(true);
        //4.反射创建对象
        Singleton instance1 = (Singleton) cons.newInstance();

        Singleton instance2 = (Singleton) cons.newInstance();

        System.out.println(instance1 == instance2);
        //输出false 说明反射破坏了单例模式
    }

}

解决方案:

public class Singleton  {

    //static是为了都能访问
    private static boolean flag = false;

    private Singleton() {
        //加上同步锁,防止多线程并发问题
        synchronized (Singleton.class) {
            //判断flag是否为true,如果为true说明不是第一次创建,抛异常
            if (flag) {
                throw new RuntimeException("不能创建多个对象");
            }
            //flag的值置为true
            flag = true;
        }
    }

    /**
     *定义一个静态内部类
     */
    private static  class SingletonHolder{
        //在静态内部类中创建外部类的对象
        private static final Singleton INSTANCE = new Singleton();
    }

    public static Singleton getInstance(){
        return SingletonHolder.INSTANCE;
    }
}

这样就不能通过之前的反射方式破坏单例模式了,但是如果通过反射修改flag的值也是可以破坏单例模式的,但是这样可以防止意外反射破坏单例模式,如果刻意破坏是很难防范的,毕竟反射太强了

(0)

相关推荐

  • Java23种设计模式中的单例模式你了解吗

    目录 1.定义 2.适用场景 3.常见写法 4.如何防止单例被破坏 1.多线程破坏单例以及解决方法 2.反射破坏单例以及解决方法 3.序列化破坏单例以及解决方法 5.优缺点 6.总结 1.定义 单例模式(Singleton Pattern)是指确保一个类在任何情况下都绝对只有一个实例,并提供一个全局访问点.隐藏其所有的构造方法.属于创建型模式. 2.适用场景 确保任何情况下都绝对只有一个实例. 3.常见写法 第一种:饿汉式单例:在单例类首次加载时就创建实例 /** * @Package: com

  • Java设计模式之单例模式示例详解

    目录 0.概述 1.饿汉式 1.1 饿汉式单例实现 1.2 破坏单例的几种情况 1.3 预防单例的破坏 2.枚举饿汉式 2.1 枚举单例实现 2.2 破坏单例 3.懒汉式 4.双检锁懒汉式 5.内部类懒汉式 6.JDK中单例的体现 0.概述 为什么要使用单例模式? 在我们的系统中,有一些对象其实我们只需要一个,比如说:线程池.缓存.对话框.注册表.日志对象.充当打印机.显卡等设备驱动程序的对象.事实上,这一类对象只能有一个实例,如果制造出多个实例就可能会导致一些问题的产生,比如:程序的行为异常.

  • Java中常用的设计模式之单例模式详解

    目录 注意 优点 缺点 使用场景 一.实现方式 二.实现方式 三.测试 总结 注意 1.单例类只能有一个实例. 2.单例类必须自己创建自己的唯一实例. 3.单例类必须给所有其他对象提供这一实例. 优点 1.在内存里只有一个实例,减少了内存的开销,尤其是频繁的创建和销毁实例(比如管理学院首页页面缓存). 2.避免对资源的多重占用(比如写文件操作). 缺点 1.没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而不关心外面怎么样来实例化. 使用场景 1.要求生产唯一序列号. 2.WE

  • Java设计模式之单例模式深入探索

    目录 什么是设计模式? 单例模式是什么? 单例模式设计的原则是什么? Java实现单例模式的5种方式? 懒汉 饿汉 静态内部类 双重校验锁DCL(Double Check Lock) 枚举(num) 小结 ❤️‍您好,我是贾斯汀,今天来聊一聊单例模式!❤️‍ 什么是设计模式? 百科: 设计模式是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结. 设计模式是软件行业的通用的设计标准,在Java同样通用,主要有23种设计模式如下: 有的小伙伴可能会问,这么多,学得完吗? 答:不好意思

  • java编程创建型设计模式单例模式的七种示例

    目录 1.什么是单例模式? 2.七种写法 2.1饿汉式(静态常量) 2.2饿汉式(静态代码块) 2.3懒汉式(线程不安全) 2.4懒汉式(线程安全,同步方法) 2.5双重校验锁 2.6静态内部类 2.7枚举 3.单例模式在JDK中的应用(简单的源码分析) 4.单例模式总结 1.什么是单例模式? 所谓类的单例设计模式,就是采取一定的方法保证在整个的软件系统中,对某个类只能存在一个对象实例,并且该类只提供一个取得其对象实例的方法(静态方法). 比如Hibernate的 SessionFactory,

  • JAVA设计模式零基础解析之单例模式的八种方式

    目录 单例模式简介: 单例模式优点: 应用场景: 单例设计模式的八种方式: 1.饿汉式(静态常量) 2.饿汉式(静态代码块) 3.懒汉式(线程不安全) 4.懒汉式(线程安全,同步方法) 5.懒汉式(线程安全,同步代码块) 6.双重检查(推荐使用) 7.静态内部类(推荐使用) 8.枚举(推荐使用) 单例模式在JDK应用的源码分析 单例模式注意事项和细节说明 单例模式简介: 单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一.这种类型的设计模式属于创建型模式,它提供了

  • 深入理解Java设计模式之单例模式

    目录 一.什么是单例模式 二.单例模式的应用场景 三.单例模式的优缺点 四.单例模式的实现 1.饿汉式 2.懒汉式 3.双重加锁机制 4.静态初始化 五.总结 一.什么是单例模式 单例模式是一种常用的软件设计模式,其定义是单例对象的类只能允许一个实例存在. 许多时候整个系统只需要拥有一个的全局对象,这样有利于我们协调系统整体的行为.比如在某个服务器程序中,该服务器的配置信息存放在一个文件中,这些配置数据由一个单例对象统一读取,然后服务进程中的其他对象再通过这个单例对象获取这些配置信息.这种方式简

  • Java超详细讲解设计模式之一的单例模式

    目录 单例模式 1.单例模式的结构 2.单例模式的实现 2.1饿汉式 2.2懒汉式 3.单例模式的破坏 3.1序列化和反序列化 3.2反射 单例模式 单例模式顾名思义就是单一的实例,涉及到一个单一的类,该类负责创建自己的对象,同时确保只有一个对象被创建,并且提供一种可以访问这个对象的方式,可以直接访问,不需要实例化该类的对象. 单例模式的特点: 1.单例类只能有一个实例 2.这个实例必须由单例类自己创建 3.单例类需要提供给外界访问这个实例 单例模式的作用: 单例模式主要为了保证在Java应用程

  • Java超详细讲解设计模式之一的工厂模式

    目录 工厂模式 1.简单工厂 1.1结构 1.2实现 1.3优缺点 1.4扩展 2.工厂方法 2.1结构 2.2实现 2.3优缺点 3.抽象工厂 3.1结构 3.2实现 3.3优缺点 4.模式扩展 4.1实现 工厂模式 在Java应用程序中对象无处不在,这些对象都需要进行创建,如果创建的时候直接new对象,那么如果我们要更换对象,所有new对象的地方都需要进行更改.违背了软件设计原则中的开闭原则.如果我们使用工厂生产对象,只需要在工厂中关注对象的改变即可,达到了与对象解耦的目的,工厂模式最大的特

  • Java 超详细讲解设计模式之中的抽象工厂模式

    目录 抽象工厂模式 1.什么是抽象工厂 2.抽象工厂模式的优缺点 3.抽象工厂模式的结构与实现 4.抽象工厂方法模式代码实现 5.抽象工厂模式的应用场景 6.抽象工厂模式的扩展 抽象工厂模式 前面文章介绍的工厂方法模式中考虑的是一类产品的生产,比如案例中的百事可乐工厂只能生产百事可乐,可口可乐工厂只能生产可口可乐,也就是说:工厂方法模式只考虑生产同等级的产品. 1.什么是抽象工厂 在现实生活中许多工厂是综合型的工厂,能生产多种类)的产品,就拿案例里面的可乐来说,在节日的时候可能会有圣诞版的可乐,

  • Java 超详细讲解设计模式之中的建造者模式

    目录 1.什么是建造者模式? 2.建造者模式的定义 3.建造者模式的优缺点 4.建造者模式的结构 5.建造者模式代码演示 6.建造者模式的应用场景 7.建造者模式和工厂模式的区别 1.什么是建造者模式? 我们知道在软件开发过程中有时需要创建一个很复杂的对象,通常由多个子部件按一定的步骤组合而成. 例如,比如我们在自己在组装一台计算机的时候,需要有 CPU.主板.内存.硬盘.显卡.机箱.显示器.键盘.鼠标等部件组装而成的.比如学校需要采购100台计算机,学校不可能自己把零件买过来自己组装,肯定是告

  • Java 超详细讲解设计模式之原型模式讲解

    目录 传统方式 原型模式基本介绍 原型模式在spring框架中源码分析 深入讨论-浅讨论和深拷贝 原型模式的注意事项和细节 传统方式 克隆羊问题 现在有一只羊 tom,姓名为: tom,年龄为:1,颜色为:白色,请编写程序创建和 tom羊属性完全相同的10只羊. 传统方式解决克隆羊问题 思路分析(图解) 代码演示: public class Sheep { private String name; private int age; private String color; public She

  • Java超详细讲解设计模式中的命令模式

    目录 介绍 实现 个人理解:把一个类里的多个命令分离出来,每个类里放一个命令,实现解耦合,一个类只对应一个功能,在使用命令时由另一个类来统一管理所有命令. 缺点:如果功能多了就会导致创建的类的数量过多 命令模式(Command Pattern)是⼀种数据驱动的设计模式,它属于行为型模式.请求以命令的形式包裹在对象中,并传给调⽤对象.调⽤对象寻找可以处理该命令的合适的对象,并把该命令传给相应的对象,该对象执⾏命令. 介绍 意图:将⼀个请求封装成⼀个对象,从⽽使您可以⽤不同的请求对客户进⾏参数化.

  • Java 超详细讲解IO操作字节流与字符流

    目录 IO操作 字节流 FileInputStream FileOutputStream 字节流读写案例 字符流 FileReader FileWriter 字节流与字符流的区别 IO操作 字节流 java.io.InputStream 输入流,主要是用来读取文件内容的. java.io.OutputStream 输出流,主要是用来将内容字节写入文件的. FileInputStream 该流用于从文件读取数据,它的对象可以用关键字 new 来创建. 有多种构造方法可用来创建对象. 可以使用字符串

  • Java 超详细讲解类的定义方式和对象的实例化

    目录 1.面对对象的初步认识 1.1什么是面向对象 1.2面向对象与面向过程 2.类的定义与使用 2.1简单认识类 2.2 类的定义格式 3.类的实例化 3.1什么是实例化? 3.2重点笔记 总结 1.面对对象的初步认识 1.1什么是面向对象 用面向对象的思想来涉及程序,更符合人们对事物的认知,对于大型程序的设计.扩展以及维护都非常友好. 1.2面向对象与面向过程 举一个买手机的例子 以面向对象的方式来处理买手机这件事的话,我们就不需要关注买手机的过程,具体手机怎么买,如何到手,用户不用去关心,

  • Java 超详细讲解对象的构造及初始化

    目录 如何初始化对象 构造方法 特性 默认初始化 就地初始化 如何初始化对象 我们知道再Java方法内部定义一个局部变量的时候,必须要初始化,否则就会编译失败 要让这串代码通过编译,很简单,只需要在正式使用a之前,给a设置一个初始值就好那么对于创造好的对象来说,我们也要进行相对应的初始化我们先写一个Mydate的类 public class MyDate { public int year; public int month; public int day; /** * 设置日期: */ pub

  • Java 超详细讲解数据结构中的堆的应用

    目录 一.堆的创建 1.向下调整(以小堆为例) 2.创建堆 3.创建堆的时间复杂度 二.堆的插入和删除 1.堆的插入 2.堆的删除 三.堆的应用 1.堆排序 2.top-k问题 [求最小的K个数] 四.常用接口的介绍 1.PriorityQueue的特性 2.优先级队列的构造 一.堆的创建 1.向下调整(以小堆为例) 让parent标记需要调整的节点,child标记parent的左孩子(注意:parent如果有孩子一定先是有左孩子) 如果parent的左孩子存在,即:child < size,

随机推荐