Java笔记八(instanceof,类型转换,static详解,抽象类,接口,内部类以及异常)
instanceof
引用类型,判断一个对象是什么类型
使用方法:
System.out.println(X instanceof Y);
代码理解:
public class Application {
public static void main(String[] args) {
//Obiect>String
//Obiect>Person>Teacher
//Obiect>Person>Student
Object object=new Student();
System.out.println(object instanceof Student);
System.out.println(object instanceof Person);
System.out.println(object instanceof Object);
System.out.println(object instanceof Teacher);
System.out.println(object instanceof String);
System.out.println("==========================");
Person person=new Student();
System.out.println(person instanceof Student);
System.out.println(person instanceof Person);
System.out.println(person instanceof Object);
System.out.println(person instanceof Teacher);
//System.out.println(person instanceof String);编译报错
System.out.println("=========================");
Student student=new Student();
System.out.println(student instanceof Student);
System.out.println(student instanceof Person);
System.out.println(student instanceof Object);
/*System.out.println(student instanceof Teacher);
System.out.println(student instanceof String);编译报错*/
}
}
其Person是Student与Teacher的父类
类型转换
父类的引用指向子类的对象
把子类转换为父类,向上转型
把父类转换为子类,向下转换:强制转换
方便方法调用,减少重复的代码
强制转换:
package com.oop.demo06;
public class Application {
public static void main(String[] args) {
//类型之间的转换
//子类转换为父类可能丢失自己本来的一些方法
Person obj=new Student();
//student将这个对象转换为Student类型,我们就可以使用Student类型的方法了
Student student=(Student)obj;
student.go();
}
}
package com.oop.demo06;
public class Student extends Person {
public void go(){
System.out.println("go");
}
}
也可以这样写
package com.oop.demo06;
public class Application {
public static void main(String[] args) {
//类型之间的转换
//子类转换为父类可能丢失自己本来的一些方法
Person obj=new Student();
//student将这个对象转换为Student类型,我们就可以使用Student类型的方法了
Student student=(Student)obj;
((Student) obj).go();
}
}
低转高自动转换
package com.oop.demo06;
public class Application {
public static void main(String[] args) {
Student student=new Student();
student.go();
Person person=student;
}
}
static关键字详解
非静态的方法可以调用静态里面的方法
静态方法只能调用静态方法的
package com.oop.demo07;
//static
public class Student {
private static int age;//静态变量
private double score;//非静态变量
public void run(){
go();//非静态的方法可以调用静态里面的方法
}
public static void go(){
}
public static void main(String[] args) {
go();//静态方法只能调用静态方法的
new Student().run();
}
}
关于static的静态代码块
package com.oop.demo07;
public class Person {
{
System.out.println("匿名代码块");//代码块(匿名代码块)
}
static{
System.out.println("静态代码块");
//静态代码块
}
public Person() {
System.out.println("构造方法");
}
public static void main(String[] args) {
Person person=new Person();
System.out.println("================");
Person person1=new Person();
}
}
运行结果
由此可见静态代码块先执行且只执行依次,匿名代码块可以用来赋初始值
静态导入包
package com.oop.demo07;
public class Text {
public static void main(String[] args) {
System.out.println(Math.random());//随机生成一个数
}
}
使用静态导入包
package com.oop.demo07;
import static java.lang.Math.random;
import static java.lang.Math.PI;
public class Text {
public static void main(String[] args) {
System.out.println(random());
System.out.println(PI);
}
}
运行结果
注意:通过final修饰的类不能被继承,也就是没有子类
抽象类
package com.oop.demo09;
//abstract 抽象类 extend:单继承 接口可以多继承
public abstract class Action {
//约束,有人帮实现
//abstract 抽象方法,只有方法名字没有方法的实现
public abstract void doSomething();
}
package com.oop.demo09;
//抽象类的所有方法,继承了它的子类,都必须要实现它的方法
public class A extends Action{
@Override
public void doSomething() {
}
}
1.不能new这个抽象类,只能靠子类区实现他
2.抽象类中可以写普通方法
3.抽象方法必须在抽象类中
接口
普通类:只有具体实现
抽象类:具体实现和规范(抽象方法)都有
接口:只有规范,自己无法写方法。专业的约束,约束和实现分离
接口的本质是契约,就像是人间的法律一样,制定好之后大家都遵守
声明类的关键字是class,声明接口的关键字是interface
实现了接口的类就需要重写接口中的方法,并且可以利用接口实现多继承
如下代码拥有两个接口,每个接口拥有多个方法
package com.oop.demo10;
//interface
public interface UserService {
//接口中的所有定义其实都是抽象的 public abstract
int AGE=99;
void add(String name);
void delete(String name);
void update(String name);
void query(String name);
}
package com.oop.demo10;
public interface TimeService {
void timer();
}
package com.oop.demo10;
//抽象类:extends
//类 可以实现接口 implements 接口
//实现了接口的类就需要重写接口中的方法
//利用接口实现多继承
public class UserServiceImpl implements UserService,TimeService{
@Override
public void query(String name) {
}
@Override
public void update(String name) {
}
@Override
public void delete(String name) {
}
@Override
public void add(String name) {
}
@Override
public void timer() {
}
}
作用:
1.约束
2.定义一些方法,让不同的人实现
3.接口不能被实例化,接口中没有构造方法
4.implements可以实现多个接口
5.必须重写接口中的方法
内部类
内部类就是在一个类的内部再定义一个类,比如,A类中定义一个B类,那么B类相对A类来说就称为内部类,而A类相对B类来说就是外部类了
1.成员内部类
package com.oop.demo11;
public class Outer {
private int id=10;
public void out(){
System.out.println("这是外部类的方法");
}
public class Inner{
public void in(){
System.out.println("这是内部类的方法");
}
//获得外部类的私有属性
public void getID(){
System.out.println(id);
}
}
}
package com.oop.demo11;
public class Application {
public static void main(String[] args) {
Outer outer=new Outer();
//通过这个外部类来实例化内部类
Outer.Inner inner=outer.new Inner();
inner.getID();
}
}
2.静态内部类:使用static修饰,则无法获得外部类的私有属性
package com.oop.demo11;
public class Outer {
private int id=10;
public void out(){
System.out.println("这是外部类的方法");
}
public static class Inner{
public void in(){
System.out.println("这是内部类的方法");
}
//获得外部类的私有属性
}
}
局部内部类
package com.oop.demo11;
public class Outer {
public void method(){
class Inner{
public void in(){
}
}
}
}
与局部变量相似
异常处理机制
这是一个错误代码,因为0不能作为除数
package com.oop.exception;
public class Test {
public static void main(String[] args) {
int a=1;
int b=0;
System.out.println(a/b);
}
}
使用catch捕获异常:
package com.oop.exception;
public class Test {
public static void main(String[] args) {
int a=1;
int b=0;
try {
System.out.println(a/b);
}catch (ArithmeticException e){//catch 捕获异常
System.out.println("程序出现异常,变量b不能为0");
}finally {//处理善后工作
System.out.println("finally");
}
}
}
假设要捕获多个异常:从小到大
package com.oop.exception;
public class Test {
public static void main(String[] args) {
int a=1;
int b=0;
try {
System.out.println(a/b);
}catch (Error e){//catch 捕获异常
System.out.println("Error");
}catch (Exception e){
System.out.println("Exception");
}catch (Throwable t){
System.out.println("Throwable");
}finally {
System.out.println("finally");
}
}
}
快捷键ctrl+alt+t,快速将代码块使用功能包裹
主动抛出异常:
package com.oop.exception;
public class Test {
public static void main(String[] args) {
int a=1;
int b=0;
try {
if (b==0){
throw new ArithmeticException();
}
System.out.println(a/b);
}catch (Exception e){
System.out.println("Exception");
}finally {
System.out.println("finally");
}
}
}
运行结果:
在方法中抛出异常:
package com.oop.exception;
public class Test {
public static void main(String[] args) {
new Test().test(1,0);
}
public void test(int a,int b){
if (b==0){
throw new ArithmeticException();//主动抛出异常,一般在方法中使用
}
//System.out.println(a/b);
}
}
可见即使不用输出也可以抛出异常
自定义异常
这里自定义一个传递数字的异常,如果传递的数字大于10则抛出异常
package com.oop.demo12;
//自定义的异常类
public class MyException extends Exception{
//传递数字,大于10抛异常
private int detail;
public MyException(int a) {
this.detail=a;
}
//toString异常的打印信息
@Override
public String toString() {
return "MyException{"+detail+'}';
}
}
package com.oop.demo12;
public class Test {
//可能会存在异常的方法
static void test(int a) throws MyException {
System.out.println("传递的参数为:"+a);
if (a > 10) {
throw new MyException(a);
}
System.out.println("OK");
}
public static void main(String[] args) {
try {
test(11);
}catch (MyException e){
System.out.println("MyException=>"+e);
}
}
}
可见我们此时传递的数字为11,此时抛出我们自定义的异常
经验总结
◆处理运行时异常时,采用逻辑去合理规避同时辅助try-catch处理
◆在多重catch块后面,可以加一个catch (Exception) 来处理可能会被遗漏的异常
◆对于不确定的代码,也可以加上try-catch,处理潜在的异常
◆尽量去处理异常,切忌只是简单地调用printStackTrace() 去打印输出
◆具体如何处理异常,要根据不同的业务需求和异常类型去决定
◆尽量添加finally语句块去释放占用的资源
在此狂神java基础笔记全部结束,用时一个月左右学完,也不能说是学完,算是看完吧,真的很喜欢狂神的java基础课,仍然记得狂神的那句话:学编程是为了更好得建模这个世界。愿我们与月作伴的日子不要忘记自己的初心。路漫漫其修远兮,吾将上下而求索!
狂神哔哩哔哩主页:遇见狂神说的个人空间-遇见狂神说个人主页-哔哩哔哩视频