遇一山,过一山,处处有风景;只要勇敢向前,一路尽是繁花盛开。 | (点击查看→)【测试干货】python/java自动化、持续集成、性能、测开、简历、笔试面试等

Java【第七篇】面向对象之类设计

Java类的继承

可以让一个类继承自另一个类,此时该类会继承另一个类中的属性和方法,这样可以少写很多代码

类继承语法规则

< 修饰符> class < 子类名称> [extends < 父类>] 
{
    <属性和方法的声明>
} 

类的继承

子类继承了父类,就继承了父类的方法和属性。(子类对象可以调用父类属性、方法)
在子类中,可以使用父类中定义的方法和属性,也可以创建新的数据和方法。
因而,子类通常比父类的功能更多,而是对父类的“扩展”。
在Java中,继承的关键字用的是“extends”,即子类不是父类的子集,而是对父类的“扩展”。

继承规则

Java只支持单继承,不允许多重继承
  一个子类只能有一个父类
  一个父类可以派生出多个子类
子类不能继承父类中私有的(private)的成员变量和方法,不能直接访问【可以调用父类的public或者protect方法去访问】

示例

父类

package com.uncleyong;

import java.util.Date;

public class Person {
    public String name;
    public int age;
    public Date birth;

    private String lover = "lucy";

    public String getInfo(){
        return "name: " + name + ", " +
                "age: " + age + ", " +
                "birth: " + birth+ ", " +
                "lover: " + lover;
    }

}

子类 

package com.uncleyong;


public class Student extends Person{
    public String school;
}

测试类

package com.uncleyong;

import java.util.Date;

public class TestPerson {
    public static void main(String[] args) {
    	Student student = new Student();
    	student.name = "Jerry";
    	student.birth = new Date();
    	student.age = 1;
    	student.school = "清华";
		System.out.println(student.getInfo());


        Person person = new Person();
		person.age = 1;
		person.birth = new Date();
		person.name = "Tom";

		System.out.println(person.getInfo());
    }
}  

访问控制

可以对Java类中定义的属性和方法进行访问控制----规定不同的保护等级: public、protected、default、private

default就是什么都不加
子类,就是可以跨包【因为子类可能跨包,所以,为了使父类中属性、方法在子类能访问,父类最严格的访问控制只能是protected】

//仅在类的内部可以访问.
private String email;

//在同一个包内该属性可以被访问.
String major;

//在子类中该属性可以被访问, 且该子类可以跨包(不在同一个包下)
//如果不是子类,跨包不能访问;如果不是子类,在同一个包下的类,也可以访问
protected int salary;

//访问权限最高, 无论是否在一个包内, 无论是否是子类都可以被访问.(在一个项目中都可以被访问)
public String name;
public int age;
public Date birth;

 

方法的重写、覆盖

在子类中可以根据需要对从父类中继承来的方法(构造方法不能被继承)进行改造—覆盖方法(方法的重置、重写),在程序执行时,子类的方法将覆盖父类的方法。

覆盖方法必须和被覆盖方法具有相同的方法名称、参数列表和返回值类型。【返回类型、方法名、参数列表必须完全相同,应用场景:继承,父类的成员方法只能被它的子类重写】

  如果参数个数不一样,就是方法的重载了(一个从父类继承的,一个自己类里写的,会根据调用方法时的传参,确定是调继承的,还是自己类里的),但是重载是不是要在一个类中的方法呢?【结果验证:可以在不同的类中,比如父类,虽然一个是父类,但是子类继承了,子类也就有了父类的方法】,见下面验证示例:

  返回值类型必须一样,否则两个方法不知道该调用哪个(一个从父类继承的,一个自己类里写的),方法名、参数列表都一样了,只有返回类型不一样,编译的时候报错返回类型不兼容

覆盖方法不能使用比被覆盖方法更严格的访问权限。

 

参数个数不一样,验证:

父类:Test.java

package com.test;

public class Test {
    public int i;
    public void t(){
        System.out.println("test");
    }
}

子类:Test2.java

package com.test;

public class Test2 extends Test{
    public void t(String s){  // 这个方法和父类t方法相比,多了一个形参,所以不是方法重写,而是方法重载
        System.out.println(s);
    }
    public static void main(String[] args) {
        Test2 test2 = new Test2();
        test2.t("qzcsbj");  // qzcsbj
        test2.t();  // test
    }
}

 

