简单的卖票程序
多个窗口卖票
创建线程的第二种方式:实现Runnable接口
步骤:
1,定义类实现Runnable 接口
2,覆盖Runnable接口中的run方法。将线程要运行的代码存放在该run方法中
3,通过Thread类建立线程对象
4,将Runnable接口的子类对象作为实际参数传递给Thread类的构造函数
为什么要将Runnable接口的子类对象传递给Thread的构造函数
因为自定义的run方法所属的对象是Runnable接口的子类对象,
所以要让线程去指定对象的run方法。就必须明确该run方法所属的对象。
5,调用Thread类的start方法开启线程并调用Runnable接口的run方法。
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();
}
}
实现方式和继承方式的区别:
实现好处:避免了单继承的局限性,在定义线程时,建议使用实现方式。
继承Thread:线程代码存放在Thread子类run方法中,
实现Runnable:线程代码存放在接口的子类run方法中
class Ticket implements Runnable//extends Thread
{
private int tick=100;
public void run()
{
while(true)
{
if(tick>0)
{
try
{
Thread.sleep(10);
}
catch(Exception e)
{
}
System.out.println(Thread.currentThread().getName()+"sale:"+tick--);
}
}
}
}
通过分析发现,打印出0,-1,-2,等错票。
多线程的运行出现安全问题 .
问题的原因:
当多条语句在操作同一个 线程共享数据时,一个线程对多条语句只执行了一部分,还没执行完,
另一个线程参与执行,导致共享数据的错误。
解决办法:对多条操作共享数据的语句,只能让一个线程都执行完,在执行过程中,
其他线程不可以参与执行
Java对于多线程的安全问题提供了专业的解决方式。
就是同步代码块。
synchronized(对象)
{
需要同步的代码;
}
对象如同锁,持有锁的线程可以在同步中执行,
没有持有锁的线程即使获取CPU的执行权,也进不去,因为没有获取锁。
同步的前提:
1,必须要有两个及以上的线程。
2,必须是多个线程使用同一个锁。
必须保证同步中只能有一个线程在执行。
好处:解决了多线程的安全问题
弊端:多个线程都需要判断锁,较为消耗资源,
class Ticket implements Runnable//extends Thread
{
private int tick=100;
Object obj = new Object;
public void run()
{
while(true)
{
synchronized(obj)
{
if(tick>0)
{
try{Thread.sleep(10);}catch(Exception e){}
System.out.println(Thread.currentThread().getName()+"sale:"+tick--);
}
}
}
}
}
*/
/*
需求:银行有一个金库,有两个储户分别存300元,每次存100,存3次。
目的:该程序是否有安全问题,若果有,怎么解决?
如何找问题:
1,哪些代码是多线程运行代码
2,明确共享数据
3,明确多线程运行代码中哪些语句是操作共享数据的
*/
class Bank
{
private int sum;
Object obj = new Object();
public void add(int n)
{
synchronized(obj)
{
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 [] ars)
{
Cus c = new Cus();
Thread t1 = new Thread(c);
Thread t1 = new Thread(c);
t1.start();
t2.start();
}
}
------
同步函数
class Bank
{
private int sum;
public synchronized void add(int n)
}
sum+=n;
try
{
Thread.sleep(10);
}
catch (Exception e)
{
}
System.out.println("sum="+sum);
}
}
同步函数:用的是哪一个锁呢?
函数需要被对象调用,那么函数都有一个所属对象引用,就是this。
所以同步函数使用的锁是this。
使用两个线程来卖票
一个线程在同步代码块中,另一个线程在同步函数中,都在执行卖票动作。
class Ticket implements Runnable
{
private int tick=100;
Object obj = new Object;
boolean flag=true;
public void run()
{
if(flag)
{
while(true)
{
synchronized(obj)
{
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()
{
if(tick>0)
{
try{Thread.sleep(10);}catch(Exception e){}
System.out.println(Thread.currentThread().getName()+"---show---"+tick--);
}
}
}
class TicketDemo
{
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();
}
}
如果同步函数被静态修饰后,使用的锁是什么呢?
通过验证,发现不再是this ,因为静态中不可以定义this。
静态进内存时,内存中没有本类对象,但是一定有该类对应的字节码文件对象。
类名.class ,该对象的类型是Class
静态的同步方法使用的锁是该方法所在类的字节码文件对象,类名.class
class Ticket implements Runnable
{
private int tick=100;
//Object obj = new Object;
boolean flag=true;
public void run()
{
if(flag)
{
while(true)
{
synchronized(Ticket.class)
{
if(tick>0)
{
try{Thread.sleep(10);}catch(Exception e){}
System.out.println(Thread.currentThread().getName()+"---code"+tick--);
}
}
}
else
while(true)
show();
}
public static synchronized void show()
{
if(tick>0)
{
try{Thread.sleep(10);}catch(Exception e){}
System.out.println(Thread.currentThread().getName()+"---show---"+tick--);
}
}
}
class TicketDemo
{
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();
}
}
/*单例设计模式*/
/*
饿汉式。
class Single
{
private static final Single s = new Single();
private Single(){}
public static Single getInstance()
{
return s;
}
}
*/
/*
懒汉式
class Single
{
private static Single s = null;
private Single(){}
public static synchronized Single getInstance()
{
if(s==null)
s = new Single();
return s;
}
}
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 Ticket implements Runnable
{
private int tick=100;
Object obj = new Object;
boolean flag=true;
public void run()
{
if(flag)
{
while(true)
{
synchronized(obj)
{
show();
}
}
else
while(true)
show();
}
public static synchronized void show()//this锁
{
synchronized(obj)
if(tick>0)
{
try{Thread.sleep(10);}catch(Exception e){}
System.out.println(Thread.currentThread().getName()+"---show---"+tick--);
}
}
}
class TicketDemo
{
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();
}
} |
|