java基础之 超类Object

1、概述:java

      Object类是全部Java类的祖先。每一个类都使用 Object 做为超类。全部对象(包括数组)都实现这个类的方法。
      在不明确给出超类的状况下,Java会自动把Object做为要定义类的超类。
      可使用类型为Object的变量指向任意类型的对象。
      Object类是Java中惟一没有父类的类
      Object类有一个默认构造方法pubilc Object(),在构造子类实例时,都会先调用这个默认构造方法。

 2、API预览
程序员

    Object()
    默认构造方法
数组

    clone()
    建立并返回此对象的一个副本。
    equals(Object obj)
    指示某个其余对象是否与此对象“相等”。
    finalize()
    当垃圾回收器肯定不存在对该对象的更多引用时,由对象的垃圾回收器调用此方法。
    getClass()
    返回一个对象的运行时类。
    hashCode()
    返回该对象的哈希码值。
    notify()
    唤醒在此对象监视器上等待的单个线程。
    notifyAll()
    唤醒在此对象监视器上等待的全部线程。
    toString()
    返回该对象的字符串表示。
    wait()
    致使当前的线程等待,直到其余线程调用此对象的 notify() 方法或 notifyAll() 方法。
    wait(long timeout)
    致使当前的线程等待,直到其余线程调用此对象的 notify() 方法或 notifyAll() 方法,或者超过指定的时间量。
    wait(long timeout, int nanos)
    致使当前的线程等待,直到其余线程调用此对象的 notify() 方法或 notifyAll() 方法,或者其余某个线程中断当前线程,或者已超过某个实际时间量。
dom

3、方法使用说明
equals() (判断两个对象是否相等)
ide

 1)基本数据类型,也称原始数据类型。byte,short,char,int,long,float,double,boolean   他们之间的比较,用双等号(==),比较的是他们的值。 函数

 2)复合数据类型(类)   当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,因此,除非是同一个new出来的对象,他们的比较后的结果为true,不然比较后结果为false。equals()这个方法的初始行为是比较对象的内存地址,但在一些类库当中这个方法被覆盖掉了,如String,Integer,Date在这些类当中equals有其自身的实现,而再也不是比较类在堆内存中的存放地址了。工具

例:this

        People p=new People();
        p.setName("234");
        
        People p2=new People();
        p2.setName("234");
        
        System.out.println(p.equals(p2)); //false
        System.out.println(p==p2);//false

        String s="123";
        String s1=new String("123");
        String s2=new String("123");
        
       
        System.out.println(s==s1);//false
        System.out.println(s.equals(s1));//true
        System.out.println(s1==s2);//false
        System.out.println(s1.equals(s2));//true
View Code

Java语言规范要求equals方法具备下面的特色: spa

  • 自反性:对于任何非空引用值 x,x.equals(x) 都应返回 true
  • 对称性:对于任何非空引用值 x 和 y,当且仅当 y.equals(x) 返回 true 时,x.equals(y) 才应返回 true
  • 传递性:对于任何非空引用值 x、y 和 z,若是 x.equals(y) 返回 true,而且 y.equals(z) 返回 true,那么 x.equals(z) 应返回 true
  • 一致性:对于任何非空引用值 x 和 y,屡次调用 x.equals(y) 始终返回 true 或始终返回 false,前提是对象上 equals 比较中所用的信息没有被修改。 对于任何非空引用值 x,x.equals(null) 都应返回 false

 hashCode()( 返回该对象的哈希码值)

  hashCode 的常规协定是(重写equals()时,必须重hashcode()并保证此协定):线程

  • 若是调用equals方法获得的结果为true,则两个对象的hashcode值一定相等
  • 若是equals方法获得的结果为false,则两个对象的hashcode值不必定不一样
  • 若是两个对象的hashcode值不等,则equals方法获得的结果一定为false
  • 若是两个对象的hashcode值相等,则equals方法获得的结果未知

咱们会有这样的协定,举个例子:

 

class User {
    String name;
    public User(String name) {
        this.name = name;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    @Override
    public boolean equals(Object o) {
        User u = (User) o;
        return name.equals(u.getName());
    }
}

 

