Java内部类 常用类

打印 上一主题 下一主题

主题 2038|帖子 2038|积分 6114

马上注册,结交更多好友,享用更多功能,让你轻松玩转社区。

您需要 登录 才可以下载或查看,没有账号?立即注册

x
内部类和常用类

内部类

概念:在一个类的内部再界说一个完整的类
  1. package com.object.demo01;
  2. public class Body {
  3.     private String name;
  4.     class Header{
  5.         //内部类也会生成一个class文件 编译之后可以生成独立的字节码文件
  6.         //内部类可为外部类提供必要的功能组件
  7.         public void show() {       
  8.             System.out.println(name);   //内部类可以直接访问外部类的私有成员,而不破坏封装
  9.         }
  10.     }
  11. }
复制代码
成员内部类
  1. package com.object.demo02;
  2. //外部类
  3. public class Outer {
  4.     //实例变量
  5.     private String name = "张三";
  6.     private int age = 18;
  7.     //内部类 与实例变量、实例方法同级别的类
  8.     class Inner {
  9.         private String address = "北京";
  10.         private String phone = "110";
  11.         private String name =  "李四";
  12.         
  13.         //private static int scores = 86; 内部类不能定义静态成员
  14.         private static final int scores = 86; //但是可以定义静态常量
  15.         //方法
  16.         public void show() {
  17.             //打印外部类的属性
  18.             System.out.println(name);   //内部类属性和外部类属性名字相同,优先打印内部类
  19.             System.out.println(Outer.this.name);   
  20.             System.out.println(age);    //可以在age前加上Outer.this这样更清晰 可读性
  21.             //打印内部类的属性
  22.             System.out.println(address);        //同理可以加上this.
  23.             System.out.println(phone);          //同理加上this.
  24.         }
  25.     }
  26. }
复制代码
  1. package com.object.demo02;
  2. public class TestOuter {
  3.     public static void main(String[] args) {
  4.         //内部类的创建必须依赖外部类对象
  5. //        //1.先创建一个外部类对象
  6. //        Outer outer = new Outer();
  7. //        //2.创建一个内部类对象
  8. //        Outer.Inner inner = outer.new Inner();
  9.         //一步到位
  10.         Outer.Inner inner = new Outer().new Inner();
  11.         inner.show();
  12.     }
  13. }
复制代码
静态内部类

不依赖内部类对象,可直接创建或通过类名访问,可声明静态成员
  1. package com.object.demo03;
  2. //外部类
  3. public class Outer {
  4.     private String name = "xxx";
  5.     private int age = 18;
  6.     //静态内部类:和外部类相同
  7.     static class Inner {    //只有内部类才可以用static修饰,正常的类不能用static
  8.         private String address = "上海";
  9.         private String phone = "110";
  10.         //静态成员
  11.         private static int count = 10000;
  12.         public void show() {
  13.             //如何调用外部类的属性呢
  14.             //System.out.println(name); 不对
  15.             Outer outer = new Outer();
  16.             System.out.println(outer.name);
  17.             System.out.println(outer.age);
  18.             //调用静态内部类的属性和方法
  19.             System.out.println(address);
  20.             System.out.println(phone);
  21.             //调用静态内部类的静态属性
  22.             System.out.println(Inner.count);
  23.         }
  24.     }
  25. }
复制代码
  1. package com.object.demo03;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         //静态内部类相当于一个外部类
  5.         //直接创建静态内部类对象
  6.         Outer.Inner inner  =new Outer.Inner();  //没有new Outer();
  7.         //调用方法
  8.         inner.show();
  9.     }
  10. }
复制代码
局部内部类

