java基础12_面向对象(中-02)

208 阅读9分钟

1.向下强制转型

1.1问题引入

有了对象的多态性以后,内存中实际上是加载了子类特有的属性和方法的,但是由于变量声明为父类类型,导致编译时,只能调用父类中声明的属性和方法。子类特有的属性和方法不能调用。

如何才能调用子类特有的属性和方法?

向下转型:使用强制类型转换符。

1.2举例

System.out.println("xin********************************");

Man m1 = (Man)p2;//p2是Person
m1.earnMoney();
m1.isSmoking = true;

//使用强转时,可能出现ClassCastException的异常。
//Man cannot be cast to class oop.instance.Woman
//Woman w1 = (Woman)p2;
//w1.goShopping();

1.3instanceof关键字的使用

/*
 * a instanceof A:判断对象a是否是类A的实例。如果是,返回true;如果不是,返回false。
 * 
 * 
 *  使用情境:为了避免在向下转型时出现ClassCastException的异常,我们在向下转型之前,先
 *  进行instanceof的判断,一旦返回true,就进行向下转型。如果返回false,不进行向下转型。
 *  
 *  若类B是类A的父类,
 *  如果 a instanceof A返回true,则 a instanceof B也返回true.
 * 
 */

image.png

1.4练习

package oop.exer1;

//考查多态的笔试题目:
public class InterviewTest1 {

	public static void main(String[] args) {
		Base1 base = new Sub1();
		base.add(1, 2, 3);//sub_1认为是重写

		Sub1 s = (Sub1)base;
		s.add(1,2,3);//sub_2
	}
}

class Base1 {
	public void add(int a, int... arr) {
		System.out.println("base1");
	}
}

class Sub1 extends Base1 {

	public void add(int a, int[] arr) {
		System.out.println("sub_1");
	}

	public void add(int a, int b, int c) {
		System.out.println("sub_2");
	}

}

2.object

2.1 == 与 equals

image.png

image.png

package oop.obj;

import java.util.Date;

/*
 * 
 * 面试题: == 和 equals() 区别
 * 
 * 一、回顾 == 的使用:
 * == :运算符
 * 1. 可以使用在基本数据类型变量和引用数据类型变量中
 * 2. 如果比较的是基本数据类型变量:比较两个变量保存的数据是否相等。(不一定类型要相同)
 *    如果比较的是引用数据类型变量:比较两个对象的地址值是否相同.即两个引用是否指向同一个对象实体
 * 
 * 
 * 二、equals()方法的使用:
 * 1. 是一个方法,而非运算符
 * 2. 只能适用于引用数据类型
 * 3. Object类中equals()的定义:
 *    public boolean equals(Object obj) {
	        return (this == obj);
	  }
 *    说明:Object类中定义的equals()和==的作用是相同的:比较两个对象的地址值是否相同.即两个引用是否指向同一个对象实体
 * 
 * 4. 像String、Date、File、包装类等都重写了Object类中的equals()方法。
 *    重写以后,比较的不是两个引用的地址是否相同,而是比较两个对象的"实体内容"是否相同。
 *    
 * 5. 通常情况下,我们自定义的类如果使用equals()的话,也通常是比较两个对象的"实体内容"是否相同。
 * 	  那么,我们就需要对Object类中的equals()进行重写.
 *    重写的原则:比较两个对象的实体内容是否相同.
 */
public class EqualsTest {

	public static void main(String[] args) {
		//基本数据类型
		int i = 10;
		int j = 10;
		double d = 10.0;
		System.out.println(i == j);//true
		System.out.println(i == d);//true
		
		boolean b = true;
		//System.out.println(i == b);baocuo
		
		char c = 10;
		System.out.println(i == c);//true
		
		char c1 = 'A';
		char c2 = 65;
		System.out.println(c1 == c2);//true
		
		
		//引用类型:比较地址值
		Customer cust1 = new Customer("Tom",21);
		Customer cust2 = new Customer("Tom",21);
		System.out.println(cust1 == cust2);//false
		
		String str1 = new String("atguigu");
		String str2 = new String("atguigu");
		System.out.println(str1 == str2);//false
		System.out.println("****************************");
		
		System.out.println(cust1.equals(cust2));//false?
		System.out.println(str1.equals(str2));//true
		
		
		Date date1 = new Date(32432525324L);
		Date date2 = new Date(32432525324L);
		System.out.println(date1.equals(date2));//true
	
	
	}
}

