数据结构_内存表示

对象与引用对象

  • 对象引用

    对于一个简单的例子:
    Person p=new Person("Jim",18);
    内存表示:

    1

    栈内存中p并不是一个对象本身,而只是一个指向对象的指针(引用).用等号赋值时,是将右侧堆内存中对象的地址赋予给对象引用.

  • 引用赋值

    若继续进行操作的话:

     Person p2=p;
     p2.name="Lucy";
    

    那么p的名字也受影响变成了lucy.

    2

    当我们将一个引用赋值给另一个引用时,我们实际上复制的是对象的地址。两个引用将指向同一对象.其中一个引用修改对象,另一个引用也看得见.

  • 垃圾回收

    一个对象可以有多个引用,一个引用不可以有多个对象.同时,一个对象要是没有引用,会被垃圾回收.

    3

  • 参数传递

    java只有值传递,java只有值传递,java只有值传递!
    如果传递参数类型是基本数据类型,那么传过来的就是这个参数的一个副本,也就是这个原始参数的值,这个与c的传值是一样的。如果在函数中改变了副本的 值不会改变原始的值.
    如果传递参数类型是引用类型,那么传过来的就是这个引用参数的副本,这个副本存放的是参数的地址。如果在函数中没有改变这个副本的地址,而是改变了地址中的 值,那么在函数内的改变会影响到传入的参数。如果只是改变地址,指针指来指去,而不去改变值,是不会影响参数的值。
    引用一段Clara_babybear代码作为例子:
      public class ParamTest {
     public static void main(String[] args){
           /**
            * Test 1: Methods can't modify numeric parameters
            */
          System.out.println("Testing tripleValue:");
           double percent = 10;
           System.out.println("Before: percent=" + percent);
           tripleValue(percent);
           System.out.println("After: percent=" + percent);
    
           /**
            * Test 2: Methods can change the state of object parameters
            */
           System.out.println("\nTesting tripleSalary:");
           Employee harry = new Employee("Harry", 50000);
           System.out.println("Before: salary=" + harry.getSalary());
           tripleSalary(harry);
           System.out.println("After: salary=" + harry.getSalary());
    
           /**
            * Test 3: Methods can't attach new objects to object parameters
            */
           System.out.println("\nTesting swap:");
           Employee a = new Employee("Alice", 70000);
           Employee b = new Employee("Bob", 60000);
           System.out.println("Before: a=" + a.getName());
           System.out.println("Before: b=" + b.getName());
           swap(a, b);
           System.out.println("After: a=" + a.getName());
           System.out.println("After: b=" + b.getName());
     }
    
     private static void swap(Employee x, Employee y) {
         Employee temp = x;
         x=y;
         y=temp;
         System.out.println("End of method: x=" + x.getName());
         System.out.println("End of method: y=" + y.getName());
     }
    
     private static void tripleSalary(Employee x) {
         x.raiseSalary(200);
         System.out.println("End of method: salary=" + x.getSalary());
     }
    
     private static void tripleValue(double x) {
         x=3*x;
         System.out.println("End of Method X= "+x);
     }
    

}

