Java并发编程——简单聊天室实现

前言

最近在面试中经常被问到有没有java多线程知识以及多线程编程的经验,因而促成了本次总结,本文是一个简单整合,搜集了以下几篇博客:
1、java多线程编程基本知识:
Java并发编程——如何创建线程以及Thread类的使用
1️⃣java中创建线程的方式:继承Thread类、实现Runnable接口

1、需要重写run()方法:将自定义业务逻辑存储在run方法,让线程运行
2、实现Runnable接口相比继承Thread类有如下优势:
    <1>、可以避免由于Java的单继承特性而带来的局限;
    <2>、增强程序的健壮性,代码能够被多个线程共享,代码与数据是独立的;
    <3>、适合多个相同程序代码的线程区处理同一资源的情况。

2️⃣线程的状态、线程的上下文切换
3️⃣Thread类中常用方法

1)start方法
  start()用来启动一个线程,当调用start方法后,系统才会开启一个新的线程来执行用户定义的子任务,在这个过程中,
  会为相应的线程分配需要的资源。

2)run方法
  run()方法是不需要用户来调用的,当通过start方法启动一个线程之后,当线程获得了CPU执行时间,便进入run方法
  体去执行具体的任务。注意,继承Thread类必须重写run方法,在run方法中定义具体要执行的任务。

3)sleep方法
	sleep方法有两个重载版本:
	sleep(long millis)     //参数为毫秒
	sleep(long millis,int nanoseconds)    //第一参数为毫秒,第二个参数为纳秒

	sleep相当于让线程睡眠,交出CPU,让CPU去执行其他的任务。但是有一点要非常注意,sleep方法不会释放锁,也就是说
如果当前线程持有对某个对象的锁,则即使调用sleep方法,其他线程也无法访问这个对象。注意,如果调用了sleep方法,
必须捕获InterruptedException异常或者将该异常向上层抛出。当线程睡眠时间满后,不一定会立即得到执行,因为此时
可能CPU正在执行其他的任务。所以说调用sleep方法相当于让线程进入阻塞状态。

4)yield方法
  调用yield方法会让当前线程交出CPU权限,让CPU去执行其他的线程。它跟sleep方法类似,同样不会释放锁。
  但是yield不能控制具体的交出CPU的时间,另外,yield方法只能让拥有相同优先级的线程有获取CPU执行时间的机会。
  yield可以直接用Thread类调用,yield让出CPU执行权给同等级的线程,如果没有相同级别的线程在等待CPU的执行权,
  则该线程继续执行。注意,调用yield方法并不会让线程进入阻塞状态,而是让线程重回就绪状态,它只需要等待
  重新获取CPU执行时间,这一点是和sleep方法不一样的。

5)join方法

	join方法有三个重载版本:
	join()
	join(long millis)     //参数为毫秒
	join(long millis,int nanoseconds)    //第一参数为毫秒,第二个参数为纳秒
	假如在main线程中,调用thread.join方法,则main方法会等待thread线程执行完毕或者等待一定的时间。如果调用
的是无参join方法,则等待thread执行完毕,如果调用的是指定了时间参数的join方法,则等待一定的时间。join方法
用线程对象调用,如果在一个线程A中调用另一个线程B的join方法,线程A将会等待线程B执行完毕后再执行。
实际上调用join方法是调用了Object的wait方法,这个可以通过查看源码得知。wait方法会让线程进入阻塞状态,
并且会释放线程占有的锁,并交出CPU执行权限。由于wait方法会让线程释放对象锁,所以join方法同样会让线程释放
对一个对象持有的锁。

6)interrupt方法
	interrupt,顾名思义,即中断的意思。单独调用interrupt方法可以使得处于阻塞状态的线程抛出一个异常,也就说,
它可以用来中断一个正处于阻塞状态的线程;直接调用interrupt方法不能中断正在运行中的线程。另外,
通过interrupt方法和isInterrupted()方法来停止正在运行的线程。

7)stop方法
	stop方法已经是一个废弃的方法,它是一个不安全的方法。因为调用stop方法会直接终止run方法的调用,
并且会抛出一个ThreadDeath错误,如果线程持有某个对象锁的话,会完全释放锁,导致对象状态不一致。
所以stop方法基本是不会被用到的。

8)destroy方法
	destroy方法也是废弃的方法。基本不会被使用到。

以下是关系到线程属性的几个方法:
1)getId
	用来得到线程ID

2)getName和setName
	用来得到或者设置线程名称。

3)getPriority和setPriority
	用来获取和设置线程优先级。

4)setDaemon和isDaemon
	用来设置线程是否成为守护线程和判断线程是否是守护线程。

	守护线程和用户线程的区别在于:守护线程依赖于创建它的线程,而用户线程则不依赖。举个简单的例子:
如果在main线程中创建了一个守护线程,当main方法运行完毕之后,守护线程也会随着消亡。而用户线程则不会,
用户线程会一直运行直到其运行完毕。在JVM中,像垃圾收集器线程就是守护线程。

	Thread类有一个比较常用的静态方法currentThread()用来获取当前线程。

2、java多线程编程简单实战:
Java小项目——聊天室(多线程版本)

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值