線程安全問題是一個比較高深的問題,是很多程序員比較難掌握的一個技術難點,如果一個程序員對線程掌握的很好的話,那麼這個程序員的內功修煉的是相當的好。
在這裡我主要說一下我對Java中如何保證線程安全的一些個人見解,希望對各位有所幫助,那裡有不對的地方敬請給位不吝賜教。
線程安全問題主要出現在訪問臨界資源的時候,就是訪問同一個對象的時候,可能會出現無法挽回的損失,特別是在關於資金安全方面的時候,當然還有數據庫事務方面的問題。他們很類似,都是要保證數據的原子性。
那麼在Java中如何保證線程安全呢?
對與共同使用的對象進行加鎖,意思是我使用的時候,那麼你就必須等待,等我用完之後你再用,反之依然。就像上廁所,你去的時候我是不能去的。
如何加鎖呢?下面寫三個加鎖的方式
首先看一下實例代碼
- 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、對代碼塊加鎖
對共同持有的對象加鎖可以把內部類寫成這樣的
- 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
- 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、對靜態方法加鎖的對象到底是誰?
- 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
如果和非靜態方法同時持有同一個對象時,可以持有同一個字節碼對象。