关于构造方法的重载、重写

父类构造方法不能被子类继承:因为如果子类继承了父类的构造方法,父类的构造在子类中不符合构造方法的规则,也不符合一般方法的规则,因为父类的构造在子类中没有返回类型,方法名也与子类的类名不相同。不符合java语法规范。继承就跟我们现实中的父子关系差不多,要有一个孩子对象那么就得先有一个父亲<会调用父类的构造方法>,所以会执行父类构造方法。另一种如果你想调用父类的带参数构造方法,那还得通过super关键字来调用<super后面括号中加上参数;如果父类构造是无参的,子类也默认调了,在子类的默认构造方法中有一句super();>。构造方法是不能继承的,想想如果能继承,那不是孩子也能构造父亲了;见下方验证示例

父类构造方法不能被子类重写(因为不能继承);

父类构造方法不能被子类重载(因为不能继承,子类就不具有这个方法;在java中类的构造函数与类的名称相同,不可能子类和父类使用相同的类名称,因此子类也就不能重载父类的构造函数,但子类可以通过super来调用父类的构造函数);

子类可以重载从父类继承过来的方法

子类可以重载自己默认的构造方法

子类可以重载父类的非构造方法:可以这样理解,子类继承了父的方法,子类本身也就有了这个方法。在子类写一个只多了一个参数其余都一样的方法,就是重载了继承的这个方法,实际也是在一个类中。

 

父类构造方法不能被子类继承,验证:

父类:A.java

package com.qzcsbj;

public class A {
    public A(){
        System.out.println("父类构造方法");
    }
    public void test(){
        System.out.println("父类A测试方法");
    }
}

子类:B.java

package com.qzcsbj;

public class B extends A {
    public int age;
    public B(){
        System.out.println("子类构造方法");
    }
    public void test(int a){
        System.out.println("子类B测试方法");
    }

    public static void main(String[] args) {
        B b = new B();
        b.test();
        b.test(1);
    }
}

输出结果:

父类构造方法
子类构造方法
父类A测试方法
子类B测试方法 

上面说明:父类中的构造方法是不能继承的,但是在实例化子类的时候会调用父类的构造方法,每个子类构造方法的第一条语句,都是隐含地调用super()

重写示例

package com.uncleyong;

/**
 * 定义一个ManKind类,包括
 * 成员变量 int sex 和 int salary;
 * 方法 void manOrWorman():根据sex的值显示“man”(sex==1)或者“women”(sex==0);
 * 方法 void employeed():根据salary的值显示“no job”(salary==0)或者“ job”(salary!=0)。
 */
public class ManKind {

    int sex;
    int salary;

    public void manOrWoman(){
        if(sex == 0){
            System.out.println("woman");
        }else if(sex == 1){
            System.out.println("man");
        }
    }

    public void employeed(){
        if(salary != 0){
            System.out.println("job");
        }else{
            System.out.println("no job");
        }
    }

}

  

package com.uncleyong;

/**
 * 定义类 Kids1 继承ManKind,并包括
 * 成员变量 int yearsOld;
 * 方法 printAge() 打印 yearsOld 的值。
 *
 * 在Kids1中重新定义employed() 方法,覆盖父类ManKind中定义的employed()方法,
 * 输出“Kids should study and no job.”
 */
public class Kids1 extends ManKind {

    int yearsOld;

    void printAge(){
        System.out.println("yearsOld: " + yearsOld);
    }

    // 方法重写
    public void employeed() {
        System.out.println("Kids should study and no job.");
    }

    //在Kids1类的main方法中实例化Kids1的对象 someKid,用该对象访问其父类的成员变量及方法。
    public static void main(String[] args) {
        Kids1 someKid = new Kids1();
        someKid.sex = 1;
        someKid.salary = 5000;
        someKid.yearsOld = 25;

        someKid.manOrWoman();
        someKid.employeed();
        someKid.printAge();
    }
}

 

super 关键字

super的功能

在Java类中使用super来引用父类的成分:

  super可用于访问父类中定义的属性
  super可用于调用父类中定义的成员方法(如下例子)
  super可用于在子类构造方法中调用父类的构造方法
  super的追溯不仅限于直接父类