 public static void main(String[] args) {
        List<User> list = new ArrayList<User>();
        list.add(new User("lily"));
        Map<User, String> map = new HashMap<User, String>();
        map.put(new User("lily"), "11");

        System.out.println(list.contains(new User("lily"))); //true
        System.out.println(map.containsKey(new User("lily")));//false
    }

缘由就不写了 。。本身领悟叭

 

toString()(返回该对象的字符串表示)

Object类中的toString()方法会打印出类名和对象的内存位置。几乎每一个类都会覆盖该方法,以便打印对该对象当前状态的表示。大多数(非所有)toString()方法都遵循以下格式:类名[字段名=值,字段名=值...],固然,子类应该定义本身的toString()方法。该方法是很是重要的调试工具,不少标准类库中的类都定义了toString()方法,以便程序员得到有用的调试信息

clone()(建立并返回此对象的一个副本

克隆的分类:
  (1)浅克隆(shallow clone),浅拷贝是指拷贝对象时仅仅拷贝对象自己和对象中的基本变量,而不拷贝对象包含的引用指向的对象。
  (2)深克隆(deep clone),深拷贝不只拷贝对象自己,并且拷贝对象包含的引用指向的全部对象。
克隆的实现:
  1. 让该类实现java.lang.Cloneable接口;
  2. 重写(override)Object类的clone()方法。
该方法是浅克隆,若是须要实现深克隆,则须要重写clone方法)
为何克隆是浅拷贝:
1) 效率和简单性,简单的copy一个对象在堆上的的内存比遍历一个对象网而后内存深copy明显效率高而且简单。
2 ) 不给别的类强加意义。若是A实现了Cloneable,同时有一个引用指向B,若是直接复制内存进行深copy的话,意味着B在乎义上也是支持Clone的,可是这个是在使用B的A中作的,B甚至都不知道。破坏了B原有的接口。
3) 有可能破坏语义。若是A实现了Cloneable,同时有一个引用指向B,该B实现为单例模式,若是直接复制内存进行深copy的话,破坏了B的单例模式。
4) 方便且更灵活,若是A引用一个不可变对象,则内存deep copy是一种浪费。Shadow copy给了程序员更好的灵活性。
克隆知足的条件:
clone()方法将对象复制了一份并返还给调用者。所谓“复制”的含义与clone()方法是怎么实现的。通常而言,clone()方法知足如下的描述:
(1)对任何的对象x,都有: x.clone()!=x。换言之,克隆对象与原对象不是同一个对象
(2)对任何的对象x,都有: x.clone().getClass() == x.getClass(),换言之,克隆对象与原对象的类型同样。
(3)若是对象x的equals()方法定义其恰当的话,那么 x.clone().equals(x)应当成立的。
在JAVA语言的API中,凡是提供了clone()方法的类,都知足上面的这些条件。JAVA语言的设计师在设计本身的clone()方法时,也应当遵照着三个条件。通常来讲,上面的三个条件中的前两个是必需的,而第三个是可选的。

建立对象的几种方式:

  1)new
     new操做符的本意是分配内存。程序执行到new操做符时, 首先去看new操做符后面的类型,由于知道了类型,才能知道要分配多大的内存空间。 分配完内存以后,再调用构造函数,填充对象的各个域,这一步叫作对象的初始化,构造方法返回后,一个对象建立完毕
  2)反射
   a: Student stu = (Student)Class.forName("根路径.Student").newInstance()
   b: Student stu = Student.class.newInstance()
区别:
> Class类的newInstance只能触发无参数的构造方法建立对象,而构造器类的newInstance能触发有参数或者任意参数的构造方法来建立对象。
> Class类的newInstance须要其构造方法是共有的或者对调用方法可见的,而构造器类的newInstance能够在特定环境下调用私有构造方法来建立对象。
> Class类的newInstance抛出类构造函数的异常,而构造器类的newInstance包装了一个InvocationTargetException异常。
  3)clone
       先分配内存,将原来对象内存里的内容copy一份。不会调用构造方法
  4)反序列化
 
 wait() & notify()
wait():
做用于持有该对象锁的当前线程,使线程从运行态,进入到等待阻塞状态,释放对象锁,释放cpu控制权,调用该方法的前提是得到对象锁
notify()/notifyAll:
做用于持有该对象锁的线程(其中一个/所有),使线程从等待阻塞状态,进入到锁池阻塞状态,调用该方法的前提是得到对象锁
举个栗子
生产者消费者模型
仓库
class Store {
    public static List<String> store = new ArrayList<String>();
    public static int limitSize = 5;
}
View Code

生产者

class Producer implements Runnable {
    @Override
    public void run() {
        while (true) {
            synchronized (Store.store) {
                while (Store.store.size() == Store.limitSize) {
                    try {
                        Store.store.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                System.out.println("建立一个商品");
                Store.store.add(UUID.randomUUID().toString());
                Store.store.notifyAll();
            }
        }
    }
}
View Code

消费者

class Consumer implements Runnable {
    @Override
    public void run() {
        while (true) {
            synchronized (Store.store) {
                while (Store.store.size() == 0) {
                    try {
                        Store.store.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
                System.out.println("消费一个商品");
                Store.store.remove(0);
                Store.store.notifyAll();
            }
        }
    }
}
View Code

启动生产者,消费者

public static void main(String[] args) {
        Thread producer = new Thread(new Producer());
        Thread consumer = new Thread(new Consumer());
        producer.start();
        consumer.start();
    }
View Code
相关文章
相关标签/搜索