黑马程序员----JAVASE之多线程

--------------  android培训 java培训 、期待与您交流! ---------------

1.进程与线程

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

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

     Java VM  启动的时候会有一个进程java.exe:

   该进程中至少一个线程负责java程序的执行。而且这个线程运行的代码存在于main方法中。该线程称之为主线程。

扩展:其实更细节说明jvm,jvm启动不止一个线程,还有负责垃圾回收机制的线程。


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

     通过对api的查找,java已经提供了对线程这类事物的描述。就Thread类。

创建线程的第一种方式:继承Thread类。

 步骤:
     1,定义类继承Thread。
     2,复写Thread类中的run方法。
    目的:将自定义代码存储在run方法。让线程运行。
     3,调用线程的start方法,该方法两个作用:启动线程,调用run方法。

      发现运行结果每一次都不同。因为多个线程都获取cpu的执行权。cpu执行到谁,谁就运行。明确一点,在某一个时刻,只能有一个程序在运行。(多核除外)cpu在做着快速的切换,以达到看上去是同时运行的效果。我们可以形象把多线程的运行行为在互相抢夺cpu的执行权。这就是多线程的一个特性:随机性。谁抢到谁执行,至于执行多长,cpu说的算。


② 为什么要覆盖run方法呢?
     Thread类用于描述线程。该类就定义了一个功能,用于存储线程要运行的代码。该存储功能就是run方法。也就是说Thread类中的run方法,用于存储线程要运行的代码。

package cn.itheima.thread;

class Demo extends Thread{
	@Override
	public void run() {
		for(int x=0; x<60; x++)
			System.out.println("demo run----"+x);
	}
}

public class TreadTest {
	
	public static void main(String[] args) {
		Demo demo = new Demo();
		demo.start();
		
		for(int x=0; x<60; x++)
			System.out.println("Hello World!--"+x);
	}
	
}

创建线程的第二种方式:实现Runable接口

步骤:
1,定义类实现Runnable接口
2,覆盖Runnable接口中的run方法,将线程要运行的代码存放在该run方法中。
3,通过Thread类建立线程对象。
4,将Runnable接口的子类对象作为实际参数传递给Thread类的构造函数。
为什么要将Runnable接口的子类对象传递给Thread的构造函数。
因为自定义的run方法所属的对象是Runnable接口的子类对象。所以要让线程去指定指定对象的run方法。就必须明确该run方法所属对象。
5,调用Thread类的start方法开启线程并调用Runnable接口子类的run方法。

package cn.itheima.thread;

class Ticket implements Runnable{

	private int tick = 100;
	public void run() {
		while (true) {
			if(tick>0){
				System.out.println(Thread.currentThread().getName() + " 买第 " + tick--);
			}
			
		}
	}
	
}

public class TicketDemo {
	public static void main(String[] args) {
		Ticket ticket = new Ticket();
		
		Thread thread1 = new Thread(ticket);
		Thread thread2 = new Thread(ticket);
		Thread thread3 = new Thread(ticket);
		Thread thread4 = new Thread(ticket);
		
		thread1.start();
		thread2.start();
		thread3.start();
		thread4.start();
	}
	
	
}
线程的安全性问题

学了了多线程基本知识过后,我们需要考虑一个问题,多个线程在运行过程中,如果同时访问同一个资源,必回带来安全问题:

问题的原因:
当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一部分,还没有执行完,
另一个线程参与进来执行。导致共享数据的错误。

解决办法:
对多条操作共享数据的语句,只能让一个线程都执行完。在执行过程中,其他线程不可以参与执行。

Java对于多线程的安全问题提供了专业的解决方式:

    就是同步代码块。
      synchronized(对象)
     {
     需要被同步的代码
     }

对象如同锁。持有锁的线程可以在同步中执行。没有持有锁的线程即使获取cpu的执行权,也进不去,因为没有获取锁。

同步的前提:
    1,必须要有两个或者两个以上的线程。
    2,必须是多个线程使用同一个锁。
    3.   必须保证同步中只能有一个线程在运行。
好处:解决了多线程的安全问题。
弊端:多个线程需要判断锁,较为消耗资源,

那么买票程序肯定存在安全性问题,那么我们重新解决这个问题:

class Ticket implements Runnable
{
	private  int tick = 1000;
	Object obj = new Object();
	public void run()
	{
		while(true)
		{
			synchronized(obj)
			{
				if(tick>0)
				{
					System.out.println(Thread.currentThread().getName()+"....sale : "+ tick--);
				}
			}
		}
	}
}

class  TicketDemo2
{
	public static void main(String[] args) 
	{

		Ticket t = new Ticket();

		Thread t1 = new Thread(t);
		Thread t2 = new Thread(t);
		Thread t3 = new Thread(t);
		Thread t4 = new Thread(t);
		t1.start();
		t2.start();
		t3.start();
		t4.start();
	}
}


单例设计模式(懒汉式)解决安全行问题:

class Single
{
	private static Single s = null;
	private Single(){}


	public static  Single getInstance()
	{
		if(s==null)
		{
			synchronized(Single.class)
			{
				if(s==null)
					s = new Single();
			}
		}
		return s;
	}
}

class SingleDemo 
{
	public static void main(String[] args) 
	{
		System.out.println("Hello World!");
	}
}

线程wait()和sleep()的区别:

都是使线程暂时停止执行的方法,但它们有很大的不同,sleep是线程类Thread 的方法,它是使当前线程暂时睡眠,可以放在任何位置。而wait是Object类的方法,它是使当前线程暂时放弃对象的使用权进行等待,必须放在同步方法或同步块里。Sleep使用的时候,线程并不会放弃对象的使用权,即不会释放对象锁,所以在同步方法或同步块中使用sleep,一个线程访问时,其他的线程也是无法访问的。而wait是会释放对象锁的,就是当前线程放弃对象的使用权,让其他的线程可以访问。线程执行wait方法时,需要另一个线程调用notify进行唤醒。而sleep只是暂时休眠一定时间,时间到了之后,自动恢复运行,不需另外的线程唤醒。

总结:
wait()方法:释放CPU执行权,释放锁,必须在同步代码块或同步方法中,是Object类的方法

sleep()方法:释放CPU执行权,不释放锁,可以放在任意地方,是Thread类的方法

class DemoObject extends Thread
{
	DemoObject(){}
	DemoObject(String name)
	{
		super(name);  //Thread中有构造函数 Thread(String name),所以这里直接拿过来用
	}
	public void run()
	{
		for (int x=0; x<100; x++)
			System.out.println(this.getName()+"-"+x);
		/*
		 * 下面的sleep语句是为了测试sleep是否会释放CPU执行权的
		 * 实验结果证明,sleep语句是会释放CPU执行权
		 */
		try{Thread.sleep(5000);System.out.println(this.getName()+"sleepthread");}catch(InterruptedException in){}
		/*
		 * this.getName()==Thread.currentThread().getName(),
		 * 注意:this的对象为继承Thread的情况下可以使用this关键字,
		 * 否则只能使用Thread.currentThread().getName(),通过Runnable接口
		 * 实现多线程方式中就需通过这种方式
		 */
	}
}
class ThreadDemo
{
	public static void main(String[] args)
	{
		DemoObject d = new DemoObject("第1个自定义线程");
		DemoObject d2 = new DemoObject("第2个自定义线程"); //创建第二个线程
		d.start(); //线程1启动并执行run()
		d2.start(); //线程2启动并执行run()
		for (int x=0; x<100; x++)
			System.out.println("ThreadDemo Run---"+x);
	}
}

                                                              -------------- android培训java培训、期待与您交流! ---------------

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值