例子:子类重写的方法中,调用父类中被重写的方法,需要用super

下面子类重写父类方法,会死循环(循环调用自己),导致栈溢出

package com.uncleyong;

public class Person {
    private String country = "China";

    public String getInfo(){
        return "country:" + country;
    }
}

  

public class Student extends Person{
    private String school = "清华大学";

    @Override
    public String getInfo() {
        return getInfo() + ", school:" + school;  // 正确写法return this.getInfo() + ", school:" + school;
    }
}

  

package com.uncleyong;

public class Test {
    public static void main(String[] args) {
        Student student = new Student();
        student.getInfo();
    }
}

  

 

 

构造方法不能继承

子类继承父类所有的成员变量和成员方法,但不继承父类的构造方法
在一个Java类中可以通过两种方式获得构造方法:
  使用系统默认的无参数构造方法
  显式定义一个或多个构造方法
一旦显式定义了构造方法,则系统不再提供默认构造方法

调用父类构造方法

在子类的构造方法中可使用super(参数列表)语句调用父类的构造方法;

如果子类的构造方法中没有显示地调用父类构造方法,也没有使用this关键字调用重载的其它构造方法(调用子类中重载子类的构造方法),则系统默认调用父类无参数的构造方法【可以在这个无参构造方法体中写内容,但是不能有参数】,见下方验证;

如果子类构造方法中既未显式调用父类构造方法,而父类中又没有无参的构造方法,则编译出错;

子类的构造方法中一定要用super方式调用父类的某一个构造方法,可以通过this间接方式调当前类的某一个构造方法(这个构造方法又通过super去调了父类的一个构造方法)
	间接方式
		子类构造方法中第一行this调用重载的构造方法,被调用的那个重载构造方法中第一行super调用父类构造方法
	直接方式
		子类构造方法中第一行直接super调用父类构造方法


父类只有有参构造
	方案一:子类的构造方法中第一行用super调用父类有参构造方法
	方案二:父类新增无参构造方法


父类中同时存在有参和无参构造方法:
	子类的构造方法的第一行可以调用父类的有参构造方法,
	也可以不调用(这样就默认调用父类无参构造方法):如果子类的构造方法中没有显示地调用父类构造方法,也没有使用this关键字调用重载的其它构造方法【调用子类中重载子类的构造方法】,则系统默认调用父类无参数的构造方法【可以在这个无参构造方法体中写内容,但是不能有参数】


super(…)和this(…)调用语句不能同时在一个构造函数中出现

super(…)或this(…)调用语句只能作为构造函数中的第一句出现

 

验证:子类调用父类构造函数

系统默认提供无参构造函数

public Animal(){
}

也可以自己在无参构造方法体中加内容,子类也可以调这个无参方法

public Animal(){
    System.out.println("父类无参构造方法");
}

  

父类

public class Animal {
    private String name;
    private int age;

    public Animal(){
        System.out.println("父类无参构造方法");
    }

//    public Animal(String name){
//        this.name = name;
//        System.out.println("父类有参构造方法,形参name: " + name);
//    }

    public String getInfor(){
        return "name: " + name;
    }
}

  

子类

public class Cat extends Animal {
    private String sex;

//    public Cat(){
//        super("tom");
//    }
}

  

测试类

public class Test {
    public static void main(String[] args) {
        Cat cat = new Cat();
        System.out.println(cat.getInfor());
    }
}

 

结果

 

如果父类有参构造方法,那么,可以两个解决方案:

方案一:如果父类只有有参构造方法,需要在子类的构造方法的第一行调用父类的有参构造方法

父类

public class Animal {
    private String name;
    private int age;

//    public Animal(){
//        System.out.println("父类无参构造方法");
//    }

    public Animal(String name){
        this.name = name;
        System.out.println("父类有参构造方法,形参name: " + name);
    }

    public String getInfor(){
        return "name: " + name;
    }
}

 

子类

public class Cat extends Animal {
    private String sex;

    public Cat(){
        super("tom");
    }
}

  

测试类

public class Test {
    public static void main(String[] args) {
        Cat cat = new Cat();
        System.out.println(cat.getInfor());
    }
}

 

结果

 

方案二:父类中同时存在有参和无参构造方法,子类的构造方法的第一行可以调用父类的有参构造方法,也可以不调用(这样就默认调用父类无参构造方法)

