异常,线程

Posted linlin1211

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了异常,线程相关的知识,希望对你有一定的参考价值。

异常,线程
一.异常
  1.什么是异常
    java程序中出现的不正常显现
  2.异常的继承体系
    面向对象语言中异常就是用类来表示
    (万物皆对象)每个异常都是一个对象
   异常的根类:Throwable
      子类1:Error,被称为错误类(操作系统本身有错误,程序员不能避免)------内存溢出,系统崩溃
      子类2:Exception,被称为异常类
  3.异常类中常用的三个方法
    -public void printStackTrace();//打印异常的详细信息
      包含了异常的类型,异常的原因,还包括异常出现的位置,在开发和调试阶段,都得使用printStackTrace。
    -public String getMessage();//获取发生异常的原因
      提示给用户的时候,就提示错误原因
    -public String toString();//获取异常的类型和异常描述信息(不用)
  4.异常的分类:
    i.编译时异常
      编译时如果异常出现就会报错
      是Exception本身以及子类(RuntimeException除外)
    ii.运行时异常
      编译时不报错,运行时报错
      就是RuntimeException本身以及它的子类
  5.异常产生过程
    JVM发现了一个异常,数组做大下标是2,但是我们要访问3
    创建一个异常对象(代表这种异常)
      new ArrayIndexOutOfBounds(3)
    把这个异常对象抛出去(不管)
      throw new ArrayIndexOutOfBounds(3)
    最终又到了:JVM
    JVM拿到异常对象之后:
    处理异常:终端处理
        立刻停止程序
        把异常对象的所有信息打印到控制台。
 列举出常见的三个运行时异常:
    ArrayIndexOutOfBoundsException 数组下标越界(运行时)
    NullPointerException 空指针引用异常
    ClassCastException 类型强制转换异常
        Animal an=new Dog();//new Cat();
        Cat cc=(Cat)an;
    ParseException 编译时异常

====================================================

    IllegalArgumentException - 传递非法参数异常。
    ArithmeticException - 算术运算异常
    ArrayStoreException - 向数组中存放与声明类型不兼容对象异常
    NegativeArraySizeException - 创建一个大小为负数的数组错误异常
    NumberFormatException - 数字格式异常
    SecurityException - 安全异常
    UnsupportedOperationException - 不支持的操作异常
    注意:
      JVM默认处理异常的机制:终断处理
二.异常处理
  1.抛出异常关键字throw
    throw作用:抛出异常对象(动词)
    throw格式:
      throw new XxxException(异常的信息)
  2.Objects类中提供的非空判断方式:
    只要我们定义的方法,有引用类型作用参数,那么第一件必须是判空
    public static<T> requireNonNull(T obj)检查指定的对象引用不是null。
  3.遇到异常的2种处理方式*******************
    throws声明抛出异常(不处理)
      使用格式:
        public 返回值类型 方法名() throws XxxException{
        //内部可能会抛出异常
         }
      作用:给方法做声明,告诉该方法的调用者,这个方法内部可能抛出某种异常,要求调用者自己处理
    try.catch捕获异常
      格式:
        try{
          可能出现异常的代码
        }catch(XxxException e){//捕获异常时,必须是可能抛出异常的类型 或者 其父类类型
          //处理异常
        e.printStackStrace();
        }
          //下面代码可以照常运行
  4.【扩展1】catch代码块中三种常见的处理方式
      a.直接打印 调用 异常对象.printStackTrace();
      b.记录日志(Log4J ---log for java)
      c.转成其他异常继续抛出
  5.【扩展2】多个异常如何捕获处理
    try{
      可能出现异常的代码
      假设:可能出现 异常1,异常2,异常3
      异常1 继承了 异常2
      异常3和 异常1 异常2 没有直接关系
     }
    a.换个异常 挨个处理
      try{
      代码段1(可能出现异常1)
      }catch(异常1 e){

      }
      try{
      代码段2(可能出现异常2)
      }catch(异常2 e){

      }
      ..........
    b.比较标准的处理方式:
      所有异常一起try,多个catch
      try{
        可能出现异常的代码
      }catch(异常1 e){

      }catch(异常2 e){

      }catch(异常3 e){

      }
    如果多个异常之间没有子父类关系,那么顺序随意
    如果其中两个异常有子父类关系,那么保证子类异常写前面,父类异常写后面,或者只写父类异常。
   e.开发中最常用的:
    所有异常一起try,一个catch
      try{
      可能出现异常的代码
      }catch(Exception e){
      此处异常应该写三个异常的共同父类,如果不知道那么直接写Exception

      }
  6.finally代码块
      finally:最终的,最后的
      和try..catch一起使用
      格式:
        try{
          可能有异常的代码
        }catch(异常类型 变量名){
          处理异常的代码
        }fimally{
          必须执行的代码
          一般用来写 释放资源的代码
          如果没有资源需要释放,那么可以省略finally代码块
        }
  7.异常的注意事项:
       编译时异常需要编译阶段处理,运行时异常编译阶段不需要管它
       如果父类的方法抛出了多个异常,子类覆盖父类方法时,只能抛出相同的异常或者是他的子集
       如果父类的方法没有抛出异常,子类也必须不能抛出异常
       当多异常处理时,捕获处理,前边的类不能是后边类的父类
       不要在finally里边写return
       finally可写可不写,通常有资源需要释放时,写finally保证资源正常释放。
