美文网首页
java.lang基础包之java.lang学习之Object

java.lang基础包之java.lang学习之Object

作者: miyakee | 来源:发表于2017-10-09 09:24 被阅读0次

    java.lang是提供利用 Java 编程语言进行程序设计的####基础类。
    我们可以将该包大致分成以下几类

    • Object
    • Class
    • 数字与字符(包装器)类
    • 线程与进程类
    • Math类
    • Runtime类
    • System类
      我们首先讲讲很重要的object类。有一句话是这样的
      在java里,一切皆对象。
      所以Object类在java里占有举足轻重的重要性。
    主要方法

    首先我们看看Object类有哪些方法:

    • private void registerNatives()
    • Class< > getClass()
    • int hashCode()
    • boolean equals(Object obj)
    • protected Object clone()
    • String toString()
    • void notify()
    • void notifyAll()
    • void wait(long timeout)
    • void wait(long timeout, int nanos)
    • void wait()
    • protected void finalize()

    1. private void registerNatives()

    对象初始化的时候会自动调用 这个方法。是在c(c++)中实现的本地方法,其主要作用是将C/C++中的方法映射到Java中的native方法,实现方法命名的解耦。

    2. Class< > getClass()

    返回的是此Object对象的类对象/运行时类

    3.int hashCode()

    • 在一个应用程序执行期间,如果一个对象的equals方法做比较所用到的信息没有被修改的话,则对该对象调用hashCode方法多次,它必须始终如一地返回同一个整数。
    • 如果两个对象相等(equal),那么必须拥有相同的哈希码(hash code)
    • 即使两个对象有相同的哈希值(hash code),他们不一定相等.
      根据上面可以看出:
      一般重写equals方法的时候hashcode方法也要被重写的,因为相等的对象的hashcode一定相等

    4.boolean equals(Object obj)

        public boolean equals(Object obj) {
            return (this == obj);
        }
    

    我们可以看到其实就是对两个对象的内存地址的比较。
    ps:有些包装类String 、Math、Integer等重写了equals()方法,就是对内容的比较了。

    5 protected Object clone()

    这里涉及到深拷贝和浅拷贝的问题:
    简单的来说就是,在有指针的情况下,浅拷贝只是增加了一个指针指向已经存在的内存,而深拷贝就是增加一个指针并且申请一个新的内存,使这个增加的指针指向这个新的内存,采用深拷贝的情况下,释放内存的时候就不会出现在浅拷贝时重复释放同一内存的错误
    我们可以看到clone方法是protect 直接调用是不可以的,但是protect在子类是可以调用的,在java所有的类都缺省继承了Object类。所以我们只需要关心重写的clone这个方法。重写的方法需要继承

    public interface Cloneable { 
     }
    

    然后在对象里重写clone()方法

    public Object clone(){
             Object o=null;
             try {
                  o=super.clone();
             } catch (CloneNotSupportedException e) {
                  e.printStackTrace();
            }
            return o;
      }
    

    重写了clone方法的当前类的复制是深拷贝,但是如果再该类里面存在另一个对象没有重写clone方法就是浅拷贝。实例如下

    
    public class ObjectUseDto implements Cloneable {
        private String name;
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        private ShallowCopy shallowCopy;
    
        public ShallowCopy getShallowCopy() {
            return shallowCopy;
        }
    
        public void setShallowCopy(ShallowCopy shallowCopy) {
            this.shallowCopy = shallowCopy;
        }
    
        
        public Object clone() {
            Object o = null;
            try {
                //Object中的clone()识别出你要复制的是哪一个对象。
                o = super.clone();
            } catch (CloneNotSupportedException e) {
                System.out.println(e.toString());
            }
            return o;
        }
    }
    
    class ShallowCopy {
        private Integer age;
    
        public Integer getAge() {
            return age;
        }
    
        public void setAge(Integer age) {
            this.age = age;
        }
    }
    

    调用方代码

            ObjectUseDto test=new ObjectUseDto();
            test.setName("小花");
            ShallowCopy shallowCopy=new ShallowCopy();
            shallowCopy.setAge(11);
            test.setShallowCopy(shallowCopy);
            ObjectUseDto test2=(ObjectUseDto)test.clone();
            System.out.println(test2.getName());
            System.out.println(test2.getShallowCopy().getAge());
            test.setName("大猫");
            shallowCopy.setAge(88);
            System.out.println(test2.getName());
            System.out.println(test2.getShallowCopy().getAge());
    
    

    输出的结果
    小花
    11
    小花
    88
    可以看出重写了clone的是深拷贝,每次我们改变的都是原有对象。原有的对象发生了改变从‘小花’变成‘大猫’了 但是拷贝的对象并没有发生改变,再看原有的age从 11 变成了 88 。

    6 String toString()

        public String toString() {
            return getClass().getName() + "@" + Integer.toHexString(hashCode());
        }
    
    

    没有被重写的时候 表示没有被覆盖的时候,返回类名@哈希值,哈希值是十六进制的

    7 protected void finalize()

    我们先说下finalize()方法
    可以看做一个析构方法 会在没有引用的时候再垃圾回收之前被垃圾回收机制调用


                            后面的方法主要是与多线程相关的了。
    

    8 void notify() 只能在同步控制方法或者同步控制块里面使用

    这个方法的主要功能是 唤醒在此对象监视器上等待的单个线程

    9 void notifyAll() 只能在同步控制方法或者同步控制块里面使用

    这个方法的主要功能是 唤醒在此对象监视器上等待的所有线程

    10 void wait() 只能在同步控制方法或者同步控制块里面使用
    这个方法的主要功能是 调用此方法所在的当前线程等待,直到在其他
    线程上调用此方法的主调(某一对象)的notify()/notifyAll()方法。
    在这里我们可以比较下sleep()的区别:
    在调用sleep()方法的过程中,线程不会释放对象锁。
    而当调用wait()方法的时候,线程会放弃对象锁,进入等待此对象的等待锁定池,只有针对此对象调用notify()方法后本线程才进入对象锁定池准备获取对象锁进入运行状态。

    11 void wait(long timeout) 只能在同步控制方法或者同步控制块里面使用

    这个方法的主要功能是 线程等待 直到在其他线程上调用此方法的主调(某一对象)的notisfy()/notisfyAll()方法 或者等到一定时间

    12 void wait(long timeout, int nanos) 只能在同步控制方法或者同步控制块里面使用

    这个方法的主要功能是 线程等待 直到在其他线程上调用此方法的主调(某一对象)的notisfy()/notisfyAll()方法 或者等到一定时间

    native 关键字

    我们可以看到在Object类里有很多的方法都声明了native关键字那么native关键字有什么作用呢?
     说明其修饰的方法是一个原生态方法,方法对应的实现不是在当前文件,而是在用其他语言(如C和C++)实现的文件中。Java语言本身不能对操作系统底层进行访问和操作,但是可以通过JNI接口调用其他语言来实现对底层的访问

    final关键字

    一旦声明这个类,方法,变量为final ,那么这个引用将不能被改变,当再次尝试初始化会编译错误

    finalize() 方法:

    可以看做一个析构方法 会在没有引用的时候再垃圾回收之前被垃圾回收机制调用

    相关文章

      网友评论

          本文标题: java.lang基础包之java.lang学习之Object

          本文链接:https://www.haomeiwen.com/subject/pavcyxtx.html