如果子类的构造方法中没有显示地调用父类构造方法,也没有使用this关键字调用重载的其它构造方法【调用子类中重载子类的构造方法】,则系统默认调用父类无参数的构造方法【可以在这个无参构造方法体中写内容,但是不能有参数】

父类

public class Animal {
    private String name;
    private int age;

    public Animal(){
        System.out.println("父类无参构造方法");
    }

    public Animal(String name){
        this.name = name;
        System.out.println("父类有参构造方法,形参name: " + name);
    }

    public String getInfor(){
        return "name: " + name;
    }
}

  

子类

public class Cat extends Animal {
    private String sex;

//    public Cat(){
//        super("tom");
//    }
}

  

测试类

public class Test {
    public static void main(String[] args) {
        Cat cat = new Cat();
        System.out.println(cat.getInfor());
    }
}

  

 

子类对象实例化过程

示例

根据下图实现类。在TestCylinder类中创建Cylinder类的对象,设置圆柱的底面半径和高,并输出圆柱的体积,下面是uml图

父类

package com.uncleyong;

public class Circle {

    protected double radius;

    public Circle() {
        this.radius = 1;
        System.out.println("父类构造方法");
    }

    public double getRadius() {
        return radius;
    }

    public void setRadius(double radius) {
        this.radius = radius;
    }

    public double findArea(){
        return 3.14 * radius * radius;
    }

}

子类

package com.uncleyong;

public class Cylinder extends Circle{

    private double length;

    public Cylinder() {
        this.length = 1;
    }

    public double getLength() {
        return length;
    }

    public void setLength(double length) {
        this.length = length;
    }

    /**
     * 返回圆柱的体积
     * @return
     */
    public double findVolume(){
        return super.findArea() * length;  // 这里要加super表示调父类的,不加super表示调本类中的findArea方法
    }

    /**
     * 返回圆柱的表面积
     */
    @Override
    public double findArea() {
        return super.findArea() * 2 + 2 * 3.14 * radius * length;
    }

}

测试类

package com.uncleyong;

public class TestCylinder {

    public static void main(String[] args) {

        Cylinder cylinder = new Cylinder();

		cylinder.setLength(2);

        //返回表面积
        System.out.println(cylinder.findArea());
        //返回体积
        System.out.println(cylinder.findVolume());
    }

}

多态性及其应用

多态性

在Java中,子类的对象可以替代父类的对象使用(父类类型的引用变量可以指向子类的对象)
一个变量只能有一种确定的数据类型
一个引用类型变量可能指向(引用)多种不同类型的对象

Person p = new Student();
Object o = new Person();//Object类型的变量o,指向Person类型的对象
o = new Student(); //Object类型的变量o,指向Student类型的对象

一个引用类型变量如果声明为父类的类型,但实际引用的是子类对象,那么该变量就不能再访问子类中新添加的属性和方法(子类有父类没有的属性和方法)

Student m = new Student();
m.school = “pku”; //合法,Student类有school成员变量
Person e = new Student(); 
e.school = “pku”;	//非法,Person类没有school成员变量 

属性是在编译时确定的,编译时e为Person类型,没有school成员变量,因而编译错误。

虚拟方法调用(Virtual Method Invocation)

正常的方法调用

Person e = new Person();
e.getInfo();
Student e = new Student();
e.getInfo();

虚拟方法调用(多态情况下)

Person e = new Student();
e.getInfo();	//调用Student类中重写的getInfo()方法

编译时类型和运行时类型:编译时e为Person类型,而方法的调用是在运行时确定的,所以调用的是Student类的getInfo()方法。—— 动态绑定

在多态的情况下,调了父类的某个方法,但是这个方法被子类重写了,运行时,其实调的是子类重写的方法。【对象是子类的,所以调子类方法】

instanceof 操作符

如果下面p2要访问子类的属性和方法,需要把父类的实例变量p2强制转换为子类类型
Person p2 = new Man();
System.out.println(((Man) p2).school);
((Man) p2).work();
Woman w2 = (Woman)p2;  // 强制转换成女人,编译的时候可以过,因为p2是人,运行的时候不能过,因为运行时发现p2是男人