三.自定义异常
  1.为什么要定义异常
      JDK中指定义一些常用的异常类,实际开发中很可能某些异常是JDK考虑不带
      所有我们需要自己定义一些异常类
  2.自定义异常的步骤
      a.创建一个异常类,但是异常类必须以eaception结尾
      b.该异常类必须继承Exception 或者 RuntimeException
      c.自定义的异常类必须是包含里两个构造(无参+String异常信息)
  3.自定义异常的练习(代码演示)
    要求:
      我们模拟注册操作,如果用户名已存在,则抛出异常并提示:亲,该用户名已经被注册。

/**
    * 自定义的异常类
    */
    //1.必须以Exception结尾
    public class MyException extends RuntimeException{
    //2.至少有有两个构造
    //无参构造
    public MyException(){}
    //带有String参数的构造
    public MyException(String info){

    super(info);

    }
    }

  测试:

    public static void main(String[] args) {
        //注册
        Scanner scanner = new Scanner(System.in);
        System.out.println("请输入您要注册的用户名");
        String next = scanner.next();
        //调用方法注册
        try {
            register(next);
            System.out.println("恭喜您注册成功");
        }catch (MyException e){
            e.printStackTrace();
        }
    }
    //注册方法
    public static void register(String username) throws MyException{
        //判断用户名是否重复
        //1.定义一个集合,模拟数据库
        ArrayList<String> usernames=new ArrayList<String>();
        //添加几个已经存在的用户
        Collections.addAll(usernames,"jake","bill","rose");
        //2.判断用户名是否重复
        if (usernames.contains(username)) {
            //如果用户名已经存在
            throw new MyException("亲,该用户名已经被注册");

        }
    }

  

四.多线程
  1.并行和并发:
      并行:指的两个事件同同一时刻进行
      并发:指的是两个时间同一时间段内进行
      宏观:我们人看到的
      微观:具体的时间片段
   2.进程和线程:
      进程:正在运行的程序,就称为进程
      线程:进程中的某一个任务,就称为线程
      进程和线程的一些区别(了解)
        进程是拥有(堆空间和栈空间)独立的空间(不同的之间是不能相互访问数据的),每个进程至少有一个线程
        线程是在进程中执行的某个任务,可以访问进行的共享数据。一个进程各种多个线程共享堆空间,栈空间是独立的,线程消耗的资源比进程小的多。
      线程调度:
        就是指CPU快速地在多个线程间进行切换
        CPU是单核的:那么这个CPU的某个时刻只能运行一个线程。
        因为CPU可以快速地在多个线程间,进行快速切换。
      CPU的两种调度:
        分时调度:
            按照时间片,多个线程轮流拥有CPU的执行权
        抢占式调度:
            CPU在多个线程之间进行随机分配(跟线程的优先级有关)
        java中使用抢占式调度,所有Java中多个线程的运行也是随机的
  3.创建新的线程方式***********
    当我们运行java程序时,java程序就变成java进程
    其中至少有两条线程,一个main线程,一条JC(java的垃圾回收器)线程
    创建新的线程的方式一:
        描述:一个是将一个类声明为一个Thread的子类
        这个子类应该重写run类的方法,然后可以分配并启动子类的实例
    步骤:
       创建子类 extends Thread
      重写Thread类中的run方法(就是线程“main”方法,写的就是子线程的任务代码)
      创建子类的对象(在main里边写)
      启动子线程 子类.start()
    代码:
      子线程:

public class Childthread extends Thread{
@Override
public void run() {
for (int i = 0; i < 10; i++) {
System.out.println("子线程");
}
}
}

      主线程:

public static void main(String[] args) {
Childthread childthread = new Childthread();
//childthread.run(); 当使用run时,就是普通方法的调用,先执行完再往下执行。
childthread.start();
for (int i = 0; i < 50; i++) {
System.out.println("主线程iiiiiiiii");
}
}

  注:
    堆空间是有垃圾回收器



 































































































































































































以上是关于异常,线程的主要内容,如果未能解决你的问题,请参考以下文章

使用实体框架迁移时 SQL Server 连接抛出异常 - 添加代码片段

异常和TCP通讯

片段中的Android致命异常

newCacheThreadPool()newFixedThreadPool()newScheduledThreadPool()newSingleThreadExecutor()自定义线程池(代码片段

mvn命令异常:An error has occurred in Javadoc report generation: Unable to find javadoc command异常已解决(代码片段

mvn命令异常:An error has occurred in Javadoc report generation: Unable to find javadoc command异常已解决(代码片段