多线程:
进程:是一个正在执行中的程序。每一个进程执行都有一个执行顺序,该顺序是一个执行路径,或者叫一个控制单元
线程:是进程中的一个独立的控制单元,线程控制着进程的执行。
一个进程至少有一个线程。
如何在自定义的代码中,自定义一个线程?
1、定义类继承Thread;
2、复写Thread类中的run方法;
3、调用线程start方法;有两个作用:启动线程调用run方法
多线程的运行行为可以看成是在抢夺cpu资源;多线程的一个特性:随机性,谁抢到谁执行,执行多长cpu说了算;
线程都有自己默认的名称 Thread-编号,该编号从0开始;
static Thread currentThread():获取当前线程对象
getName():获取线程名称。
设置线程名称:setName或者构造函数。
创建线程的第二种方式,实现Runnable接口
步骤:
1、定义类实现Runnable接口。
2、覆盖Runnable接口中的run方法。
3、通过Thread类建立的对象。
4、将Runnable接口的子类对象作为实际参数传递给Thread类的构造函数。
why? 因为自定义的run方法所属的对象是Runnable接口的子类对象,所以要让线程去指定对象的run方法,必须明确所属的对象。
5.、调用Thread类的start方法开启线程并调用Runnable接口子类的run方法。
实现方式和继承方式有什么区别?
实现方式的好处:避免了单继承的局限性。
在定义线程时,建议使用接口实现。
继承Thread线程代码存放在Thread子类run方法中。
实现Runnable线程代码存放在接口子类的run方法中。
当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行一部分,还没有执行完,另一个线程参与进来执行,导致共享数据错误。
java解决方法就是同步代码块。
synchronized(对象) { 需要同步的代码 }
对象如同锁,持有锁的线程可以在同步中执行。没有持有锁的线程即使获取的cpu的执行权,也进不去,因为没有获取锁。
同步的前提:
1,必须是要有两个或者两个以上的线程。
2,必须是多个线程使用同一个锁。
必须保证同步中只能有一个线程在运行
好处:解决了多线程的安全
弊端:多个线程需要判断锁,较为消耗资源
同步函数默认的锁是this;
同步函数被静态修饰后默认的锁不是this,因为静态方法中也不可以定义this,静态进内存时,内存张没有本类对象,但是一定有该类对应的字节码文件对象,
类名.class 该对象的类型是class
静态的同步方法使用的锁是该方法所在类的字节码文件对象, 类名.class;
死锁
线程间通信:
其实就是多个线程在操作同一个资源,但是操作的动作不同
wait() notify() notifyAll(); 都使用在同步个,因为要对持有监视器(锁)的线程操作,所以要在使用在同步中,因为只有同步才具有锁。
为什么这些操作线程的方法要定义在object类中?
因为这些方法在操作同步线程时,都必须要标识他们所在操作线程只有的锁,只有在同一个锁上的被等待线程,可以被同一个锁上notify唤醒,不可以对不同锁中的线程进行唤醒也就是说,等待和唤醒必须是同一个锁。
生产者消费者栗子
class Res
{
private String name;
private String sex;
private boolean flag = false ;
public synchronized void set(String name ,String sex)
{
if (flag)
{
try {this.wait();} catch (Exception e){}
}
this.name = name;
this.sex = sex;
flag = true;
this.notify();
}
public synchronized void out()
{
if(!flag)
try{this.wait();} catch (Exception e){}
System.out.println(name+"...."+sex);
flag = false;
this.notify();
}
}
class Input implements Runnable
{
int x;
Res r = new Res();
Input(Res r)
{
this.r = r;
}
public void run()
{
while(true)
{
synchronized (r)
{
// if (r.flag)
// {
// try {r.wait();} catch (Exception e){}
// }
if ( x ==0)
{
r.set("mike","man");
}
else
{
r.set("jeans","woman");
}
x = (x+1)%2;
// r.flag = true;
// r.notify();
}
}
}
}
class Output implements Runnable
{
private Res r;
Output (Res r)
{
this.r = r;
}
public void run()
{
while (true)
{
// synchronized (r)
// {
// // if (!r.flag)
// // {
// // try {r.wait();} catch (Exception e){}
// // }
// // System.out.println(r.name+"...."+r.sex);
// // r.flag = false;
// // r.notify();
// }
r.out();
}
}
}
class InputOutputDemo
{
public static void main(String args[])
{
Res r = new Res();
Input in = new Input(r);
Output out = new Output(r);
Thread t1 = new Thread(in);
Thread t2 = new Thread(out);
t1.start();
t2.start();
}
}
如何停止线? 只有一种,run方法结束,开启多线程运行,运行代码通常是循环结构,只要控制住循环,就可以让run方法结束,也就是线程结束。
特殊情况:当线程处于了冻结状态,就不会读取到标记,那么线程就不会结束。
当没有指定的方式让冻结的线程恢复到运行状态时,这是需要对冻结进行清除,强制让线程恢复到运行状态中来,这样就可以操作标记让线程结束。
Thread类中提供了该方法, interrupt();
举个栗子:
class StopThreadDemo
{
public static void main(String args[])
{
StopThread st = new StopThread();
Thread t1 = new Thread(st);
Thread t2 = new Thread(st);
t1.start();
t2.start();
int num = 0;
while (true)
{
if (num++ ==60 )
{
// st.changeFlag();
t1.interrupt();
t2.interrupt();
break;
}
System.out.println(Thread.currentThread().getName()+"....."+num);
}
}
}
class StopThread implements Runnable
{
private boolean flag = true;
public synchronized void run()
{
while (flag)
{
try
{
wait();
}
catch(InterruptedException e)
{
System.out.println(Thread.currentThread().getName()+"....Exception");
flag = false;
}
System.out.println(Thread.currentThread().getName()+"....run");
}
}
public void changeFlag()
{
flag = false;
}
}