Java多线程1——概述

一、概念

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

2.线程:就是进程中的一个独立的控制单元。线程在控制着进程的执行。

3.java虚拟机启动的时候会有一个进程java.exe,该进程中至少有一个线程负责java程序的执行,而且这个线程运行的代码存在于main方法中,该线程称之为主线程。

二、创建线程

   方法一:继承Thread类
    步骤:1.定义类继承Thread。
	  2.重写Thread类中的run方法
		目的:存储自定的代码,让线程去运行。
	  3.调用线程的start方法(该方法两个作用:启动线程,调用run方法 
Demo1:继承Thread类创建线程
class Demo extends Thread {
	Demo(String name) {
		super(name);
	}
	
	public void run() {
		for(int x=0; x<60; x++) {
			System.out.println(Thread.currentThread().getName() + "run-----" + x);
		}
	}
}
public class ThreadDemo {
	public static void main(String[] args) {
		Demo d1 = new Demo("one---");
		Demo d2 = new Demo("two---");
		d1.start();
		d2.start();
		
		for(int x=0; x<60; x++) {
			System.out.println("Hello World---" + x);
		}
	}
}
P.S   1.线程的名字可以调用父类的构造函数,直接定义的时候就传进去
       2.获取当前线程:Thread.currentThread()
      3.获取线程名字:getName()
 
 
    
方法二:实现Runnable接口

	步骤:  1.定义类实现Runnable接口
		2.重写Runnable接口中的run方法
		3.通过Thread类建立线程对象
		4.将Runnable接口的子类对象最为实际参数传递给Thread类的构造函数
		5.调用Thread类的start方法开启线程并调用Runnable接口子类的run方法
P.S    由于Java是只支持单继承,当某些类已经有父类时,就不能再继承Thread类,这时应该使用Runnable接口实现多线程,这种方法也是比较常用的方法。
Dmeo2:实现Runnable接口创建线程
class TickDemo implements Runnable {
	private int tick = 100;
	public void run() {
		while(true) {
			if(tick > 0) {
				System.out.println(Thread.currentThread().getName() + "sale--" + tick--);
			}
		}
	}
}
public class Ticket {
	public static void main(String[] args) {
		TickDemo t = new TickDemo();
		
		Thread t1 = new Thread(t);
		Thread t2 = new Thread(t);
		t1.start();
		t2.start();	
	}
}
    运行一下,能够把100张票顺利卖完,但是这个程序隐含了一个安全问题。假如有这种情况,当票只剩一张的时候,t1线程判断进去判断if语句,票大于0,所以进入,但是进去之后在输出语句执行之前,t1线程的执行权被t2线程给抢走了,这时t2线程也判断if语句,也成功进入了,那么当t1,t2都执行完后将会卖掉101张票,这就出问题了。要模拟这种情况我们可以在输出语句执行之前加个sleep语句,手动让先进来的线程放弃执行权,让另一个线程进入。
    经过分析我们得出这样的结论,产生这个问题的原因:当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一部分,还没有执行完,另一个线程参与进来执行。导致共享数据的错误。解决办法:对多条操作共享数据的语句,只能让一个线程都执行完,在执行过程中,其他线程不可以参与执行。
    Java对多线程安全问题提供了专业的解决方式,其中一种就是同步代码块:
    synchronized(对象){
	需要被同步的代码
    }
Demo3:synchronized同步代码块

class TickDemo implements Runnable {
	private int tick = 100;
	Object obj = new Object();
	public void run() {
		while(true) {
			synchronized(obj){
				if(tick > 0) {
					System.out.println(Thread.currentThread().getName() + "sale--" + tick--);
				}
			}
		}
	}
}
public class Ticket {
	public static void main(String[] args) {
		TickDemo t = new TickDemo();
		
		Thread t1 = new Thread(t);
		Thread t2 = new Thread(t);
		t1.start();
		t2.start();
		
	}
}

    这段代码运行完票可以正常卖光,synchronized(对象)这里面的对象可以是任何对象,为了方便这里直接new了一个object对象,这个对象如同一把锁,当有一个线程进入synchronized()后,这个锁就锁上了,其他线程想进来也进不来,只有当第一个线程执行完synchronized这块代码后,锁才会重新打开,后面的线程就可以进入了,当然一个线程进入后,又会锁上,这样就保证了线程的安全性。不过安全的东西总要付出代价的,就是速度会变慢,因为每次执行代码都要判断是否可以进入。
    synchronized不仅可以用于代码块,也可以用在函数上
Demo4:synchronized同步函数

class TickDemo implements Runnable {
	private int tick = 100;
//	Object obj = new Object();
	public void run() {
		while(true) {
			show();
		}
	}
	public synchronized void show() {
		if(tick > 0) {
			System.out.println(Thread.currentThread().getName() + "sale--" + tick--);
		}
	}
}
public class Ticket {
	public static void main(String[] args) {
		TickDemo t = new TickDemo();
		
		Thread t1 = new Thread(t);
		Thread t2 = new Thread(t);
		t1.start();
		t2.start();
		
	}
}

    这段代码也可以运行成功,但是之前我们同步代码块的时候,synchronized可以用对象作为锁,那么同步函数又是什么作为这个锁呢?答案是this,就是调用这个函数的对象。我们可以写个程序来验证一下。
Demo5:验证同步函数用的是this锁

class TickDemo implements Runnable {
	private int tick = 1000;
	Object obj = new Object();
	boolean flag = true;
	public void run() {
		if(flag) {
			while(true) {
				synchronized(this) {
					if(tick > 0) {
						try {Thread.sleep(10);} catch (InterruptedException e) {}
						System.out.println(Thread.currentThread().getName() + "sale--" + tick--);
					}
				}
			}
		}
		else
			while(true) 
				show();
	}
	public synchronized void show() {
		if(tick > 0) {
			try {Thread.sleep(10);} catch (InterruptedException e) {}
			System.out.println(Thread.currentThread().getName() + "sale--" + tick--);
		}
	}
}
public class Ticket {
	public static void main(String[] args) {
		TickDemo t = new TickDemo();
		
		Thread t1 = new Thread(t);
		Thread t2 = new Thread(t);
		t1.start();
		try {Thread.sleep(10);} catch (InterruptedException e) {}
		t.flag = false;
		t2.start();
		
	}
}

    这段代码思路是:让t1进入同步代码块,让t2进入同步函数,我们把锁设定为this,如果他们同步成功,说明的确用的是this锁。运行结果是成功的,是this锁,如果将代码块的对象改成obj,那么就会出错。
    同步函数如果用static修饰,那么锁就不是this,因为初始化静态函数的时候还没有对象。因此静态同步方法使用的锁是该方法所在类的字节码文件对象,也就是类名.class。如果想验证可以将上面的同步函数用static修饰,然后在同步代码块传入Ticket.class,一样可以运行成功。

	多线程还有可能会出现死锁,比如在同步中嵌套了同步。
Demo6:死锁
class Test implements Runnable {
	private boolean flag;
	Test(boolean flag) {
		this.flag = flag;
	}

	public void run() {
		if(flag) {
			while(true) {
				synchronized(MyLock.locka) {
					System.out.println(Thread.currentThread().getName()+"...if locka ");
					synchronized(MyLock.lockb) {
						System.out.println(Thread.currentThread().getName()+"..if lockb");					
					}
				}
			}
		}
		else {
			while(true) {
				synchronized(MyLock.lockb) {
					System.out.println(Thread.currentThread().getName()+"..else lockb");
					synchronized(MyLock.locka) {
						System.out.println(Thread.currentThread().getName()+".....else locka");
					}
				}
			}
		}
	}
}

class MyLock {
	static Object locka = new Object();
	static Object lockb = new Object();
}

class  DeadLockTest {
	public static void main(String[] args) {
		Thread t1 = new Thread(new Test(true));
		Thread t2 = new Thread(new Test(false));
		t1.start();
		t2.start();
	}
}

     上面的程序有可能会出现这种情况:当t1线程进入if之后,拿到了a锁,还未执行b锁那,此时cpu执行权被t2拿到,t2拿到了b锁,那么t1,t2两个线程就会僵持,程序就死掉了。编程时应避免这种情况的发生。

Dmeo7:匿名内部类创建线程

class ThreadTest {
	public static void main(String[] args) {
		
		new Thread(){
			public void run(){
				for(int x=0; x<100; x++){
					System.out.println(Thread.currentThread().getName()+"....."+x);
				}
			}
		}.start();
		

		Runnable r  = new Runnable(){
			public void run(){
				for(int x=0; x<100; x++){
					System.out.println(Thread.currentThread().getName()+"....."+x);
				}
			}
		};
		new Thread(r).start();

		for(int x=0; x<100; x++)
		{
			System.out.println(Thread.currentThread().getName()+"....."+x);
		}

	}
}




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值