image.png

2.2 toString

package oop.obj;

import java.util.Date;

/*
 * Object类中toString()的使用:
 * 
 * 1. 当我们输出一个对象的引用时,实际上就是调用当前对象的toString()
 * 
 * 2. Object类中toString()的定义:
 *   public String toString() {
        return getClass().getName() + "@" + Integer.toHexString(hashCode());
     }
 * 
 * 3. 像String、Date、File、包装类等都重写了Object类中的toString()方法。
 *    使得在调用对象的toString()时,返回"实体内容"信息
 *    
 * 4. 自定义类也可以重写toString()方法,当调用此方法时,返回对象的"实体内容"
 */
public class ToStringTest {

	public static void main(String[] args) {
		Customer cust1 = new Customer("Tom",21);
		System.out.println(cust1.toString());//oop.obj.Customer@7de26db8
		System.out.println(cust1);//oop.obj.Customer@7de26db8
		//重写后:Customer [name=Tom, age=21]
		
		
		String str = new String("MM");
		System.out.println(str);//MM肯定是重写过
	
		Date date = new Date(4534534534543L);
		System.out.println(date.toString());//Mon Sep 11 08:55:34 CST 2113
		
	}
}

3.Object综合练习

package oop.exer3;

public class GeometricObject {

	protected String color;
	protected double weight;
	
	
	public GeometricObject() {
		super();
		this.color = "white";
		this.weight = 1.0;
	}
	
	
	public GeometricObject(String color, double weight) {
		super();
		this.color = color;
		this.weight = weight;
	}


	public String getColor() {
		return color;
	}
	public void setColor(String color) {
		this.color = color;
	}
	public double getWeight() {
		return weight;
	}
	public void setWeight(double weight) {
		this.weight = weight;
	}
	
}

package oop.exer3;

public class Circle extends GeometricObject{

	private double radius;

	
	
	public Circle() {
		super();
		radius = 1.0;
//		color = "blue";
//		weight = 2.0;
		//super得到父类的初始化了
	}

	public Circle(double radius) {
		super();
		this.radius = radius;
	}

	public Circle(double radius,String color,double weight) {
		super(color,weight);
		this.radius = radius;
	}
	
	
	public double getRadius() {
		return radius;
	}

	public void setRadius(double radius) {
		this.radius = radius;
	}
	
	
	//求圆的面积
	public double findArea(){
		return 3.14 * radius * radius;
	}
	
	//比较两个圆的半径是否相等,如相等,返回true。
		@Override
		public boolean equals(Object obj) {
			
			if(this == obj){
				return true;
			}
			
			if(obj instanceof Circle){
				Circle c = (Circle)obj;
				return this.radius == c.radius;
			}
			
			return false;
			
		}

		@Override
		public String toString() {
			// TODO Auto-generated method stub
			return "Circle [radius=" + radius + "]";
		}
		
		
		
}

package oop.exer3;
/*
 * 写一个测试类,创建两个Circle对象,判断其颜色是否相等;
 * 利用equals方法判断其半径是否相等;利用toString()方法输出其半径。
 */
public class CircleTest {
	public static void main(String[] args) {
		Circle circle1 = new Circle(2.3);
		Circle circle2 = new Circle(3.3, new String("white"), 2.0);
		
		System.out.println("颜色是否相等:" + circle1.getColor().equals(circle2.getColor()));
		System.out.println(circle1.getColor());
		System.out.println(circle2.getColor());

		System.out.println("半径是否相等:" + circle1.equals(circle2));
		
		System.out.println(circle1);
		System.out.println(circle2.toString());
	}
}

4.包装类

4.1单元测试

package oop.junit;

import java.util.Date;

import org.junit.Test;

/*
 * Java中的JUnit单元测试
 * 
 * 步骤:
 * 1.选中当前工程 - 右键选择:build path - add libraries - JUnit 5 - 下一步
 * 2.创建Java类,进行单元测试。
 *   此时的Java类要求:① 此类是public的  ②此类提供公共的无参的构造器
 * 3.此类中声明单元测试方法。
 *   此时的单元测试方法:方法的权限是public,没有返回值,没有形参
 * 
 * 4.此单元测试方法上需要声明注解:@Test,并在单元测试类中导入:import org.junit.Test;
 * 
 * 5.声明好单元测试方法以后,就可以在方法体内测试相关的代码。
 * 6.写完代码以后,左键双击单元测试方法名,右键:run as - JUnit Test
 * 
 * 说明:
 * 1.如果执行结果没有任何异常:绿条
 * 2.如果执行结果出现异常:红条
 */