界说在外部类方法中,作用范围仅限于当前方法
  1. package com.object.demo04;
  2. public class Outer {
  3.     private String name = "张三";
  4.     private int age = 18;
  5.     public void show() {
  6.         //定义局部变量
  7.         final String address = "深圳";        //局部内部类访问外部类当前方法的局部变量时,因无法保障变量的声明周期与自身相同,变量修饰必须为final
  8.         //局部内部类:注意不能加任何访问修饰符
  9.         class Inner {   //在堆里 不会消失
  10.             private String phone = "110";
  11.             private String email = "111@qq.com";
  12.            // private static int count = 0; 局部内部类里不能有静态属性,同样可以有静态常量
  13.             
  14.             public void show2() {
  15.                 //访问外部类的属性 但是show被static修饰就不能访问
  16.                 //因为static 和类一起加载 外部类属性还没加载出来
  17.                 System.out.println(name);   //可以加上Outer.this. 可读性强
  18.                 System.out.println(age);
  19.                 //访问内部类属性
  20.                 System.out.println(phone);  //可以加上this.
  21.                 System.out.println(email);  //同上
  22.                
  23.                 //访问局部变量 jdk1.7要求是常量 jdk1.8自动加上final
  24.                 System.out.println(address);
  25.             }
  26.         }
  27.         
  28.         //创建内部类对象
  29.         Inner inner = new Inner();
  30.         inner.show2();
  31.     }
  32. }
复制代码
  1. package com.object.demo04;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         Outer outer = new Outer();
  5.         outer.show();
  6.     }
  7. }
复制代码
匿名内部类


  • 没有类名的局部内部类
  • 必须继续一个父类或者实现一个接口
  • 优点:减少代码量
  • 缺点:可读性较差
  1. package com.object.demo05;
  2. //接口
  3. public interface USB {
  4.     //定义一个服务方法
  5.     void service();
  6. }
复制代码
  1. package com.object.demo05;
  2. public class Mouse implements USB {
  3.     @Override
  4.     public void service() {
  5.         System.out.println("连接成功,鼠标开始工作");
  6.     }
  7. }
复制代码
  1. package com.object.demo05;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         //创建接口类型变量
  5.         //USB usb = new Mouse();
  6.         //usb.service();
  7.         //局部内部类
  8.     class Fan implements USB {
  9.         @Override
  10.         public void service() {
  11.             System.out.println("连接成功,风扇开始工作");
  12.         }
  13.     }
  14.         USB usb = new Fan();
  15.         usb.service();
  16.     }
  17. }
复制代码
这是使用局部内部类。局部内部类只使用一次,可以用匿名内部类进行优化->
  1. package com.object.demo05;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         //使用匿名内部类创建对象(相当于创建了一个局部内部类)
  5.         // USB usb = new USB();  报错-接口不能实例化
  6.         USB usb = new USB() {
  7.             @Override
  8.             public void service() {
  9.                 System.out.println("连接成功,风扇开始工作");
  10.             }
  11.         };
  12.         usb.service();
  13.         //使用匿名内部类优化
  14.     }
  15. }
复制代码
常用类

Object类

在java.lang包下
超类,所有对象都继续这个类的方法
getClass()方法

返回class类型
应用:判断两个引用中实际存储对象类型是否一致
  1. package com.object.demo06;
  2. public class Student {
  3.     private String name;
  4.     private int age;
  5.    
  6.     public Student(String name, int age) {
  7.         this.name = name;
  8.         this.age = age;
  9.     }
  10.     public String getName() {
  11.         return name;
  12.     }
  13.     public void setName(String name) {
  14.         this.name = name;
  15.     }
  16.     public int getAge() {
  17.         return age;
  18.     }
  19.     public void setAge(int age) {
  20.         this.age = age;
  21.     }
  22. }
复制代码
  1. package com.object.demo06;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         Student s1 = new Student("aaa", 20);
  5.         Student s2 = new Student("bbb", 22);
  6.         //判断s1和s2是不是同一个类型
  7.         Class class1 = s1.getClass();
  8.         Class class2 = s2.getClass();
  9.         if (class1 == class2) {
  10.             System.out.println("s1 is the same class");
  11.         } else {
  12.             System.out.println("s1 is not the same class");
  13.         }
  14.     }
  15. }
复制代码
hashCode()方法

