# Java对象初始化

Java对象初始化

标签(空格分隔): java


初始化的形式

  1. 显示创建对象
       通过new关键字来调用一个类的构造函数——在java规范中被称为“由执行类实例创建表达式而引起的对象创建”。

2.隐式对象创建——不是通过new关键字来完成的
  情况一:加载一个包含String字面量的类或者接口会引起一个新的String对象被创建,除非包含相同字面量的String对象已经存在与虚拟机内了(JVM会在内存中会为所有碰到String字面量维护一份列表,程序中使用的相同字面量都会指向同一个String对象)

class StringLiteral {
    private String str = "literal";
    private static String sstr = "s_literal";
}

情况二:自动装箱机制可能会引起一个原子类型的包装类对象被创建

class PrimitiveWrapper {
    private Integer iWrapper = 1;
}

情况三:String连接符也可能会引起新的String或者StringBuilder对象被创建,同时还可能引起原子类型的包装对象被创建,比如(本人试了下,在mac ox下1.6.0_29版本的javac,对待下面的代码会通过StringBuilder来完成字符串的连接,并没有将i包装成Integer,因为StringBuilder的append方法有一个重载,其方法参数是int),

 public class StringConcatenation {
    private static int i = 1;
 
    public static void main(String... args) {
        System.out.println("literal" + i);
    }
}

Java如何初始化对象

当一个对象被创建之后,虚拟机会为其分配内存,主要用来存放对象的实例变量及其从超类继承过来的实例变量(即使这些从超类继承过来的实例变量有可能被隐藏也会被分配空间)。在为这些实例变量分配内存的同时,这些实例变量也会被赋予默认值。

那些对象的方法呢???????????


关于实例变量隐藏

只有成员变量(不管是不是静态)和静态方法可以被隐藏。

对于隐藏的诀窍:(谨记这两点就可以解决所有情况)
父类操作父类的实例变量,子类操作子类的实例变量
父类的方法操作父类的实例变量,子类的方法操作子类的实例变量

class Animal {
    public double weight;//隐藏变量一
    public String name="animal";//隐藏变量二
    
    public void setOldWeight(double b) {
        weight = b;
    }
    //这个方法是让子类继承的
    public String setName(String name) {
        return this.name = name;
    }

}

class Cat extends Animal {
    public int weight;
    public String name = "cat";

    public void setNewWeight(int b) {
        weight = b;
    }

    public double getOldWeight() {
        return super.weight;
    }

    public int getNewWeight() {
        return this.weight;
    }
    
    public String getThisName(){
        return this.name;
    }
    public String getSuperName(){
        return super.name;
    }

}

public class HideTest2 {
/**对于隐藏的诀窍诀窍诀窍
 * 父类操作父类的实例变量,子类操作子类的实例变量
 * 父类的方法操作父类的实例变量,子类的方法操作子类的实例变量
 * 
 * @param args
 */
    public static void main(String[] args) {
        // TODO Auto-generated method stub
        Cat cat = new Cat();
        
        //验证一 ***父类与子类分别有自己不同的方法***
        cat.setOldWeight(10.01);//打印的是10.01    父类的方法操作父类的实例变量
        System.out.println(cat.getOldWeight());
        cat.setNewWeight(10);//打印的是10   子类的方法操作子类的实例变量
        System.out.println(cat.getNewWeight());
        
         //验证二   *子类调用从父类继承的方法*
        cat.setName("human");//子类调用从父类继承的方法,但实际上操作父类的是实例变量。因为父类的方法操作父类的实例变量
        System.out.println("cat.name "+cat.name);//打印的是cat
        System.out.println("Cat的名字:"+cat.getThisName());//打印的是cat
        System.out.println("Animal的名字:"+cat.getSuperName());//打印的是human
        
        //验证三  ##上转型对象##
        Animal animal=new Cat();
        System.out.println(animal.flag);//打印的是false  父类操作父类的实例变量
    }

}

关于Java类成员(成员变量和方法)的覆盖与隐藏归纳

点击请看博客链接

Java的构造函数

