Java(六)-Java异常与多线程

2 篇文章 0 订阅

异常

异常的概念与体系

在这里插入图片描述
在这里插入图片描述

异常的分类

在这里插入图片描述

public static void main(String[] args) throws ParseException {
    SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
    Date date = simpleDateFormat.parse("202106-01"); //ParseException
    System.out.println(date);
}

public static void main(String[] args) {
    SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
    Date date = null;
    try {
        date = simpleDateFormat.parse("202106-01");//ParseException
    } catch (ParseException e) {
        e.printStackTrace();
    }
    System.out.println("后续代码");
    System.out.println(date);
}

    int[] a = new int[3];
    System.out.println(a[3]); //ArrayIndexOutOfBoundsException

    int[] a = new int[1024*1024*1024];
    System.out.println(a[3]);//OutOfMemoryError

异常的产生过程

在这里插入图片描述
【注意】JVM在其中有着巨大的作用

异常的处理

异常处理的5个关键字:try,catch,finally,throw,throws.
throw关键字
在这里插入图片描述

public static int getElement(int[] arr,int index)  {
    if (arr == null) {
        // NullPointerException运行期异常(runtimeException
        throw new NullPointerException("传递的数组是空");
    }
    if (index < 0 || index >= arr.length) {
        //ArrayIndexOutOfBoundsException是运行期异常(runtimeException
        throw new ArrayIndexOutOfBoundsException("数组越界了!");
    }
    return arr[index];
}

objects的方法
在这里插入图片描述
下边的写法是等价的

 Objects.requireNonNull(arr, "传递的数组是空");
 /*if (arr == null) {
     // NullPointerException运行期异常
     throw new NullPointerException("传递的数组是空");
 }*/

throws关键字:异常的第一种处理方法
在这里插入图片描述
在这里插入图片描述
try…catch: 异常的第二种使用方法
在这里插入图片描述

public static void main(String[] args) {
    try {
        readFile("d:\\a.tx");
    } catch (IOException e) {
        System.out.println("文件路径不对");
    }

    System.out.println(666666666);
}

public static void readFile(String fileName) throws IOException{
    if (!fileName.endsWith(".txt")) {
        throw new IOException("文件的后缀名不对!");
    }
}

throwable类的3个异常处理
在这里插入图片描述
finally代码块
在这里插入图片描述
多异常的捕获处理
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
finally中有return语句
在这里插入图片描述
【注意】尽量避免finally中有return语句。
字符类异常
在这里插入图片描述
在这里插入图片描述
自定义异常类
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

多线程

并发与并行

进程

程序和进程

线程

在这里插入图片描述
线程调度
分时调度
抢占式的(优先级,否则随机),Java使用的是抢占式
主线程(main线程)
主线程:执行主(main)方法的线程
单线程程序:Java程序中只有一个线程
执行从main开始,从上到下依次执行
在这里插入图片描述
创建多线程的第一种方法(继承Thread类)在这里插入图片描述
多线程原理
在这里插入图片描述
多线程原理_多线程内存图解
在这里插入图片描述
【注意】栈空间
线程的名称
在这里插入图片描述
在这里插入图片描述
设置线程的名称
在这里插入图片描述
常用方法——Sleep
在这里插入图片描述
【注意】,是对当前线程的,可以使用Thread的静态方法。
创建多线程程序的第二种方式
在这里插入图片描述

public class Demo01Runnable implements Runnable {
    @Override
    public void run() {
        System.out.println("run");
    }
}

public static void main(String[] args) {
    Demo01Runnable demo01Runnable = new Demo01Runnable();
    new Thread(demo01Runnable).start();
}

使用接口的好处(thread和runnable的区别)
在这里插入图片描述
匿名内部类实现线程
在这里插入图片描述

public class Demo01InnerClassThread {
    public static void main(String[] args) {
        new Thread(){
            @Override
            public void run(){
                for (int i = 0; i < 20; i++) {
                    System.out.println("liu"+"--->"+i);
                }
            }
        }.start();


        Runnable r = new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 20; i++) {
                    System.out.println("chun"+"--->"+i);
                }
            }
        };
        new Thread(r).start();
        
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 20; i++) {
                    System.out.println("lei"+"--->"+i);
                }
            }
        }).start();
    }
}

线程同步机制

线程安全概述
在这里插入图片描述
多线程访问了共享的数据,会产生线程安全问题。

public class Demo05Test {
    public static void main(String[] args) {
        //创建一个实现类对象,供三个线程进行调用
        RunnableImpl r = new RunnableImpl();
        Thread t1 = new Thread(r);
        Thread t2 = new Thread(r);
        Thread t3 = new Thread(r);
        //开启线程
        t1.start();
        t2.start();
        t3.start();
    }
}

线程安全问题产生的原理
在这里插入图片描述
解决线程安全的方法
1.同步代码块
2.同步方法
3.锁机制
1.同步代码块
在这里插入图片描述

private int ticket=100;
Object obj = new Object();
@Override
public void run() {
    while (true) {
        synchronized(obj) {
            if (ticket > 0) {
                try {
                    Thread.sleep(10);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName() + "--->正在卖第" + ticket + "张票");
                ticket--;
            }
        }
    }
}

同步技术的原理
在这里插入图片描述
2.同步方法
在这里插入图片描述

public synchronized void ticketSent(){
    if (ticket > 0) {
        try {
            Thread.sleep(10);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(Thread.currentThread().getName() + "--->正在卖第" + ticket + "张票");
        ticket--;
    }
}

静态同步方法
在这里插入图片描述
【注意】RunnableImpl.class也可以加锁
3.Lock锁
在这里插入图片描述

public class RunnableImpl implements Runnable {

    private int ticket=100;
    Lock l = new ReentrantLock();
    @Override
    public void run() {
        while (true) {
            l.lock();
            if (ticket > 0) {
                try {
                    Thread.sleep(10);
                    System.out.println(Thread.currentThread().getName() + "--->正在卖第" + ticket + "张票");
                    ticket--;
                } catch (InterruptedException e) {
                    e.printStackTrace();
                } finally {
                    l.unlock();//无论程序是否异常,都会释放锁
                }
            }
        }
    }
}

线程等待唤醒机制

线程状态图
在这里插入图片描述
等待唤醒案例
在这里插入图片描述
在这里插入图片描述

public static void main(String[] args) {
    Object obj = new Object();

    new Thread(){
        @Override
        public void run() {
            synchronized (obj) {
                System.out.println("告诉老板信息");
                try {
                    obj.wait();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("拿到包子了");
            }
        }
    }.start();

    new Thread(){
        @Override
        public void run() {
            try {
                Thread.sleep(5000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            synchronized (obj){
                System.out.println("老板花了5秒,做好包子了");
                obj.notify();
            }
        }
    }.start();
}

在这里插入图片描述
【注意】notify是随机的唤醒一个等待线程
线程间的通信(对资源的使用需要商量着来)
等待唤醒机制
在这里插入图片描述
在这里插入图片描述
线程池:频繁创建和销毁线程太耗费资源,线程池可以复用线程
在这里插入图片描述

在这里插入图片描述
线程池
在这里插入图片描述

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值