黑马程序员--多线程

多线程


进程

线程(例:FlashGet)

多线程存在的意义。

线程的创建方式

多线程的特性

创建线程方式一

继承Thread类


1. 子类覆盖父类中的run方法,将线程运行的代码存放在run中。
2. 建立子类对象的同时线程也被创建。
3. 通过调用start方法开启线程。


线程的四种状态


sleep方法需要指定睡眠时间,单位是毫秒。一个特殊的状态:就绪。具备了执行资格,但是还没有获取资源。
class Bank
{
private int sum;
//Object obj = new Object();
public synchronized void add(int n)
{
//synchronized(obj)
//{
sum = sum + n;
try{Thread.sleep(10);}catch(Exception e){}
System.out.println("sum="+sum);
//}
}
}


class Cus implements Runnable
{
private Bank b = new Bank();
public void run()
{
for(int x=0; x<3; x++)
{
b.add(100);
}
}
}




class  BankDemo
{
public static void main(String[] args) 
{
Cus c = new Cus();
Thread t1 = new Thread(c);
Thread t2 = new Thread(c);
t1.start();
t2.start();
}
}
创建线程方式二

实现Runnable接口

1. 子类覆盖接口中的run方法。
2. 通过Thread类创建线程,并将实现了Runnable接口的子类对象作为参数传递给Thread类的构造函数。
3. Thread类对象调用start方法开启线程。
class Ticket implements Runnable//extends Thread
{
private  int tick = 100;
public void run()
{
while(true)
{
if(tick>0)
{
System.out.println(Thread.currentThread().getName()+"....sale : "+ tick--);
}
}
}
}




class  TicketDemo
{
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();




/*
Ticket t1 = new Ticket();
//Ticket t2 = new Ticket();
//Ticket t3 = new Ticket();
//Ticket t4 = new Ticket();


t1.start();
t1.start();
t1.start();
t1.start();
*/


}
}

线程安全问题

导致安全问题的出现的原因:

多个线程访问出现延迟。

线程随机性。

注:线程安全问题在理想状态下,不容易出现,但一旦出现对软件的影响是非常大的。


同步(synchronized)

格式:

synchronized(对象)

{

需要同步的代码;

}


同步可以解决安全问题的根本原因就在那个对象上。

该对象如同锁的功能。

同步的前提:

1.同步需要两个或者两个以上的线程。2.多个线程使用的是同一个锁。

未满足这两个条件,不能称其为同步。

同步的弊端:

当线程相当多时,因为每个线程都会去判断同步上的锁,这是很耗费资源的,无形中会降低程序的运行效率。

同步函数

格式:在函数上加上synchronized修饰符即可。




思考1:wait(),notify(),notifyAll(),用来操作线程为什么定义在了Object类中?


1,这些方法存在与同步中。


2,使用这些方法时必须要标识所属的同步的锁。


3,锁可以是任意对象,所以任意对象调用的方法一定定义Object类中。


思考2:wait(),sleep()有什么区别?


wait():释放cpu执行权,释放锁。


sleep():释放cpu执行权,不释放锁。

停止线程


1. 定义循环结束标记

因为线程运行代码一般都是循环,只要控制了循环即可。


2. 使用interrupt(中断)方法。

该方法是结束线程的冻结状态,使线程回到运行状态中来。


注:stop方法已经过时不再使用。
class Ticket implements Runnable
{
private  int tick = 100;
Object obj = new Object();
boolean flag = true;
public  void run()
{
if(flag)
{
while(true)
{
synchronized(this)
{
if(tick>0)
{
try{Thread.sleep(10);}catch(Exception e){}
System.out.println(Thread.currentThread().getName()+"....code : "+ tick--);
}
}
}
}
else
while(true)
show();
}
public synchronized void show()//this
{
if(tick>0)
{
try{Thread.sleep(10);}catch(Exception e){}
System.out.println(Thread.currentThread().getName()+"....show.... : "+ tick--);
}
}
}




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


Ticket t = new Ticket();


Thread t1 = new Thread(t);
Thread t2 = new Thread(t);
t1.start();
try{Thread.sleep(10);}catch(Exception e){}
t.flag = false;
t2.start();














// Thread t3 = new Thread(t);
// Thread t4 = new Thread(t);
// t3.start();
// t4.start();




}

线程类的其他方法


setPriority(int num)

setDaemon(boolean b)

join()

自定义线程名称

toString()
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值