Java要求一个对象被初始化之前,其超类也必须被初始化,这一点是在构造函数中保证的。Java强制要求Object对象(Object是Java的顶层对象,没有超类)之外的所有对象构造函数的第一条语句必须是超类构造函数的调用语句或者是类中定义的其他的构造函数,如果我们即没有调用其他的构造函数,也没有显式调用超类的构造函数,那么编译器会为我们自动生成一个对超类构造函数的调用指令
  
  更多详细信息请见博客


关于类的加载的问题

可以看一下毕向东的day6-06视频

  1. 只要经过java xxx.class运行命令后,并且用到了类中的内容时(例如 用构造函数new对象或者用类名调用静态方法,因为构造函数与静态方法是类里面的内容),类就会加载进内存了,所以不一定要new 对象才会加载类进内存。

  2. 静态的东西,不管是静态代码块,还是静态变量,静态方法都是优先于主函数调用,要记住主函数虽然是静态的,但它是特殊的静态,只是程序的入口,java虚拟机是将类加载进内存后,才回去调用主函数作为程序入口去执行程序。


当程序主动使用某个类时,若该类还没加载到内存中,系统会通过加载,链接,初始化3个操作对类进行初始化。
类字面常量”,class”创建Class对象的引用时,不会自动地初始化该Class对象,准备工作包含3个步骤:
1.加载:由类加载器执行,该步骤查找字节码,并从这些字节码中创建一个Class对象
2.链接:在链接阶段将验证类中的字节码,为静态域分配存储空间,并且如果必需的话,将解析这个类创建的对其他类的所有引用。
3.初始化:如果该类有超类,则对其初始化,执行静态初始化器和静态初始化块


1.创建类的实例
2.访问类或接口的静态变量(static final常量除外,static final变量可以)
3.调用类的静态方法
4.反射(Class.forName(packageName.className))
5.初始化类的子类(子类初始化问题:满足主动调用,即访问子类中的静态变量、方法,否则仅父类初始化)
6.java虚拟机启动时被标明为启动类的类
注:加载顺序:启动类的static block最先加载

我们需要明白在JAVA中任何class都要装载在虚拟机上才能运行,而forClass就是装载类用的,这是要和new不一样,要分清楚哦。
A a = (A)Class.forName(“package.A”).newInstance();和 A a = new A;是等价的。
记住一个概念,静态代码是和class绑定的,class装载成功就表示执行了你的静态代码,而且以后不会再走这套静态代码了。
Class.forName(xxx.xx.xx)的作用是要求JVM查找并加载指定的类,也即是说JVM会执行该类的静态代码段。

关于类的加载与反射的介绍的博客

对象的初始化过程的内存顺序

分析Person person=new Person();这句话都做了什么

在栈内存中分配空间给主函数main()[因为肯定是先有主函数],然后分配内存空间给对象变量person
1.类的加载进入内存
2.执行静态代码块初始化(凡是静态的都加载进方法区)
3.静态变量加载进方法区并且初始化,加载静态方法的代码进入方法区但是还没调用(凡是静态的都加载进方法区);在静态方法都加载进方法区后,就加载非静态的方法进入方法区
【静态变量优先于静态方法执行,所以静态变量也优先于main函数执行】
4.先在堆内存中开辟空间,给对象分配内存空间,即给类里面的属性分配空间(注意是现在才给对象分配空间)
      疑问:类的属性分配了地址,那类的方法呢?
5.实例变量的默认初始化
6.实例变量可见初始化
7.构造代码块初始化
8.构造函数初始化
9.将堆内存的对象实例【即new Person()】地址[即引用]赋值给栈内存的对象[即person]

成员变量(在类里面声明)在声明时可以不给它初始化,编译器会自动给这个成员变量初始化,但局部变量(在方法里面声明)在声明时一定要给它初始化,因为编译器不会自动给局部变量初始化,任何变量在使用之前必须对它进行初始化。

关于执行person.setName("zhangsan");这句话时内存中都发生什么事,请见毕向东的day6-08视频,这里不做过多叙述

你可能感兴趣的:(# Java对象初始化)