返回int类型
哈希值根据对象的地址字符串数字使用hash算法计算出来的int类型的数值
一般情况下雷同对象返回雷同哈希码
以上节课的代码为例
  1. package com.object.demo06;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         Student s1 = new Student("aaa", 20);
  5.         Student s2 = new Student("bbb", 22);
  6.         
  7.         //hashCode方法
  8.         System.out.println(s1.hashCode());        //开辟第一块空间
  9.         System.out.println(s2.hashCode());        //开辟第二块空间
  10.         Student s3 = s1;
  11.         System.out.println(s3.hashCode());        //将第一块空间赋予它
  12.                 //1、3相等,1、2不等
  13.     }
  14. }
复制代码
toString()方法

返回值类型String
可以根据程序需求重写该方法
  1. package com.object.demo06;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         Student s1 = new Student("aaa", 20);
  5.         Student s2 = new Student("bbb", 22);
  6.         
  7.         //3.toString
  8.         System.out.println(s1.toString());       
  9.         System.out.println(s2.toString());
  10.     }
  11. }
复制代码
得到的是@+16进制哈希值
想要看到具体的属性就要对toString进行重写
  1. package com.object.demo06;
  2. public class Student {
  3.     private String name;
  4.     private int age;
  5.     public Student(String name, int age) {
  6.         this.name = name;
  7.         this.age = age;
  8.     }
  9.     public String getName() {
  10.         return name;
  11.     }
  12.     public void setName(String name) {
  13.         this.name = name;
  14.     }
  15.     public int getAge() {
  16.         return age;
  17.     }
  18.     public void setAge(int age) {
  19.         this.age = age;
  20.     }
  21.         //快捷键alt+ins
  22.     @Override
  23.     public String toString() {
  24.         return "Student [name=" + name + ", age=" + age + "]";
  25.     }
  26. }
复制代码
equals()方法

返回布尔类型
  1. package com.object.demo06;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         Student s1 = new Student("aaa", 20);
  5.         Student s2 = new Student("bbb", 22);
  6.         
  7.         //4.equals方法:判断两个对象是否相等
  8.         System.out.println("-----------------------------------");
  9.         System.out.println(s1.equals(s2));
  10.         Student s4 = new Student("小明",17);
  11.         Student s5 = new Student("小明", 17);
  12.         System.out.println(s4.equals(s5));  //虽然两者的属性相同,但是分别开辟了一个内存空间 false
  13.     }
  14. }
复制代码
要让属性雷同返回true就要对equals进行重写
  1. @Override
  2.     public boolean equals(Object obj) {
  3.         //1.判断两个对象是否是同一个引用
  4.         if (this == obj) {
  5.             return true;
  6.         }
  7.         //2.判断obj是否为null
  8.         if (obj == null) {
  9.             return false;
  10.         }
  11.         //3.判断是否为同一个类型
  12. //        if(this.getClass() == obj.getClass()) {
  13. //            return true;
  14. //        }
  15.         if (obj instanceof Student) {    //instanceof判断对象是否是某种类型
  16.             //4.强制类型转换
  17.             Student s = (Student) obj;
  18.             //5.比较属性
  19.             if (this.name.equals(s.getName()) && this.age == s.getAge()) {
  20.                 return true;
  21.             }
  22.         }
  23.         return false;
  24.     }
复制代码
finalize()方法

该方法一般情况下,程序员不会调用

  • 当对象被判断为垃圾对象时,由JVM自动调用此方法,用以标志垃圾对象,进入回收队列
  • 垃圾对象:没有有效引用指向此对象时,为垃圾对象
  • 垃圾回收:由GC销毁垃圾对象,释放数据存储空间
  • 自动回收机制:JVM的内存耗尽,释放数据存储空间
  • 手动回收机制:使用System.gc(),关照JVM执行垃圾回收
  1. @Override
  2.     protected void finalize() throws Throwable {
  3.         System.out.println(this.name +"对象被回收了");
  4.     } //对finalize方法进行重写
复制代码
  1. package com.object.demo06;
  2. public class Test2 {
  3.     public static void main(String[] args) {
  4.         Student s1 = new Student("aaa", 20);
  5.         new Student("bbb", 20);
  6.         //回收垃圾
  7.         System.gc();
  8.         System.out.println("回收垃圾");        //aaa正常,bbb被回收
  9.     }
  10. }