x instanceof A:检验x(x指向的对象)是否为类A的对象,返回值为boolean型。
要求x所属的类(x这个句柄的类型)与类A必须是子类和父类的关系,否则编译错误。
如果x属于类A的子类B,x instanceof A值也为true。

 

使用 instanceof 运算符. 注意, 使用 instanceof 运算符, 必须要求前面的引用指向的变量和后边的类之间存在父子关系【前面<=后面就返回true】

总结:
instanceof前后必须有继承关系,不管哪个继承哪个,否则类型不兼容报错 如果前面继承后面,结果为true 如果前面等于后面,结果为true 如果后面继承前面,结果为false 总之,有继承关系时,前面<=后面,就返回true

示例:

        Boy b = new Boy();
        System.out.println(">>>>>>>>>>>>>>>>");
        // Boy继承了Student
//        System.out.println(b instanceof Student);  // true;如果Boy没有继承Student,报类型不兼容错误
        // Boy2继承了Boy
        System.out.println(b instanceof Boy2); // false,因为b不是Boy2的对象
        System.out.println(b instanceof Boy); // true,因为b是Boy的对象
        // Boy2继承了Boy
        Boy2 b2 = new Boy2();
        System.out.println(b2 instanceof Boy); // true,因为b2是Boy的对象

 

实例2

public class TestPerson {
    public static void main(String[] args) {
        //多态

        //1. 创建一个 Man 的实例
        Man m1 = new Man();

        //2. 创建一个 Woman 的实例
        Woman w1 = new Woman();

        //3. 创建一个 Person 的实例
        Person p1 = new Person();
        /**
         * 多态: 在Java中,父类的引用可以指向子类的对象.【子类的对象可以替代父类的对象使用】
         * 1. 在多态情况下, 父类的实例变量不能再访问子类中新添加的属性和方法【来了一个人,是男是女都不知道,所以不能访问子类特有的属性和方法;如果下面p2要访问子类的属性和方法,需要把父类的实例变量p2强制转换为子类类型】
         * 2. 方法的调用是在运行时确定的,所以调用的是 Man 类的 getInfo() 方法【在多态的情况下,调用了父类的方法,这个方法已经被子类重写,运行的时候,实际上调用的是子类已经重写的那个方法,对象是子类的,所以调子类方法】。―― 动态绑定(虚拟方法调用)
         * 3. 在存在父子关系(多态)的情况下, 可以把父类的引用类型强制转换为子类的引用类型. 若实际上不能进行转换则
         *    系统会抛出 java.lang.ClassCastException 类型转换异常.
         * 4. 如何避免出现 java.lang.ClassCastException 异常呢? 在转换之前可以先判断一下对象实际上是否为指定的子类类型.
         *    使用 instanceof 运算符. 注意, 使用 instanceof 运算符, 必须要求前面的引用指向的变量和后边的类之间存在父子关系
         */
        //需要一个人, 但来的是一个男人! OK. 因为男人一定是一个人.
        Person p2 = new Man();
        System.out.println(((Man) p2).school);  // 如果p2要访问子类的属性和方法(子类有父类没有的),需要把父类的实例变量p2强制转换为子类类型
        ((Man) p2).work();
        System.out.println(p2.getInfo());  // Man's getInfo;按住ctrl点getInfo,跳转到Person类的getInfo,方法的调用是在运行时确定的,最终调用的是Man重写的的getInfo―― 动态绑定(虚拟方法法调用)

        //需要一个人, 但来的是一个女人! OK. 因为女人一定是一个人
        Person p3 = new Woman();

        //在多态情况下, 可以进行强制的类型转换
        Man m2 = (Man) p2;
//        Man m4 = (Man)p3;  // 会抛出 java.lang.ClassCastException 异常

        System.out.println(p3 instanceof Man);  // false
        System.out.println(p3 instanceof Woman);  // true

//		System.out.println(m2 instanceof Woman);  // Man和Woman无父子关系
        System.out.println(m2 instanceof Man);  // true
        System.out.println(m2 instanceof Person);  // true

        //需要一个男人, 但来的是个人! NO. 因为人不一定是男人.
        //Man m2 = new Person();

        //需要个男人, 但来的是一个女人。 NO!
        //Man m3 = new Woamn();

    }
}

  