public class JUnitTest {

	int num = 10;
	
	@Test
	public void testEquals(){
		String s1 = "MM";
		String s2 = "MM";
		System.out.println(s1.equals(s2));
		
		//ClassCastException的异常
		Object obj = new String("GG");
		String str = (String)obj;//正确
		System.out.println(obj.equals(str));
//		Date date = (Date)obj;//错了
		System.out.println(num);//10
		show();//show()....
		
	}
	
	public void show(){
		num = 20;
		System.out.println("show()....");
	}
	
	@Test
	public void testToString(){
		String s2 = "MM";
		System.out.println(s2.toString());
	}
	
	
}

4.2包装类介绍

image.png

image.png

4.3基本数据类型--包装类

//基本数据类型 --->包装类:调用包装类的构造器
package oop.junit;

import org.junit.Test;

/*
 * 包装类的使用:
 * 1.java提供了8种基本数据类型对应的包装类,使得基本数据类型的变量具有类的特征
 * 
 * 2.掌握的:基本数据类型、包装类、String三者之间的相互转换
 */
public class WrapperTest {

	@Test
	public void test1() {
		int num1 = 10;
//		System.out.println(num1.toString());
		Integer in1 = new Integer(num1);
		System.out.println(in1.toString());
		
		Integer in2 = new Integer("123");
		System.out.println(in2.toString());
		
		//报异常
//		Integer in3 = new Integer("123abc");
//		System.out.println(in3.toString());
		
		Float f1 = new Float(12.3f);
		//构造函数Float(Float)自版本9起就被弃用,并标记为删除
		Float f2 = new Float("12.3");
		System.out.println(f1);
		System.out.println(f2);
		
		Boolean b1 = new Boolean(true);
		Boolean b2 = new Boolean("TrUe");
		System.out.println(b2);//true
		Boolean b3 = new Boolean("true123");
		System.out.println(b3);//false
		
		Order order = new Order();
		System.out.println(order.isMale);//false
		System.out.println(order.isFemale);//null
		
	}
}


class Order{
	
	boolean isMale;
	Boolean isFemale;//类了
}
//包装类--->基本数据类型:调用包装类Xxx的xxxValue()
    @Test
    public void test2(){
            Integer in1 = new Integer(12);

            int i1 = in1.intValue();
            System.out.println(i1 + 1);

            Float f1 = new Float(12.3);
            float f2 = f1.floatValue();
            System.out.println(f2 + 1);

    }
/*
	 * JDK 5.0 新特性:自动装箱 与自动拆箱
	 */
	@Test
	public void test3(){
//		int num1 = 10;
//		//本来  int 不能给 Object的
//		//基本数据类型-->包装类的对象
//		method(num1);
		
		//自动装箱:基本数据类型 --->包装类
		int num2 = 10;
		//Integer in1 = new Integer(num2);//对比
		Integer in1 = num2;//自动装箱
		
		boolean b1 = true;
		Boolean b2 = b1;//自动装箱
		
		//自动拆箱:包装类--->基本数据类型
		System.out.println(in1.toString());
		int num3 = in1;//自动拆箱
		
	}
	
	public void method(Object obj){
		System.out.println(obj);
	}
//基本数据类型、包装类--->String类型:调用String重载的valueOf(Xxx xxx)
    @Test
    public void test4(){

            int num1 = 10;
            //方式1:连接运算
            String str1 = num1 + "";
            System.out.println(str1);
            //方式2:调用String的valueOf(Xxx xxx)
            float f1 = 12.3f;
            String str2 = String.valueOf(f1);//"12.3"
            System.out.println(str2);

            Double d1 = new Double(12.4);
            String str3 = String.valueOf(d1);
            System.out.println(str3);//"12.4"

    }
//String类型 --->基本数据类型、包装类:调用包装类的parseXxx(String s)
    @Test
    public void test5(){
            String str1 = "123";
            //错误的情况:
//			int num1 = (int)str1;//没啥关系,不能转
//			Integer in1 = (Integer)str1;//没啥子父类关系,不能转
            //可能会报NumberFormatException
            int num2 = Integer.parseInt(str1);
            System.out.println(num2 + 1);//124

            String str2 = "true1";//false
            boolean b1 = Boolean.parseBoolean(str2);
            System.out.println(b1);
    }

