Java类和对象

一般的类只能是public或者default的,若是public的,类名必须和文件名一样。一般在一个文件只写一个类,故这个类一般是加上public。

内部类还可以是private和protected,一般私有化(private),因为只有当前这个类才需要。这样只有它的外部类可以直接访问。所以里面的数据也无所谓是public还是private或者default的。

public class Out {
    private int d = 10;
    public int add() {
      // 外部类不能直接访问内部类的数据,除非new出内部类对象,如下
      // Out.In abc = new Out().new In();
        a = 3; // false
        b = 5; // false
    }
  // 内部类
    private class In {
        private int a = 100;
        int b;
        public int c;
        public void add() {
            System.out.println(d); // true, 内部类却可以直接访问外部类的数据
        }
    }

    public static void main(String[] args) {
        Out bb = new Out();
        Out.In abc = new Out().new In();
        bb.d = 20; // 本类可以直接访问private,一般不这样,而是写个set函数,搭配get函数
        System.out.println(aa.a); // 100
        System.out.println(bb.d); // 被修改为20

    }
}
  • 匿名对象
new Abc().run(); // 调用一次后这个对象就销毁
new Abc().run(); // 这是另外一个Abc对象了
  • 关于private
  • private变量:封装数据,只给用户看方法。例如private String name不能随修改,但是提供setName()、getName()方法可调用,实现数据的封装,更安全。
  • private方法:不用用户调用的函数,完成一些内部实现的细节,就可以定义成private。例如选择排序算法写成一个方法,里面有交换两个数据的过程,封装成函数。这个函数不需给用户操作(用户只需用一个sort()就行),这个swap()就定义成private的。

单例设计模式

可以保证一个类在内存中的对象唯一性,需对过个程序使用同一配置信息对象时,就需要保证对象的唯一性,这时就需要单例

工具类中,未用到(访问)对象成员数据,则可不用创建对象。为了防止创建对象,可以将构造函数定义为private

如何保证对象的唯一性?

  1. 不允许其他程序用new创建该类对象,构造函数设为private
  2. 在该类创建一个唯一的类实例
  3. 对外提供一个方法使其他程序可以获取到这个唯一对象

单例设计步骤

  1. 私有化(private)该类的构造函数
  2. 通过new在本类汇总穿件一个本类对象
  3. 定义一个puiblic方法,将2中创建的对象返回

饿汉模式

// 饿汉式--不管用不用得到,先new出一个再说
public class SingleDemo {
    // private不能将这个实例暴露出去,只能通过getInstance()获取,另为何是static?
    private static SingleDemo s = new SingleDemo();
    private SingleDemo() {}
    // 供外界调用来获取这个对象
    public static SingleDemo getInstance() {
        return s;
    }
  // 只有本类中可以new对象并调用add
    public void add() {
        System.out.println("danli");
    }
    public static void main(String[] args) {
      // 本类可以new
        SingleDemo a = new SingleDemo();
        a.add();
    }
}

public class Test {
    public static void main(String[] args) {
      // fasle, 其他类中若用到这个单例,不能new
        SingleDemo aa = new SingleDemo();
      // true
      SingleDemo aa = SingleDemo.getInstance();
    }
}

懒汉模式

// 另外一种单例设计模式--懒汉模式--用到了才new,延时加载
public class SingleDemo {
    // 先设为空
    private static SingleDemo s;
    private SingleDemo() {}
    // 供外界调用来获取这个对象
    public static SingleDemo getInstance() {
      if (s == null)
        s = new SingleDemo();
        return s;
    }

Q:为何成员变量是static,方法也是static?

A:因为在其他类中,不允许new出新对象,故不能调用getInstance(),如果我们设成静态方法,就可以通过Single.getInstance()来获取而无需new对象,又静态方法里的数据也必须是static的。故这个SingleDemo对象也必须是static的。

Q:饿汉模式和懒汉模式的区别?

A:懒汉模式是延时加载的,用到了才new;在多线程时不能保证对象的唯一性,是线程不安全的。饿汉模式是随着类的加载,就new出这个对象,不管用不用得上,是线程安全的。

继承

Java一般是单继承,不能直接多继承 --> 因为多个父类可能有相同的成员,调用会导致二义性。

但是可以多层继承,如 A --> B -->C--> D,可以表示A继承自B,而B继承自C,以此类推。