针对上面第一条,如果下面p2要访问子类的属性和方法,需要把父类的实例变量p2强制转换为子类类型
Person p2 = new Man();
System.out.println(((Man) p2).school);
((Man) p2).work();
Woman w2 = (Woman)p2; // 强制转换成女人,编译的时候可以过,因为p2是人,运行的时候不能过,因为运行时发现p2是男人

对象类型转换 (Casting )

基本数据类型的Casting

小的数据类型可以自动转换成大的数据类型
  如long g=20; double d=12.0f
可以把大的数据类型强制转换(casting)成小的数据类型
  如 floate f=(float)12.0 int a=(int)1200L

对Java对象的强制类型转换称为造型

从子类到父类的类型转换可以自动进行
从父类到子类的类型转换必须通过造型(强制类型转换)实现
无继承关系的引用类型间的转换是非法的,编译报错
在造型前可以使用instanceof操作符测试一个对象的类型

从小到大(小东西放到大容器中),自动转
从大到小(大东西放到小容器中),强制转

 

示例

package com.uncleyong;

public class Person {
    protected String name="person";
    protected int age=50;
    public String getInfo() {
        return "Name: "+ name + "\n" +"age: "+ age;
    }
}
class Student extends Person {
    protected String school="pku";
    public String getInfo() {
        return  "Name: "+ name + "\nage: "+ age
                + "\nschool: "+ school;
    }

}
class Graduate extends Student{
    public String major="IT";
    public String getInfo()
    {
        return  "Name: "+ name + "\nage: "+ age
                + "\nschool: "+ school+"\nmajor:"+major;
    }
}

 

package com.uncleyong;

public class TestInstance {

	/*
	        在类中定义方法method1(Person e);
		在method1中:
		(1)根据e的类型调用相应类的getInfo()方法。
		(2)根据e的类型执行:
		如果e为Person类的对象,输出:“a person”;
		如果e为Student类的对象,输出
		“a student”
		“a person ”
		如果e为Graduate类的对象,输出:
		“a graduated student”
		“a student”
		“a person”
	 */

    public void method1(Person e){  // 多态的体现,Person类型的形参可以接收实际Person子类对象
        String info = e.getInfo();  // 动态绑定
        System.out.println(info);

        if(e instanceof Graduate){
            System.out.println("a graduated student");
        }
        if(e instanceof Student){
            System.out.println("a student");
        }
        if(e instanceof Person){
            System.out.print("a person");
        }

        System.out.println("\n");

    }

    public static void main(String[] args) {

        TestInstance ti = new TestInstance();

        Person p1 = new Person();
        ti.method1(p1);

        Person p2 = new Student();
        ti.method1(p2);

        Person p3 = new Graduate();
        ti.method1(p3);

    }

}  

Object 类及其主要方法

Object类

Object类是所有Java类的根父类

如果在类的声明中未使用extends关键字指明其父类,则默认父类为Object类

public class Person {
...
}
等价于:
public class Person extends Object {
...
}
例:

method(Object obj){…}  // 可以接收任何类作为其参数
Object o=new Person; 
method(o);

==操作符与equals方法

==操作符

