培训第2个月的第24天----一些基础知识点总结

              培训还有最后一天,也就是今天是倒数第二天,后天考核,但是不怎么会,我决定明天再看考核的内容。今天呢,老师讲了些东西,可是我没听,有那么点小尴尬,因为自己有一些不懂的东西,想弄懂。

             今天也会说一些知识点,其中就包括昨天没有解答上来的那个问题----数值在 -128到127之间的Integer类型对象创建的问题。

             下面就说一下我今天总结的内容,以反问的形式。

一.局部变量的修饰符问题

  1.局部变量能使用权限修饰符(private,dafault,protected,public)修饰吗?

             答:权限修饰符可以修饰成员变量和成员方法,但是不可以修饰局部变量。因为局部变量定义在方法体中,它的作用范围只在该方法的方法体内有效。方法体在类中,就算最小的权限private修饰的成员在本类也是可以被访问的。所以我们可以这样理解,方法体的 “权限” 比private还要小,因为只要在本类中就可以访问private修饰的成员,而想访问局部变量,只能在这个方法体内进行访问,出了方法体就访问不到这个局部变量了。所以既然方法体的 “权限” 比private还小,那么在局部变量前添加访问权限修饰符显然是没有意义的。

  2.局部变量能够被static来修饰吗?

             答:是不能的,因为static修饰的东西说明这个东西是属于类的。而局部变量是属于代码块的,出了代码块它就不能被访问到。即使创建对象,它也不能通过对象.  的方法访问,也不能通过类名. 的方式访问。所以说明它不属于类(不能属于类),也就不能被static来修饰了。

 3.局部变量可以使用final来修饰吗?

             答:当然可以了,如果不用final来修饰局部变量,那么这个局部变量就是一个局部变量,但是如果用final来修饰,那么这个局部变量就变成了一个局部常量,这并不影响什么。final的作用是,在修饰基本数据类型变量时,一旦变量初始化后,将不允许在次赋值,在修饰引用数据类型时,一旦引用指向了一个对象时,那么这个引用只能指向这个对象,而不能在次指向其他的对象。

  4.局部变量一定要初始化吗,为什么成员变量就可以不初始化?

             答:Java中的变量分为两种,一种是局部变量,一种是成员变量。其实对于变量,不管是局部变量还是成员变量,我们在使用它们之前都要进行初始化,因为例如输出语句,想打印一个局部变量a,在你没初始化的时候,它是没有值的,jvm不知道输出什么,你应该先给它进行初始化,告诉jvm你想要输出的确切值。当然了,如果你不使用局部变量,那么即使局部变量放在那里不初始化编译也是可以通过的。

                    局部变量需要我们手动初始化,因为当我们不给它初始化的时候,它就真的没有初始化。而成员变量则不同,对于基本数据类型,即使我们没有给它进行初始化,jvm也会按照对应的基本类型来给它赋值,进行初始化。对于引用数据类型,如果我们不进行初始化,那么jvm会给它初始化为空。

二.Integer对象的问题

              这个问题也就是昨天我没有回答上来的问题,今天看了一些博客,了解了一下。我们先来看一段代码。

package com.java.text;

public class TextInnerClass {
	   
   public static void main(String[] args) {
	   TextInnerClass a=new TextInnerClass();
	   a.text1();
	}
   public void text1() {	  
       // Integer f1 = 100, f2 = 100, f3 = 150, f4 = 150;   和下面的代码等同
	   Integer a = 100;
	   Integer b = 100;
	   Integer c = 225;
	   Integer d = 225;
	   
	   System. out.println( a == b);   //true
	   System. out.println( c == d);   //false
		

   }
}

                 为什么第一个为true,第二个为false呢?

                 当我们通过表达式创建一个Integer对象时,也就是我们给一个Integer赋予一个int类型的时候会调用Integer类的静态方法valueOf()(自动装箱)。上面的代码我也可以写成下面这样。

                   这里我们可以看一下静态方法 valueOf() 的源码:

                  这里看到如果传进来的int值出了这个边界,就new 一个Integer对象。而如果在这个边界内,就从cache数组中取出值来给这个Integer类型的引用,也就是这个 Integer引用指向这个数组的某一个值。那么边界到底是多少呢?

private static class IntegerCache {
        static final int low = -128;
        static final int high;
        static final Integer cache[];

