您的位置:首页 > 编程语言 > Java开发

Java【多线程知识总结(7)】多线程同步问题-关于synchronized代码块和synchronized方法的应用

2011-10-07 16:15 761 查看
下面的程序会逐步引出synchronized的用法:

[例]

class TicketMan
{
 public static void main(String[]aresg)
 {
  Ticket t=new Ticket();
  //创建4个线程
  new Thread(t).start();
  new Thread(t).start();
  new Thread(t).start();
  new Thread(t).start();
 }
}

class Ticket implements Runnable
{
 int tk=10;
 public void run()
 {
  while(tk>0)
  {
   System.out.println(Thread.currentThread().getName()+"窗口正在售出"+tk--+"号票.");
  }
 }
}
//输出结果:4个窗口,成功把10张票随机分配售完.
/*
Thread-0窗口正在售出10号票.
Thread-1窗口正在售出7号票.
Thread-1窗口正在售出5号票.
Thread-3窗口正在售出8号票.
Thread-2窗口正在售出9号票.
Thread-3窗口正在售出3号票.
Thread-3窗口正在售出1号票.
Thread-1窗口正在售出4号票.
Thread-0窗口正在售出6号票.
Thread-2窗口正在售出2号票.
*/

结论:

使用Runnable接口创建多线程,适合多个相同的程序代码的线程去处理分享同一个资源的情况,把虚拟CPU(线程)同程序的代码数据有效分离,较好体现了面向对象的设计思想.

【多线程安全问题考虑】

程序像上面那样写并不安全,输出的结果有可能输出“售出负号票”,理由:假设Thread-0线程执行到while语句里时,这时恰巧tk=1;突然停了.切换到Thread-1线程,也是运行到while语句里突然切换到了Thread-2线程,这样while语句里有三个线程有待执行,tk=1,等这三个线程执行完毕tk就变成负数了.下面我们用sleep()来模拟这个假设:

class TicketMan
{
 public static void main(String[]aresg)
 {
  Ticket t=new Ticket();
  //创建4个线程
  new Thread(t).start();
  new Thread(t).start();
  new Thread(t).start();
  new Thread(t).start();
 }
}

class Ticket implements Runnable
{
 int tk=10;
 public void run()
 {
  while(tk>0)
  {
   try
   {
     Thread.sleep(1);//当前线程暂停1毫秒,处理器会去执行其他线程. 
   }
   catch (Exception ex)
   {
     ex.printStackTrace();//输出造成异常更为详细的信息.
   }
   System.out.println(Thread.currentThread().getName()+"窗口正在售出"+tk--+"号票.");
  }
 }
}
//输出结果:
/*
Thread-0窗口正在售出9号票.
Thread-2窗口正在售出7号票.
Thread-3窗口正在售出8号票.
Thread-1窗口正在售出10号票.
Thread-2窗口正在售出5号票.
Thread-0窗口正在售出6号票.
Thread-2窗口正在售出2号票.
Thread-1窗口正在售出3号票.
Thread-3窗口正在售出4号票.
Thread-2窗口正在售出0号票.
Thread-0窗口正在售出1号票.
*/

多次运行TicketMan类调试,结果都不一样,有时出现0,有时不出现,模拟的程序已经达到要求了.那么针对这个卖票程序,到底怎样写才安全呢?这就是即将要引出的线程同步问题:synchronized关键字,代表这个方法枷锁,一次只能允许一个线程通过.比如线程A执行到synchronized代码快或方法体时,首先的工作就是检测有没有其他线程在执行该语句,有的话,就等其他线程出来后,再进去.

synchronized代码块的用法-synchronized(Object){}举例如下:



class TicketMan
{
 public static void main(String[]aresg)
 {
  Ticket t=new Ticket();
  //创建4个线程
  new Thread(t).start();
  new Thread(t).start();
  new Thread(t).start();
  new Thread(t).start();
 }
}

class Ticket implements Runnable
{
 int tk=10;
 String s=new String("");
 public void run()
 {
  //run()方法体里千万别重置s对象,不然会放进很多线程进来,那就失去本来意义了,不妨把String s=new String("")移到run方法里试试看;.
  while(tk>0)
  {

  //synchronized代码块开始
  synchronized(s)
  {
     //注意: synchronized()方法体里千万别重置s对象.
     if(tk>0)
     {
        try
        {
          Thread.sleep(50);//当前线程暂停1毫秒,处理器会去执行其他线程. 
        }
        catch (Exception ex)
        {
          ex.printStackTrace();//输出造成异常更为详细的信息.
        }

        System.out.println(Thread.currentThread().getName()+"窗口正在售出"+tk--+"号票.");
     }
   }
  //synchronized代码块结束

  }
 }
}

synchronized方法的用法需要与run()方法两个结合起来用:



/*
语法:
public synchronized void duMuQiao()
{}

public void run()
{
  .....
  调用synchronized修改过的方法:duMuQiao();
  .....
}
举例如下:
*/
class TicketMan
{
 public static void main(String[]aresg)
 {
  Ticket t=new Ticket();
  //创建4个线程
  new Thread(t).start();
  new Thread(t).start();
  new Thread(t).start();
  new Thread(t).start();
 }
}

class Ticket implements Runnable
{
 int tk=10;

 public void run()
 {
  while(tk>0)
  {
   duMuQiao();//调用synchronized修改过的方法:duMuQiao();
  }
 }

  //synchronized方法开始
  public synchronized void duMuQiao()
  //一个线程进入这个方法后 这个方法的大门就会暂时关闭(不许其他线程进入)直到这个线程走出这个方法后,该方法的大门才会敞开.
  {

     if(tk>0)
     {
        try
        {
          Thread.sleep(50);//当前线程暂停1毫秒,处理器会去执行其他线程. 
        }
        catch (Exception ex)
        {
          ex.printStackTrace();//输出造成异常更为详细的信息.
        }

        System.out.println(Thread.currentThread().getName()+"窗口正在售出"+tk--+"号票.");
     }
   }
  //synchronized方法结束
}




内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