java自学笔记之多线程

多线程:

进程:是一个正在执行中的程序。每一个进程执行都有一个执行顺序,该顺序是一个执行路径,或者叫一个控制单元

线程:是进程中的一个独立的控制单元,线程控制着进程的执行。

一个进程至少有一个线程。  

如何在自定义的代码中,自定义一个线程?

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;
	}
}
	


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值