****背景:****以单核CPU为例,只使用单个线程先后完成多个任务(调用多个方 法),肯定比用多个线程来完成用的时间更短,为何仍需多线程呢?
多线程程序的优点:
应用场景:
public class Thread_01_Create {
public static void main(String[] args) {
// 创建线程对象,直接new即可
Thread thread=new Processor1();
// 线程启动
thread.start();
for (int i = 0; i < 5; i++) {
System.out.println("main线程:"+i);
}
}
}
// 继承 Thread 类的方法
class Processor1 extends Thread{
@Override
public void run() {//实现接口run()方法
for (int i = 0; i < 5; i++) {
System.out.println("测试线程:"+i);
}
}
}
public class Thread_02_Create {
public static void main(String[] args) {
// 继承Thread类和实现Runnable 接口创建对象时会有所差别
Thread thread=new Thread(new Processor2());
// 启动线程
thread.start();
for (int i = 0; i < 5; i++) {
System.out.println("main线程:"+i);
}
}
}
// 实现 Runnable 接口的方式
class Processor2 implements Runnable{
@Override
public void run() {//都要实现run()方法
for (int i = 0; i < 5; i++) {
System.out.println("接口线程:"+i);
}
}
}
优先使用实现,保留类继承
源码:
/**
* The minimum priority that a thread can have.
*/
public final static int MIN_PRIORITY = 1;
/**
* The default priority that is assigned to a thread.
*/
public final static int NORM_PRIORITY = 5;
/**
* The maximum priority that a thread can have.
*/
public final static int MAX_PRIORITY = 10;
//设置优先级
setPriority
//获取优先级
getPriority
//获取线程名称
getName
//设置线程名字,如果不设置名字,默认为 Thread_0 开始,以此类推
setName
//获取当前线程对象
static currentThread
//让当前线程进入睡眠状态,参数为毫秒
static sleep
public class Thread_03_Priority {
public static void main(String[] args) {
Thread t1=new Processor3();
Thread t2=new Processor3();
// 设置线程名字,要在线程启动之前设置
t1.setName("t1");
t2.setName("t2");
// 设置线程优先级 1(min)~10(max)
t1.setPriority(1);
t2.setPriority(10);
// 想要调用线程一定要 启动线程
t1.start();
t2.start();
try {
// 静态方法,现在哪里就对谁生效,此时这个就对main方法生效
// 因为最后都等于类名调用,所以main方法 最后执行完毕
t1.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
for (int i = 0; i < 5; i++) {
System.out.println(Thread.currentThread().getName()+"-->"+i);
}
}
}
class Processor3 extends Thread{
@Override
public void run() {
for (int i = 0; i < 5; i++) {
//currentThread() 获取当前线程对象
//getName()获取对象名字
System.out.println(Thread.currentThread().getName()+"-->"+i);
}
}
}
public class Thread_04_Interrupt {
public static void main(String[] args) {
Thread thread=new Thread(new Processor4());
// 不要忘记启动线程 start !!!
thread.start();
try {
thread.sleep(5000);
// 5s后会打断thread线程,会抛出异常
thread.interrupt();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
class Processor4 implements Runnable{
@Override
public void run() {
try {
//写在Processor4中所以对Processor4生效
Thread.sleep(10000);
System.out.println("睡醒了");
} catch (InterruptedException e) {
e.printStackTrace();
System.out.println("被叫醒");
}
System.out.println("执行结束");
}
}
public class Thread_05_Stop {
public static void main(String[] args) {
Processor5 p5=new Processor5();
p5.start();
try {
Thread.sleep(3000);
// 直接结束p5进程不推荐使用(有删除线),可能会导致死锁
//p5.stop();
p5.run=false;
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
class Processor5 extends Thread{
boolean run=true;
@Override
public void run() {
for (int i = 0; i < 5; i++) {
if(!run){
return;
}
System.out.println(Thread.currentThread().getName()+"-->"+i);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println("测试线程结束");
}
}
public class Thread_06_Join {
public static void main(String[] args) throws InterruptedException {
Thread t1=new Thread(new Processor6());
Thread t2=new Thread(new Processor6());
t1.setName("t1");
t2.setName("t2");
t1.start();
t2.start();
// 把t1和main合并,就是把t1写到main线程中,t1执行完后main再执行
//对t2没有影响
t1.join();
for (int i = 0; i <5; i++) {
System.out.println(Thread.currentThread().getName()+"-->"+i);
Thread.sleep(1000);
}
}
}
class Processor6 implements Runnable{
@Override
public void run() {
for (int i = 0; i < 5; i++) {
System.out.println(Thread.currentThread().getName()+"-->"+i);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
public class Thread_07_Yield {
public static void main(String[] args) throws InterruptedException {
Thread t1=new Thread(new Processor7());
t1.setName("t1");
t1.setPriority(5);
Thread.currentThread().setPriority(5);
t1.start();
for (int i = 0; i <10; i++) {
if(i%3==0){
// yield() : 给 同优先级 让位 , 只有资源不够时才会进行让位
Thread.yield();
}
System.out.println(Thread.currentThread().getName()+"-->"+i);
}
}
}
class Processor7 implements Runnable{
@Override
public void run() {
for (int i = 0; i < 10; i++) {
System.out.println(Thread.currentThread().getName()+"-->"+i);
}
}
}
*问题的提出*
多个线程执行的不确定性引起执行结果的不稳定
多个线程对账本的共享,会造成操作的不完整性,会破坏数据
public class Thread_08_Synchronization {
public static void main(String[] args) {
Account account=new Account("1001",666);
Thread t1=new Thread(new Processor8(account));
Thread t2=new Thread(new Processor8(account));
t1.start();
t2.start();
}
}
class Processor8 implements Runnable{
Account account;
@Override
public void run() {
account.withDraw(100);
System.out.println(Thread.currentThread().getName()+"还剩:"+account.getBalance());
}
public Processor8(Account account){
this.account=account;
}
}
// 实体类
class Account{//账户
private String actNO;//编号
private double balance;//存款
// 修改余额
public void withDraw(double money){
System.out.println(Thread.currentThread().getName());
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
balance=balance-money;
}
public double getBalance() {
return balance;
}
public void setBalance(double balance) {
this.balance = balance;
}
public Account(String actNO, double balance) {
super();
this.actNO = actNO;
this.balance = balance;
}
public Account() {
super();
}
}
对修改余额的方法进行加锁处理,不让多个线程同时进入执行
// 使用 synchornized 修饰符 修饰方法后,该方法只能有一个线程进入,其他线程必须等这个线程执行完后才能进入
public synchronized void withDraw(double money){
System.out.println(Thread.currentThread().getName());
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
balance=balance-money;
}
public void withDraw(double money){
System.out.println(Thread.currentThread().getName());
// 语句块锁,只需要锁住需要同步的代码即可,对方法内的其他代码不会同步
synchronized (this) {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
balance=balance-money;
}
}
从 JDK 5.0开始,Java提供了更强大的线程同步机制——通过显式定义同
步锁对象来实现同步。同步锁使用Lock对象充当。
java.util.concurrent.locks.Lock 接口是控制多个线程对共享资源进行访问的工具。锁提供了对共享资源的独占访问,每次只能有一个线程对 Lock 对象加锁,线程开始访问共享资源之前应先获得 Lock 对象。
ReentrantLock 类实现了 Lock ,它拥有与 synchronized 相同的并发性和内存语义,在实现线程安全的控制中,比较常用的是ReentrantLock,可以显式加锁、释放锁。
// 创建锁对象
Lock lock=new ReentrantLock();
public void withDraw(double money){
System.out.println(Thread.currentThread().getName());
// 开启锁,同步
lock.lock();
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
balance=balance-money;
// 释放锁,结束同步
lock.unlock();
}
public class Thread_10_Daemon {
public static void main(String[] args) {
Thread t1=new Thread(new Processor10());
//设置守护线程
t1.setDaemon(true);// 要在线程开始执行之前设置守护线程
t1.start();
for (int i = 0; i < 5; i++) {
try {
Thread.sleep(1000);
System.out.println(Thread.currentThread().getName()+"-->"+i);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
class Processor10 implements Runnable{
@Override
public void run() {
int i=1;
while (true) {
try {
Thread.sleep(1000);
System.out.println(Thread.currentThread().getName()+"-->"+i++);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
public class Thread_11_Timer {
public static void main(String[] args) {
// 创建定时器
Timer time=new Timer();
// 开启定时器
// 第一个参数:要做的事,TimerTask子类对象
// 第二个参数:开始时间,传入毫秒代表多久之后开始运行,也可以传入Date对象
// 第三个参数:间隔时间
time.schedule(new LogTimerTask(), 3000,1000);// 开启新线程,不会影响到其他线程的运行
}
}
class LogTimerTask extends TimerTask{
@Override
public void run() {
// 要做的事
for (int i = 0; i < 5; i++) {
System.out.println(i);
}
}
}