4.4面试题

image.png

package oop.junit;

import org.junit.Test;

public class InterviewTest {
	@Test
	public void test1() {
		Object o1 = true ? new Integer(1) : new Double(2.0);
		Object o2 = false ? new Integer(1) : new Double(2.0);
		System.out.println(o1);// 1.0
                System.out.println(o2);// 2.0
		//跟true,false无关。编译时统一比较类型--提升成double

	}

	@Test
	public void test2() {
		Object o2;
		if (true)
			o2 = new Integer(1);
		else
			o2 = new Double(2.0);
		System.out.println(o2);// 1

	}

	@Test
	public void test3() {
		Integer i = new Integer(1);
		Integer j = new Integer(1);
		System.out.println(i == j);//false。比较地址
		
		//Integer内部定义了IntegerCache结构,IntegerCache中定义了Integer[],
		//保存了从-128~127范围的整数。如果我们使用自动装箱的方式,给Integer赋值的范围在
		//-128~127范围内时,可以直接使用数组中的元素,不用再去new了。目的:提高效率
		
		Integer m = 1;
		Integer n = 1;
		System.out.println(m == n);//true

		Integer x = 128;//相当于new了一个Integer对象
		Integer y = 128;//相当于new了一个Integer对象
		System.out.println(x == y);//false
	}

}

谈谈你对多态性的理解?

① 实现代码的通用性。

② Object类中定义的public boolean equals(Object obj){ }

JDBC:使用java程序操作(获取数据库连接、CRUD)数据库(MySQL、Oracle、DB2、SQL Server)

③ 抽象类、接口的使用肯定体现了多态性。(抽象类、接口不能实例化)

一个题

package oop.exer3;

import java.util.Scanner;
import java.util.Vector;

/*
 *  利用Vector代替数组处理:从键盘读入学生成绩(以负数代表输入结束),找出最高分,并输出学生成绩等级。
	提示:数组一旦创建,长度就固定不变,所以在创建数组前就需要知道它的长度。
	而向量类java.util.Vector可以根据需要动态伸缩。
	
	创建Vector对象:Vector v=new Vector();
	给向量添加元素:v.addElement(Object obj);   //obj必须是对象
	取出向量中的元素:Object  obj=v.elementAt(0);
	注意第一个元素的下标是0,返回值是Object类型的。
	计算向量的长度:v.size();
	若与最高分相差10分内:A等;20分内:B等;
	      30分内:C等;其它:D等 
 * 
 */
public class ScoreTest {

	public static void main(String[] args) {
		//1.实例化Scanner,用于从键盘获取学生成绩
		Scanner scan = new Scanner(System.in);
		
		//2.创建Vector对象:Vector v=new Vector();相当于原来的数组
		Vector v = new Vector();
		
		//3.通过for(;;)或while(true)方式,给Vector中添加数组
		
		
		int maxScore = 0;
		for(;;){
			System.out.println("请输入学生成绩(以负数代表输入结束)");
			int score = scan.nextInt();
			
			//3.2 当输入是负数时,跳出循环
			if(score < 0){
				break;
			}
			
			if(score > 100){
				System.out.println("输入的数据非法,请重新输入");
				continue;
			}
			//3.1 添加操作::v.addElement(Object obj)
			//jdk5.0之前:
//			Integer inScore = new Integer(score);
//			v.addElement(inScore);//多态
			//jdk5.0之后:
			v.addElement(score);//自动装箱
			//4.获取学生成绩的最大值
			if(maxScore < score){
				maxScore = score;
			}
			
		}
		
		
		//5.遍历Vector,得到每个学生的成绩,并与最大成绩比较,得到每个学生的等级。
		char level;
		for(int i = 0;i < v.size();i++){
			Object obj = v.elementAt(i);
		
			//之前
//			Integer inScore = (Integer)obj;//基本数据类型 --->包装类
//			int score = inScore.intValue();//包装类--->基本数据类型:调用包装类Xxx的xxxValue()
			
			//jdk 5.0之后:
			int score = (int)obj;
			
			if(maxScore - score <= 10){
				level = 'A';
			}else if(maxScore - score <= 20){
				level = 'B';
			}else if(maxScore - score <= 30){
				level = 'C';
			}else{
				level = 'D';
			}
			
			System.out.println("student-" + i + " score is " + score + ",level is " + level);
			
			
			
		}
	}
}