复制代码
包装类


  • 根本数据所对应的引用数据类型
根本数据类型包装类booleanBooleanbyteBytecharCharactershortShortintIntegerlongLongfloatFloatdoubleDouble类型转换与装箱拆箱

栈里面的对象拿到堆里面:装箱
堆里面的对象拿到栈里面:拆箱
  1. package com.object.demo07;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         //类型转换:装箱:基本类型转成引用类型的过程
  5.         int num1 = 18;  //基本类型 栈
  6.         //使用Integer类创建对象 下面两种方式都行
  7.         Integer integer1 = new Integer(num1);
  8.         Integer integer2 = Integer.valueOf(num1);
  9.         System.out.println("装箱");
  10.         System.out.println(integer1);
  11.         System.out.println(integer2);
  12.         //类型转换:拆箱:引用类型转成基本类型
  13.         Integer integer3 = new Integer(100);
  14.         int num2 = integer3.intValue();
  15.         System.out.println("拆箱");
  16.         System.out.println(num2);
  17.         //JDK1.5以后,提供自动装箱和拆箱
  18.         int age = 18;
  19.         //自动装箱
  20.         Integer integer4 =age;
  21.         System.out.println("自动装箱");
  22.         System.out.println(integer4);
  23.         //自动拆箱
  24.         int age2 =  integer4;
  25.         System.out.println("自动拆箱");
  26.         System.out.println(age2);
  27.     }
  28. }
复制代码
根本类型转换
  1. package com.object.demo07;
  2. public class Test {
  3.     public static void main(String[] args) {
  4.         
  5.         //基本类型转成字符串
  6.         int n1 = 15;
  7.         //1.使用+号
  8.         String s1 = n1 + "";
  9.         //2.使用Integer中的toString()方法
  10.         String s2 = Integer.toString(n1);
  11.         String s3 = Integer.toString(n1,16);    //toString重载方法,将它转成指定进制
  12.         System.out.println(s1);
  13.         System.out.println(s2);
  14.         System.out.println(s3);
  15.         //字符串转成基本类型
  16.         String str = "150";
  17.         //使用Integer.parseXXX();
  18.         int n2 = Integer.parseInt(str);
  19.         System.out.println(n2);
  20.         //boolean字符串形式转成基本类型,"true"-->true,只要这个字符串不是true就是false
  21.         String str2 = "true";
  22.         boolean b1 = Boolean.parseBoolean(str2);
  23.         System.out.println(b1);
  24.     }
  25. }
复制代码
整数缓冲区

Java预先创建了256个常用的整数包装类型对象Integer
我们接下来从一个程序来加深这句话的理解
  1. package com.object.demo07;
  2. public class Test2 {
  3.     public static void main(String[] args) {
  4.         //面试题
  5.         Integer integer1 = new Integer(100);
  6.         Integer integer2 = new Integer(100);
  7.         System.out.println(integer1 == integer2);   //引用类型在堆内存开辟了两个不同空间 false
  8.         Integer integer3 = 100; //自动装箱  即Integer integer3 = Integer.valueOf(100) 后续同理
  9.         Integer integer4 = Integer.valueOf(100); //自动装箱
  10.         System.out.println(integer3 == integer4);
  11.         //integer3,integer4都是引用类型,答案也是false吗? 结果是true
  12.         Integer integer5 = 200;
  13.         Integer integer6 = 200;
  14.         System.out.println(integer5 == integer6);
  15.         //这三行和上面三行几乎没差别 结果却相反 是false
  16.     }
  17. }
复制代码
自动装箱是使用Integer.valueOf()
问题就出在这个方法,ctrl+鼠标看valueOf()源码
[code]public static Integer valueOf(int i) {        if (i >= IntegerCache.low && i
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

杀鸡焉用牛刀

论坛元老
这个人很懒什么都没写!
快速回复 返回顶部 返回列表