第十二周
什么是序列化
对象序列化(Serialize) 指将一个Java对象写入IO流中,与此对应的是,对象的反序列化(Deserialize) 则指从IO流中恢复该Java对象。如果想让某个Java对象能够序列化,则必须让它的类实现java.io.Serializable 接口
序列化
ObjectOutputStream 类继承了OutputStream 类,
同时实现了ObjectOutput接口,提供将对象序列化并写入流中的功能
对象流(高级)作用:用于Java对象写入文件(被写的对象必须实现接口)
反序列化
ObjectInputStream类继承了InputStream类,
同时实现了ObjectInput接口,提供了将对象序列化并从流中读取出来的功能
输入对象流 (高级流)
作用:将Java对象还原出来
要求:还原对象的序列化版本号 必须要求相同
序列编号
序列化编号有两种显示生成方式:
1.默认的1L,比如: private static final long serialVersionUID = 1L
2.根据类名、接口名、成员方法及属性等来生成一个64位的哈希字段。
多线程
进程:是计算机中的程序关于某数据集合上的一次运行运动,是操作系统结构的基础,进程是程序的基本执行实体,在当代面向线程设计的计算机结构中,进程是线程的容器,程序是指令,数据及其组织的描述,进程是程序的实体。
程序 = 算法 + 数据结构
线程:是操作系统能够进行运算调度的最小单位,它被包含在进程内,进程是实际运作单位,一个进程可以并发多个线程。
线程和进程
进程和线程是包含关系,但是多任务既可以由多进程实现,也可以由单进程内的多线程实现,还可以混合多进程+多线程。
和多线程相比,多进程的缺点在于:
1、创建进程比创建线程开销大,尤其是在Windows系统上;
2、进程间通信比线程间通信要慢,因为线程间通信就是读写同一个变量,速度很快。
而多进程的优点在于:
多进程稳定性比多线程高,因为在多进程的情况下,一个进程崩溃不会影响其他进程,而在多线程的情况下,任何一个线程崩溃 会直接导致整个进程崩溃。
Java语言内置了多线程支持:
一个Java程序实际上是一个JVM进程,JVM进程用一个主线程来执行`main()`方法,在`main()`方法内部,我们又可以启动多个线程。此外,JVM还有负责垃圾回收(jc)的其他工作线程等
对于大多数Java程序来说,我们说多任务,实际上是说如何使用多线程实现多任务
Java多线程编程的特点又在于:
1.多线程模型是Java程序最基本的并发模型;
2.网络、数据库、Web开发等都依赖Java多线程模型。
创建新线程
1.通过继承Thread来创建线程
2.实现 Runnable 接口创建线程
线程的优先级
线程的状态
新建:当一个Thread类或其子类的对象被声明并创建时,新生的线程对象处于新建状态
就绪:处于新建状态的线程被start()后,将进入线程队列等待CPU时间片,此时它已具备了运行的条件,只是没分配到CPU资源
运行:当就绪的线程被调度并获得CPU资源时,便进入运行状态,run()方法定义了线程的操作和功能
阻塞:在某种特殊情况下,被人为挂起或执行输入输出操作时,让出 CPU并临时中止自己的执行,进入阻塞状态
死亡:线程完成了它的全部工作或线程被提前强制性地中止或出现异常导致结束
在Java程序中,一个线程对象只能调用一次start()方法启动新线程,并在新线程中执行run()方法。一旦run()方法执行完毕,线程就结束了。Java线程的状态有以下几种:
1.New:新创建的线程,尚未执行;
2.Runnable:运行中的线程,正在执行`run()`方法的Java代码;
3.Blocked:运行中的线程,因为某些操作被阻塞而挂起;
4.Waiting:运行中的线程,因为某些操作在等待中;
5.Timed Waiting:运行中的线程,因为执行`sleep()`方法正在计时等待;
6.Terminated:线程已终止,因为`run()`方法执行完毕。
线程终止的原因有:
1.线程正常终止:run()方法执行到return语句返回;
2.线程意外终止:run()方法因为未捕获的异常导致线程终止;
3.对某个线程的Thread实例调用stop()方法强制终止(强烈不推荐使用)。
守护线程
Java程序入口就是由JVM启动main线程,main线程又可以启动其他线程。当所有线程都运行结束时,JVM退出,进程结束。如果有一个线程没有退出,JVM进程就不会退出。
所以,必须保证所有线程都能及时结束。但是有一种线程的目的就是无限循环,
守护线程又称为后台线程,默认创建的线程都是普通线程,或者称为前台线程。
线程提供了一个方法: void setDaemon(boolean on)
只有调用该方法并且传入参数为true时,该线程才会被设置为守护线程。
守护线程在使用上与普通线程没有差别,但是在结束时会有一个差别。
即:线程结束时,所有正在运行的守护线程都会被强制停止 * 进程结束:当一个进程中所有的普通线程都结束时,进程才会结束
并发线程
多线程并发的安全问题
产生:当多个线程并发操作时,由于线程切换实际的不确定性,会导致操作资源的代码顺序未按照设计顺序执行,出现操作混乱的情况,严重时可能导致系统瘫痪
解决:将并发操作同一资源改为同步运行,有先后顺序的操作
同步与异步:
同步:程序运行有先后顺序
异步:程序运行没有先后顺序
同步方法小结:
1、用synchronized修饰方法可以把整个方法变为同步代码块,synchronized方法加锁对象是this;
2、通过合理的设计和数据封装可以让一个类变为“线程安全”;
3、一个类没有特殊说明,默认不是thread-safe;
4、多线程能否安全访问某个非线程安全的实例,需要具体问题具体分析。
死锁
死锁发生后,没有任何机制能解除死锁,只能强制结束JVM进程。
在编写多线程应用时,要特别注意防止死锁。因为死锁一旦形成,就只能强制结束进程。
乐观锁和悲观锁是两种思想,用于解决并发场景下的数据竞争问题
线程池
作用:重复利用资源
Java语言虽然内置了多线程支持,启动一个新线程非常方便,但是,创建线程需要操作系统资源(线程资源,栈空间等),频繁创建和销毁大量线程需要消耗大量时间。