引用类型比较引用(是否指向同一个对象,即:是否指向同一块內存空間,要求 == 两边的类型必须一致或存在着父子关系, 否则编译出错;
  Person p1=new Person();

  Person p2=new Person();

  if (p1==p2){…}

基本类型比较值:

  int a=5; if(a==6){…}

  用"=="进行比较时,符号两边的数据类型必须一致(可自动转换的基本数据类型除外),否则编译出错;

equals()方法

equals()方法是Object类的方法,由于所有类都继承Object类,也就继承了equals()方法。

equals()只能比较引用类型,其作用与“==”相同,比较是否指向同一个对象(可以比较任意两个对象)。格式:obj1.equals(obj2)

可以在类中重写 equals 方法, 以达到定制比较两个对象内容是否相等的目的

特例:当用equals()方法进行比较时,对类File、String、Date及封装类(Wrapper Class)来说,是比较类型及内容而不考虑引用的是否是同一个对象;

原因:在这些类中,已经重写了equals()方法,可以通过 equals 方法來判定其內容是否相同

 

 

比较两个字符串的內容是否相同, 一定要使用 equals() 方法, 而不能使用 ==

toString 方法
1. Object 类定义的方法, 所以任何对象都可以来调用 toString() 方法
2. 默认情况下, toString() 方法 全类名@hash码
3. 可以根据需要重写 toString() 方法, 通常用于测试. 个别时候用于显示,增加可读性.
4. JDK 中的很多类都重写了 toString() 方法

toString()方法在Object类中定义,其返回值是String类型,返回值:类名和它的引用地址。

在进行String与其它类型数据的连接操作时,自动调用toString()方法

Date now=new Date();
System.out.println(“now=”+now); 相当于 System.out.println(“now=”+now.toString());  // now=Date@122345

可以根据需要在用户自定义类型中重写toString()方法,如String 类重写了toString()方法,返回字符串的值。

s1=“hello”;
System.out.println(s1);  // 相当于System.out.println(s1.toString());【打印对象,会自动调对象的toString方法】
基本类型数据转换为String类型时,调用了对应封装类的 toString()方法,int a=10; System.out.println(“a=”+a);

 

示例:

父类

package com.uncleyong;

public class GeometricObject {
    protected  String  color;
    protected  double  weight;
}

子类

package com.uncleyong;

public class Circle1 extends GeometricObject{
    private double radius;

    public double getRadius() {
        return radius;
    }

    public void setRadius(double radius) {
        this.radius = radius;
    }

    public Circle1() {
        this.radius = 1;
        this.weight = 1;
        this.color = "white";
    }

    public Circle1(double radius) {
        this.radius = radius;
        this.weight = 1;
        this.color = "white";
    }

    public Circle1(double radius, double weight, String color) {
        this.radius = radius;
        this.weight = weight;
        this.color = color;
    }

    public double findArea(){
        return Math.PI * this.radius * this.radius;
    }

    @Override
    public boolean equals(Object obj) {
        if(this == obj){
            return true;
        }

        if(obj == null){
            return false;
        }

        if(!(obj instanceof Circle1)){
            return false;
        }

        Circle1 c = (Circle1) obj;

        return c.getRadius() == this.radius;
    }

    @Override
    public String toString() {
        return "" + this.radius;  // 写为return this.radius;不行,因为radius是double
    }
}

测试类

package com.uncleyong;

public class TestCircle1 {

    /**
     * 写一个测试类,创建两个Circle对象,判断其颜色是否相等;
     * 利用equals方法判断其半径是否相等;
     * 利用toString()方法输出其半径。
     */
    public static void main(String[] args) {

        Circle1 c1 = new Circle1(1, 2, "Black");
        Circle1 c2 = new Circle1(2, 3, "Black");
        Circle1 c3 = new Circle1(2, 2, "Black");
        Circle1 c4 = new Circle1(2, 3, "Red");

        // 比颜色
        System.out.println(c1.color.equals(c2.color));  // true
        // 比半径
        System.out.println(c1.equals(c2));  // false

        // 比颜色
        System.out.println(c3.color.equals(c4.color));  // false
        // 比半径
        System.out.println(c3.equals(c4));  // true

        System.out.println(c1);  // 1.0   打印对象,会自动调对象的toString方法

    }
}

封装类

针对八种基本定义相应的引用类型—封装类

int i = 10;
Integer j = 10;  // Integer是封装类,基本数据类型10赋给j,自动装箱

 

自动装箱
Integer a = 128;

 

反编译生成的class文件:

Integer a = Integer.valueOf(128);

这就是基本数据类型的自动装箱,128是基本数据类型,然后被解析成Integer类。

注意:自动装箱规范要求 byte<= 127、char<=127、-128<=short <=127、-128<=int <=127都被包装到固定的对象中(缓存)。

 

自动拆箱

将 Integer 类表示的数据赋值给基本数据类型int,就执行了自动拆箱。

Integer a = new Integer(128);
int m = a;

反编译生成的class文件:

Integer a = new Integer(128);
int m = a.intValue();

 

自动装箱就是:Integer a = Integer.valueOf(int i);

自动拆箱就是:
  Integer a = new Integer(128);
  int m = a.intValue();

参考:

https://www.cnblogs.com/uncleyong/p/9804074.html

https://www.cnblogs.com/uncleyong/p/13844652.html

 

练习题

https://www.cnblogs.com/uncleyong/p/15828510.html

 

posted @ 2018-10-03 23:51  全栈测试笔记  阅读(955)  评论(0编辑  收藏  举报
浏览器标题切换
浏览器标题切换end