输出结果  
```java
 Testing tripleValue:
 Before: percent=10.0
 End of Method X= 30.0
 After: percent=10.0

 Testing tripleSalary:
 Before: salary=50000.0
 End of method: salary=150000.0
 After: salary=150000.0

 Testing swap:
 Before: a=Alice
 Before: b=Bob
 End of method: x=Bob  //可见引用的副本进行了交换
 End of method: y=Alice
 After: a=Alice  //引用本身没有交换
 After: b=Bob

对象内存表示

  • 对象内存基础

    java中内存主要包含4块,即heap(堆内存)、stack(栈内存)、data segment(静态变量或是常量存放区)、codesegment(方法区).
    堆内存: 存放的是new出的对象,new出的对象只包含成员变量.
    栈内存: 存放的是局部成员变量。对于基本的数据类型存放的是基本变量的值,而对于对象变量,存放的是堆内存的地址。
    静态,常量区: 存放的是静态变量(类变量)或是常量(字符串常量和基本类型常量(public static final))。
    方法区: 存放的是对象的方法。因此即使new出多个对象也是只存在一个方法。
    对于一段简单的代码:
 class Student{
   private String name;
   private int id;
   private Class class;
   public void study(){};
 }
 class Class{
   private String name;
   private int id;
   public void exam(){};
 }
 public class MainClass(){
     public static void main(String[] args){
       Student s=new Student();
       s.name="Jim";
       s.id=001;
       s.study();
       Student s2=new Student();
       s2.name="Lucy"
       Class c=new Class();
       c.name="三班";
       c.id=03;
       s.class=c;
       c.exam();
     }
 }

关于该段内存表示:

4

考虑一些复杂的情况:

 class Student{
   private String name;
   private int id;
   private Class class;
   public Student(String iname,int iid){
     name=iname;
     id=iid;
   }
   public void study(int i){
     id=i;
     class.exam();
   };
 }
 class Class{
   private String name;
   private int id;
   public void exam(){};
 }
 public class MainClass(){
     public static void main(String[] args){
       Student s=new Student("Jim",001);
       Class c=new Class();
       c.name="三班";
       c.id=03;
       s.class=c;
       s.study(010);
     }
 }

内存表示:

5

形参,局部变量基本类型的值都是在栈内存中

  • 关于继承与多态内存表示

一个简单的例子:

 class Animal(){
   private String name;
   public void eat(){}
   public void run(){
     eat();
   }
 }
 class Birds extends Animal(){
   private Wing wing;
   public void eat(){}
 }
 public MainClass(){
    public static void main(String[] args){
    Animal a=new Birds();
    a.eat();
    a.run();
    }
 }
6

分析方法调用过程:
创建的对象a指向Birds类,this层为Birds类那层.
a.eat()方法从this层调起,调用Bird类的eat()方法.
a.run()方法从this层调起,没有则去调Animal的run()方法.这时候run()方法要调用eat()方法,同样从this层调起,调用Birds类的eat()方法.

  • 串的内存表示

    String是一个特殊的包装类数据。可以用String str = new String("abc");String str = "abc"; 两种的形式来创建.
    第一种用new创建的对象会放在堆中,每调用一次会创建一个新的对象.
    第二种先在栈中创建一个对 String类的对象引用变量str,然后通过符号引用去字符串常量池 里找有没有"abc",如果没有,则将"abc"存放进字符串常量池 ,并令str指向”abc”,如果已经有”abc” 则直接令str指向“abc”。
    值得一提的是:

    1. new String()和new String("")都是申明一个新的空字符串,是空串不是null;
    2. "fredal"=="fre"+"dal";
      "fredal"!="fre"+new Stirng("dal");
      String s="a";String s2="b";"ab"!=s+s2
    3. equals对于String简单来说就是比较两字符串的Unicode序列是否相当;而==是比较两字符串的地址是否相同,也就是是否是同一个字符串的引用。
    4. String是不可变的,所有会产生很多临时变量,可以采用StringBuffer,StringBuilder.
    5. 关于final,例如final String ia="a",由于对于final修饰的变量,它在编译时被解析为常量值的一个本地拷贝存储到自己的常量池中或嵌入到它的字节码流中,用的时候ia和"a"没啥区别.2的第三个加了final就是相等的.
  • 数组内存表示

    首先数组的初始化有静态初始化和动态初始化,前者类似于String[] names = { "Jim", "Lili", "Lucy" }这种,后者则是String[] names = new String[4];这种.
    根据数组的存储对象的类型呢也有两种,一种存基本数据类型,一种引用数据类型.
    当然还有一维数组和多维数组的区别.
    一股脑举个例子:

      String[] names={"Jim", "Lili", "Lucy"};
     
     int [] number=new int[3];
     for(int i=1;i<=3;i++){number[i-1]=i}
     
     Animal[] animal = new Animal[2];
     Animal cat = new Animal("cat", 1);
     Animal dog = new Animal("dog", 2);
     animal[0] = dog;
     animal[1] = cat;
     
     int[][] number2=new int[2][];
     number2[0]=new int[2];
     number2[1]=new int[3];
    
    7

    更多文章与相关下载请看扩展阅读

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容