  • 本类成员和局部变量之间用this区分
  • 子类、父类的成员同名时用super区分
  • 派生类也不能访问基类的private成员,但是确实是继承过来了,只是不能访问而已,可以调用父类的set、get函数来操作子类的private成员,对父类没影响。
  • 若成员是protected等权限更高的,则不用set、get就可直接访问该成员。
  • 父类与子类的该成员是相互独立的,子类改变它自己的成员,父类保持原样。
package Chap1;

public class Fu {
  // 父类的成员是私有的,子类继承过去了也不能直接访问

    private int age;
    private String name;
    public Fu(int age, String name) {
        this.age = age;
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

class Zi extends Fu {
  // 特有的job,父不能访问
    private String job;
    Zi(int age, String name, String job) {
        super(age, name);
        this.job = job;
    }
    public String getJob() {
        return job;
    }

    public static void main(String[] args) {
        Zi aa = new Zi(23, "Zi","Study");
        Fu bb = new Fu(55, "Fu");
      // 改变子类不会影响父类
        aa.setAge(24);
        aa.setName("zi");
      // zi 24 Study
      // Fu 55
        System.out.println(aa.getName()+" "+aa.getAge() + " " + aa.getJob());
        System.out.println(bb.getName()+" "+bb.getAge());

    }
}
  • 子类隐式执行super(),new一个子对象时,先调用父类的构造函数,再执行子类的构造函数。
  • 默认的构造函数是无参的,有参数的构造函数会覆盖无参的。这时候需要显式使用super(args),且必须放在第一行,若不显式写super,则执行默认的super(),即无参的父类构造函数。
class Fu {
  // 构造函数
  Fu() {
    System.out.println("father");
  }
  // 带参的构造函数
   public Fu(int age, String name) {
        this.age = age;
        this.name = name;
    }
}

class Zi extends Fu {

  Zi() {
    private String job;
    // super() 会打印“father”
    // 无参数时是默认构造器,这句super()可以不写。父类的构造函数默认执行,先执行
    // 

  }

  Zi(int age, String name, String job) {
    // 若不写,会默认调用无参的super()打印father
        super(age, name);
        this.job = job;
    }
}
  • 类是public,构造函数也是public的。他们的修饰符是对应的。

一个对象的实例化过程

Person p = new Person();

  1. JVM读取Person.class文件,加载进内存;若有父类,会先加载父类。
  2. 在堆内存中开辟空间,分配地址。
  3. 在对象空间中,对对象的属性进行默认初始化,如int age = 0; String name = null;
  4. 显示初始化,如private int age = 9。若是子类,则会先调用父类的构造器。
  5. (子)类的 构造函数进行特定的初始化。如下程序中age变成100。
  6. 初始化后,将地址传递给引用变量。Person p <-- new Person();
package Test;
// 先打印9
// 再打印100

public class Demo {
    private int age = 9;
    {
        System.out.println(age);
    }
    Demo(int age) {
        this.age = age;
    }
    public static void main(String[] args) {
        Demo aa = new Demo(100);
        System.out.println(aa.age);
    }
}

by @sunhaiyu

2016.12.8

Java基础之类和对象、单例模式、继承的更多相关文章

  1. 第二十七节:Java基础面向对象-静态,单例模式,继承详情知识点

    前言 Java基础面向对象-静态,单例模式,继承详情知识点.静态-static关键字,static变量,静态代码块,代码块(不加静态),对象创建过程,单例模式,继承. 静态-static关键字 // ...

  2. 第31节:Java基础-类与对象

    前言 Java基础-类与对象,方法的重载,构造方法的重载,static关键字,main()方法,this关键字,包,访问权限,类的继承,继承性,方法的重写,super变量. 方法的重载:成员方法的重载 ...

  3. Java基础-面向对象第二特征之继承(Inheritance)

    Java基础-面向对象第二特征之继承(Inheritance) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.继承的概述 在现实生活中,继承一般指的是子女继承父辈的财产.在程序 ...

  4. Java基础-IO流对象之压缩流(ZipOutputStream)与解压缩流(ZipInputStream)

    Java基础-IO流对象之压缩流(ZipOutputStream)与解压缩流(ZipInputStream) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 之前我已经分享过很多的J ...

  5. Java基础-IO流对象之随机访问文件(RandomAccessFile)

    Java基础-IO流对象之随机访问文件(RandomAccessFile) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.RandomAccessFile简介 此类的实例支持对 ...

  6. Java基础-IO流对象之内存操作流(ByteArrayOutputStream与ByteArrayInputStream)

    Java基础-IO流对象之内存操作流(ByteArrayOutputStream与ByteArrayInputStream) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.内存 ...

  7. Java基础-IO流对象之数据流(DataOutputStream与DataInputStream)

    Java基础-IO流对象之数据流(DataOutputStream与DataInputStream) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.数据流特点 操作基本数据类型 ...

  8. Java基础-IO流对象之打印流(PrintStream与PrintWriter)

    Java基础-IO流对象之打印流(PrintStream与PrintWriter) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.打印流的特性 打印对象有两个,即字节打印流(P ...

  9. Java基础-IO流对象之序列化(ObjectOutputStream)与反序列化(ObjectInputStream)

    Java基础-IO流对象之序列化(ObjectOutputStream)与反序列化(ObjectInputStream) 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.对象的序 ...

  10. java基础-IO流对象之Properties集合

    java基础-IO流对象之Properties集合 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.Properties集合的特点 Properties类表示了一个持久的属性集. ...

随机推荐

  1. Linux命令 文件备份归档恢复

    cp [功能说明] 文件的备份 英文xxxx  #cp命令将源文件复制到另外安全的地方,复制的文件和源文件是两个相互独立的文件,对认识一个文件的操作不影响另一个文件,但与符号链接文件中的硬链接是有区别 ...

  2. CNN压缩:为反向传播添加mask(caffe代码修改)

    神经网络压缩的研究近三年十分热门,笔者查阅到相关的两篇博客,博主们非常奉献的提供了源代码,但是发发现在使用gpu训练添加mask的网络上,稍微有些不顺,特此再进行详细说明. 此文是在 基于Caffe的 ...

  3. module.exports,exports,export和export default,import与require区别与联系【原创】

    还在为module.exports.exports.export和export default,import和require区别与联系发愁吗,这一篇基本就够了! 一.首先搞清楚一个基本问题: modu ...

  4. arcgis api for js入门开发系列十一地图统计图

    上一篇实现了demo的叠加SHP图层,本篇新增地图统计图,截图如下: 地图统计图实现的思路如下:利用拓展arcgis api的js文件(MapChartGraphic.js以及MapChartGrap ...

  5. javascript中apply,call,bind区别,bind兼容等问题总结

    1 三者的相似之处: (1).都是用来改变函数的this对象的指向的 (2).都是用第一个参数来做this对象的指向 (3).都可以传参数进去 那么,具体到它们有什么区别呢?请看下面的例子: 两个对象 ...

  6. 深入浅出TCP/IP协议栈

    TCP/IP协议栈是一系列网络协议的总和,是构成网络通信的核心骨架,它定义了电子设备如何连入因特网,以及数据如何在它们之间进行传输.TCP/IP协议采用4层结构,分别是应用层.传输层.网络层和链路层, ...

  7. IE11中navigator.userAgent的变化

    在原来判断浏览器是否是ie时,我们可以根据navigator.userAgent中时候有MSIE,但是IE11进行变革,userAgent中不在包含MSIE字段, 在实际项目中,入到类似的在控制台报错 ...

  8. php产生随机字符串

    /** * 产生随机字符串 * * @param int $length 输出长度 * @param string $chars 可选的 ,默认为 0123456789 * @return strin ...

  9. linq中日期格式转换或者比较,程序报错说不支持方法的解决办法

    public void TestMethod1(){using (var _context = new hotelEntities()){var rq = DateTime.Now.Date;var ...

  10. 基于HTTP协议的下载功能实现

    超文本传输协议 (HTTP-HyperText Transfer Protocol)是一种使用极为广泛的协议,它由请求和响应构成,是一种无状态的应用层协议.设计HTTP协议的初衷是为了提供一种传输HT ...