        static {
            // high value may be configured by property
            int h = 127;
            String integerCacheHighPropValue =
                sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
            if (integerCacheHighPropValue != null) {
                try {
                    int i = parseInt(integerCacheHighPropValue);
                    i = Math.max(i, 127);
                    // Maximum array size is Integer.MAX_VALUE
                    h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
                } catch( NumberFormatException nfe) {
                    // If the property cannot be parsed into an int, ignore it.
                }
            }
            high = h;

            cache = new Integer[(high - low) + 1];
            int j = low;
            for(int k = 0; k < cache.length; k++)
                cache[k] = new Integer(j++);

            // range [-128, 127] must be interned (JLS7 5.1.7)
            assert IntegerCache.high >= 127;
        }

        private IntegerCache() {}
    }

               我对这部分代码也不是很懂,但是我知道它将  -128到127这个区间的值放到了cache数组中。

               通过这两个代码块我们就可以解释上面的问题了,当传入的int值为100时,这个int值在这个区间内,所以直接将cache数组中的100值取出来给这个索引a,也就是这个索引a指向这个cache数组的这个100元素所在的地址,同理b也一样,所以a==b为true。而当闯入的int值为255时,不在这个区间内,那么就不从这个数组中取值,而是通过new 操作符在堆中创建一个新的对象,所以c和d其实指向的是不同的对象,当然是false了。

                 我们在来看一段代码:

                 这个值应该是什么呢?

                 不管是通过new操作符创建的对象,还是通过表达式创建的对象,它们都是对象包装类Integer类型,所以当Integer类型和int类型进行比较时,我们先将 Integer 拆装为int类型,然后在比较值的大小。所以两个输出结果都是true。

                 java中还有与Integer类似的是Long类型,它也有一个缓存,在区间[-128,127]范围内获取缓存的值,而Long与long比较的时候先转换成long类型再做值的比较。而Double类型,它没有缓存,但是当Double与double比较的时候会先转换成double类型,再做值的比较。

三.数组和集合的存放位置

         1.数组?

                  答:数组在定义的时候就已经定义了要存放的数据类型,所以一种数组只能存放一种数据类型。按照这个分,数组可以分为两种:一个是基本数据类型数组,另一种是引用数据类型数组。当一个对象使用关键字“new”创建时,会在堆上分配内存空间,然后返回地址给这个对象的引用,这对数组来说也是一样的,因为数组也是一个对象;

        2.数组对象及其引用存放在内存中的哪里?

                   答:如上代码,让我们来调用方法m1,看看发生了什么:当m1被调用的时候,栈帧Frame-1被创建并push到栈中,同时局部变量i也在栈帧Frame-1内创建。然后,m2方法在m1方法内部被调用,栈帧Frame-2被创建并push到栈中,在m2方法中,一个新的对象A在堆中被创建,而它的引用则被put到栈帧Frame-2里;现在内存中堆和栈的大致情况如下图:

                数组同样是对象,所以数组对象以及引用在内存中的分布如上所示 ;

链接:http://www.programcreek.com/2013/04/what-does-a-java-array-look-like-in-memory/

        3.数组是什么数据结构?

                  答:从逻辑上说是线性表中的顺序存储结构,从物理上说是在物理内存里分配出一块连续的空间按顺序存储。

        4.ArrayList在内存中的存储方式(图解)?

                  例如要存入集合的对象如下:

                  

                   内存图解如下: 

                首先,在堆中创建一个集合对象(默认初始容量为10),地址指向 al ,在集合中添加元素并不是在集合中直接添加, 
而是在堆内存中重新为添加的对象分配空间, 其对象的地址保存在集合容器中。 这种存储方法类似于“拉链法”, 迭代器的原理也是如此。 

  四.  default

               default可以说是包访问权限,是一种默认的权限,在同一个包中可以访问default修饰的东西。以前知道这个,但是今天忽然发现了一个问题,就是在属性前(变量),方法,类前我们如果不写修饰符,那么  属性 ,方法,和类就是默认是default修饰的,也就是包访问权限。它们可以不写修饰符,但是不能写default(显示的写出)来修饰它们,这样会报错。

                 我现在怎么觉得数据结构挺重要的呢。。。。。。。       

猜你喜欢

转载自blog.csdn.net/qq_41160264/article/details/82776846
今日推荐