Java内部类 常用类
内部类和常用类内部类
概念:在一个类的内部再定义一个完整的类
package com.object.demo01;
public class Body {
private String name;
class Header{
//内部类也会生成一个class文件 编译之后可以生成独立的字节码文件
//内部类可为外部类提供必要的功能组件
public void show() {
System.out.println(name); //内部类可以直接访问外部类的私有成员,而不破坏封装
}
}
}成员内部类
package com.object.demo02;
//外部类
public class Outer {
//实例变量
private String name = "张三";
private int age = 18;
//内部类 与实例变量、实例方法同级别的类
class Inner {
private String address = "北京";
private String phone = "110";
private String name ="李四";
//private static int scores = 86; 内部类不能定义静态成员
private static final int scores = 86; //但是可以定义静态常量
//方法
public void show() {
//打印外部类的属性
System.out.println(name); //内部类属性和外部类属性名字相同,优先打印内部类
System.out.println(Outer.this.name);
System.out.println(age); //可以在age前加上Outer.this这样更清晰 可读性
//打印内部类的属性
System.out.println(address); //同理可以加上this.
System.out.println(phone); //同理加上this.
}
}
}package com.object.demo02;
public class TestOuter {
public static void main(String[] args) {
//内部类的创建必须依赖外部类对象
// //1.先创建一个外部类对象
// Outer outer = new Outer();
// //2.创建一个内部类对象
// Outer.Inner inner = outer.new Inner();
//一步到位
Outer.Inner inner = new Outer().new Inner();
inner.show();
}
}静态内部类
不依赖内部类对象,可直接创建或通过类名访问,可声明静态成员
package com.object.demo03;
//外部类
public class Outer {
private String name = "xxx";
private int age = 18;
//静态内部类:和外部类相同
static class Inner { //只有内部类才可以用static修饰,正常的类不能用static
private String address = "上海";
private String phone = "110";
//静态成员
private static int count = 10000;
public void show() {
//如何调用外部类的属性呢
//System.out.println(name); 不对
Outer outer = new Outer();
System.out.println(outer.name);
System.out.println(outer.age);
//调用静态内部类的属性和方法
System.out.println(address);
System.out.println(phone);
//调用静态内部类的静态属性
System.out.println(Inner.count);
}
}
}package com.object.demo03;
public class Test {
public static void main(String[] args) {
//静态内部类相当于一个外部类
//直接创建静态内部类对象
Outer.Inner inner=new Outer.Inner();//没有new Outer();
//调用方法
inner.show();
}
}局部内部类
定义在外部类方法中,作用范围仅限于当前方法
package com.object.demo04;
public class Outer {
private String name = "张三";
private int age = 18;
public void show() {
//定义局部变量
final String address = "深圳"; //局部内部类访问外部类当前方法的局部变量时,因无法保障变量的声明周期与自身相同,变量修饰必须为final
//局部内部类:注意不能加任何访问修饰符
class Inner { //在堆里 不会消失
private String phone = "110";
private String email = "111@qq.com";
// private static int count = 0; 局部内部类里不能有静态属性,同样可以有静态常量
public void show2() {
//访问外部类的属性 但是show被static修饰就不能访问
//因为static 和类一起加载 外部类属性还没加载出来
System.out.println(name); //可以加上Outer.this. 可读性强
System.out.println(age);
//访问内部类属性
System.out.println(phone);//可以加上this.
System.out.println(email);//同上
//访问局部变量 jdk1.7要求是常量 jdk1.8自动加上final
System.out.println(address);
}
}
//创建内部类对象
Inner inner = new Inner();
inner.show2();
}
}package com.object.demo04;
public class Test {
public static void main(String[] args) {
Outer outer = new Outer();
outer.show();
}
}匿名内部类
[*]没有类名的局部内部类
[*]必须继承一个父类或者实现一个接口
[*]优点:减少代码量
[*]缺点:可读性较差
package com.object.demo05;
//接口
public interface USB {
//定义一个服务方法
void service();
}package com.object.demo05;
public class Mouse implements USB {
@Override
public void service() {
System.out.println("连接成功,鼠标开始工作");
}
}package com.object.demo05;
public class Test {
public static void main(String[] args) {
//创建接口类型变量
//USB usb = new Mouse();
//usb.service();
//局部内部类
class Fan implements USB {
@Override
public void service() {
System.out.println("连接成功,风扇开始工作");
}
}
USB usb = new Fan();
usb.service();
}
}这是使用局部内部类。局部内部类只使用一次,可以用匿名内部类进行优化->
package com.object.demo05;
public class Test {
public static void main(String[] args) {
//使用匿名内部类创建对象(相当于创建了一个局部内部类)
// USB usb = new USB();报错-接口不能实例化
USB usb = new USB() {
@Override
public void service() {
System.out.println("连接成功,风扇开始工作");
}
};
usb.service();
//使用匿名内部类优化
}
}常用类
Object类
在java.lang包下
超类,所有对象都继承这个类的方法
getClass()方法
返回class类型
应用:判断两个引用中实际存储对象类型是否一致
package com.object.demo06;
public class Student {
private String name;
private int age;
public Student(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}package com.object.demo06;
public class Test {
public static void main(String[] args) {
Student s1 = new Student("aaa", 20);
Student s2 = new Student("bbb", 22);
//判断s1和s2是不是同一个类型
Class class1 = s1.getClass();
Class class2 = s2.getClass();
if (class1 == class2) {
System.out.println("s1 is the same class");
} else {
System.out.println("s1 is not the same class");
}
}
}hashCode()方法
返回int类型
哈希值根据对象的地址或字符串或数字使用hash算法计算出来的int类型的数值
一般情况下相同对象返回相同哈希码
以上节课的代码为例
package com.object.demo06;
public class Test {
public static void main(String[] args) {
Student s1 = new Student("aaa", 20);
Student s2 = new Student("bbb", 22);
//hashCode方法
System.out.println(s1.hashCode()); //开辟第一块空间
System.out.println(s2.hashCode()); //开辟第二块空间
Student s3 = s1;
System.out.println(s3.hashCode()); //将第一块空间赋予它
//1、3相等,1、2不等
}
}toString()方法
返回值类型String
可以根据程序需求重写该方法
package com.object.demo06;
public class Test {
public static void main(String[] args) {
Student s1 = new Student("aaa", 20);
Student s2 = new Student("bbb", 22);
//3.toString
System.out.println(s1.toString());
System.out.println(s2.toString());
}
}得到的是@+16进制哈希值
想要看到具体的属性就要对toString进行重写
package com.object.demo06;
public class Student {
private String name;
private int age;
public Student(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
//快捷键alt+ins
@Override
public String toString() {
return "Student ";
}
}equals()方法
返回布尔类型
package com.object.demo06;
public class Test {
public static void main(String[] args) {
Student s1 = new Student("aaa", 20);
Student s2 = new Student("bbb", 22);
//4.equals方法:判断两个对象是否相等
System.out.println("-----------------------------------");
System.out.println(s1.equals(s2));
Student s4 = new Student("小明",17);
Student s5 = new Student("小明", 17);
System.out.println(s4.equals(s5));//虽然两者的属性相同,但是分别开辟了一个内存空间 false
}
}要让属性相同返回true就要对equals进行重写
@Override
public boolean equals(Object obj) {
//1.判断两个对象是否是同一个引用
if (this == obj) {
return true;
}
//2.判断obj是否为null
if (obj == null) {
return false;
}
//3.判断是否为同一个类型
// if(this.getClass() == obj.getClass()) {
// return true;
// }
if (obj instanceof Student) { //instanceof判断对象是否是某种类型
//4.强制类型转换
Student s = (Student) obj;
//5.比较属性
if (this.name.equals(s.getName()) && this.age == s.getAge()) {
return true;
}
}
return false;
}finalize()方法
该方法一般情况下,程序员不会调用
[*]当对象被判定为垃圾对象时,由JVM自动调用此方法,用以标记垃圾对象,进入回收队列
[*]垃圾对象:没有有效引用指向此对象时,为垃圾对象
[*]垃圾回收:由GC销毁垃圾对象,释放数据存储空间
[*]自动回收机制:JVM的内存耗尽,释放数据存储空间
[*]手动回收机制:使用System.gc(),通知JVM执行垃圾回收
@Override
protected void finalize() throws Throwable {
System.out.println(this.name +"对象被回收了");
} //对finalize方法进行重写package com.object.demo06;
public class Test2 {
public static void main(String[] args) {
Student s1 = new Student("aaa", 20);
new Student("bbb", 20);
//回收垃圾
System.gc();
System.out.println("回收垃圾"); //aaa正常,bbb被回收
}
}包装类
[*]基本数据所对应的引用数据类型
基本数据类型包装类booleanBooleanbyteBytecharCharactershortShortintIntegerlongLongfloatFloatdoubleDouble类型转换与装箱拆箱
栈里面的对象拿到堆里面:装箱
堆里面的对象拿到栈里面:拆箱
package com.object.demo07;
public class Test {
public static void main(String[] args) {
//类型转换:装箱:基本类型转成引用类型的过程
int num1 = 18;//基本类型 栈
//使用Integer类创建对象 下面两种方式都行
Integer integer1 = new Integer(num1);
Integer integer2 = Integer.valueOf(num1);
System.out.println("装箱");
System.out.println(integer1);
System.out.println(integer2);
//类型转换:拆箱:引用类型转成基本类型
Integer integer3 = new Integer(100);
int num2 = integer3.intValue();
System.out.println("拆箱");
System.out.println(num2);
//JDK1.5以后,提供自动装箱和拆箱
int age = 18;
//自动装箱
Integer integer4 =age;
System.out.println("自动装箱");
System.out.println(integer4);
//自动拆箱
int age2 =integer4;
System.out.println("自动拆箱");
System.out.println(age2);
}
}基本类型转换
package com.object.demo07;
public class Test {
public static void main(String[] args) {
//基本类型转成字符串
int n1 = 15;
//1.使用+号
String s1 = n1 + "";
//2.使用Integer中的toString()方法
String s2 = Integer.toString(n1);
String s3 = Integer.toString(n1,16); //toString重载方法,将它转成指定进制
System.out.println(s1);
System.out.println(s2);
System.out.println(s3);
//字符串转成基本类型
String str = "150";
//使用Integer.parseXXX();
int n2 = Integer.parseInt(str);
System.out.println(n2);
//boolean字符串形式转成基本类型,"true"-->true,只要这个字符串不是true就是false
String str2 = "true";
boolean b1 = Boolean.parseBoolean(str2);
System.out.println(b1);
}
}整数缓冲区
Java预先创建了256个常用的整数包装类型对象Integer
我们接下来从一个程序来加深这句话的理解
package com.object.demo07;
public class Test2 {
public static void main(String[] args) {
//面试题
Integer integer1 = new Integer(100);
Integer integer2 = new Integer(100);
System.out.println(integer1 == integer2); //引用类型在堆内存开辟了两个不同空间 false
Integer integer3 = 100; //自动装箱即Integer integer3 = Integer.valueOf(100) 后续同理
Integer integer4 = Integer.valueOf(100); //自动装箱
System.out.println(integer3 == integer4);
//integer3,integer4都是引用类型,答案也是false吗? 结果是true
Integer integer5 = 200;
Integer integer6 = 200;
System.out.println(integer5 == integer6);
//这三行和上面三行几乎没差别 结果却相反 是false
}
}自动装箱是使用Integer.valueOf()
问题就出在这个方法,ctrl+鼠标看valueOf()源码
public static Integer valueOf(int i) { if (i >= IntegerCache.low && i
页:
[1]