Java中线程安全问题个人理解

简介:

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

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

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

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

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

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

  首先看一下实例代码

  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. }

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

1、对代码块加锁

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

  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.  }

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

  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.  }

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

  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.   }

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

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



本文出自seven的测试人生公众号最新内容请见作者的GitHub页:http://qaseven.github.io/

目录
相关文章
|
2月前
|
安全 算法 Java
Java 多线程:线程安全与同步控制的深度解析
本文介绍了 Java 多线程开发的关键技术,涵盖线程的创建与启动、线程安全问题及其解决方案,包括 synchronized 关键字、原子类和线程间通信机制。通过示例代码讲解了多线程编程中的常见问题与优化方法,帮助开发者提升程序性能与稳定性。
117 0
|
2月前
|
Java API 调度
从阻塞到畅通:Java虚拟线程开启并发新纪元
从阻塞到畅通:Java虚拟线程开启并发新纪元
275 83
|
3月前
|
存储 SQL 安全
Java 无锁方式实现高性能线程实战操作指南
本文深入探讨了现代高并发Java应用中单例模式的实现方式,分析了传统单例(如DCL)的局限性,并提出了多种无锁实现方案。包括基于ThreadLocal的延迟初始化、VarHandle原子操作、Record不可变对象、响应式编程(Reactor)以及CDI依赖注入等实现方式。每种方案均附有代码示例及适用场景,同时通过JMH性能测试对比各实现的优劣。最后,结合实际案例设计了一个高性能配置中心,展示了无锁单例在实际开发中的应用。总结中提出根据场景选择合适的实现方式,并遵循现代单例设计原则以优化性能和安全性。文中还提供了代码获取链接,便于读者实践与学习。
88 0
|
2月前
|
存储 Java 调度
Java虚拟线程:轻量级并发的革命性突破
Java虚拟线程:轻量级并发的革命性突破
221 83
|
4月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
180 0
|
3月前
|
存储 Java
说一说 JAVA 内存模型与线程
我是小假 期待与你的下一次相遇 ~
|
3月前
|
移动开发 Java
说一说 Java 是如何实现线程间通信
我是小假 期待与你的下一次相遇 ~
|
3月前
|
Java 数据挖掘 调度
Java 多线程创建零基础入门新手指南:从零开始全面学习多线程创建方法
本文从零基础角度出发,深入浅出地讲解Java多线程的创建方式。内容涵盖继承`Thread`类、实现`Runnable`接口、使用`Callable`和`Future`接口以及线程池的创建与管理等核心知识点。通过代码示例与应用场景分析,帮助读者理解每种方式的特点及适用场景,理论结合实践,轻松掌握Java多线程编程 essentials。
214 5

热门文章

最新文章