JAVA线程安全的问题

线程安全问题是一个比较高深的问题,是很多程序员比较难掌握的一个技术难点,如果一个程序员对线程掌握的很好的话,那么这个程序员的内功修炼的是相当的好。

在这里我主要说一下我对java中如何保证线程安全的一些个人见解,希望对各位有所帮助,那里有不对的地方敬请给位不吝赐教。

线程安全问题主要出现在访问临界资源的时候,就是访问同一个对象的时候,可能会出现无法挽回的损失,特别是在关于资金安全方面的时候,当然还有数据库事务方面的问题。他们很类似,都是要保证数据的原子性。

那么在java中如何保证线程安全呢?

对与共同使用的对象进行加锁,意思是我使用的时候,那么你就必须等待,等我用完之后你再用,反之依然。就像上厕所,你去的时候我是不能去的。

如何加锁呢?下面写三个加锁的方式

首先看一下实例代码

Html代码 复制代码
  1. public class TraditionalSynchornizedTest {   
  2.  /**   
  3.   * @param args   
  4.   */   
  5.  public static void main(String[] args) {   
  6.   new TraditonalSynchornizedTest().sartThread();   
  7.  }   
  8.  public void sartThread(){   
  9.   final Outerput outerput = new Outerput();   
  10.   new Thread(new Runnable(){   
  11.    @Override   
  12.    public void run() {   
  13.     while(true){   
  14.      try {   
  15.       Thread.sleep(5);   
  16.      } catch (InterruptedException e) {   
  17.       e.printStackTrace();   
  18.      }   
  19.      outerput.print("zhangsanfeng");   
  20.     }   
  21.    }   
  22.       
  23.   }).start();   
  24.   new Thread(new Runnable(){   
  25.    @Override   
  26.    public void run() {   
  27.     while(true){   
  28.      try {   
  29.       Thread.sleep(5);   
  30.      } catch (InterruptedException e) {   
  31.       e.printStackTrace();   
  32.      }   
  33.      outerput.print("luxiaofeng");   
  34.     }   
  35.    }   
  36.       
  37.   }).start();   
  38.  }   
  39.  public class Outerput{   
  40.   public void print(String name){   
  41.    for(int i = 0;i < name.length(); i++){   
  42.     System.out.print(name.charAt(i));   
  43.    }   
  44.    System.out.println();   
  45.   }   
  46.  }   
  47. }  

public class TraditionalSynchornizedTest { /** * @param args */ public static void main(String[] args) { new TraditonalSynchornizedTest().sartThread(); } public void sartThread(){ final Outerput outerput = new Outerput(); new Thread(new Runnable(){ @Override public void run() { while(true){ try { Thread.sleep(5); } catch (InterruptedException e) { e.printStackTrace(); } outerput.print("zhangsanfeng"); } } }).start(); new Thread(new Runnable(){ @Override public void run() { while(true){ try { Thread.sleep(5); } catch (InterruptedException e) { e.printStackTrace(); } outerput.print("luxiaofeng"); } } }).start(); } public class Outerput{ public void print(String name){ for(int i = 0;i < name.length(); i++){ System.out.print(name.charAt(i)); } System.out.println(); } } }

 

以上代码没有对共同持有的对象outerput加锁,所以会出现线程安全问题

1,对代码块加锁

对共同持有的对象加锁可以把内部类写成这样的

Java代码 复制代码
  1. public class Outerput{   
  2.   public void print(String name){   
  3.    synchronized (this) {   
  4.     for(int i = 0;i < name.length(); i++){   
  5.      System.out.print(name.charAt(i));   
  6.     }   
  7.     System.out.println();   
  8.    }   
  9.   }   
  10.  }  

public class Outerput{ public void print(String name){ synchronized (this) { for(int i = 0;i < name.length(); i++){ System.out.print(name.charAt(i)); } System.out.println(); } } }

 

2,对非静态方法加锁,加锁的对象是this

Java代码 复制代码
  1. public class Outerput{   
  2.   public synchronized void print(String name){   
  3.    for(int i = 0;i < name.length(); i++){   
  4.     System.out.print(name.charAt(i));   
  5.    }   
  6.    System.out.println();   
  7.   }   
  8.  }   

public class Outerput{ public synchronized void print(String name){ for(int i = 0;i < name.length(); i++){ System.out.print(name.charAt(i)); } System.out.println(); } }

 

3,对静态方法加锁的对象到底是谁?

Java代码 复制代码
  1. public static synchronized  void print2(String name){   
  2.    for(int i = 0;i < name.length(); i++){   
  3.     System.out.print(name.charAt(i));   
  4.    }   
  5.    System.out.println();   
  6.   }  

public static synchronized void print2(String name){ for(int i = 0;i < name.length(); i++){ System.out.print(name.charAt(i)); } System.out.println(); }

 

其实加锁的对象是字节码对象,Outerput.class

如果和非静态方法同时持有同一个对象时,可以持有同一个字节码对象。

 

原文链接:http://flystar007.iteye.com/blog/1404430

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值