解决线程不安全的方式(Java)
一、同步代码块
package com.synchronized1; // 买票示例 // 使用同步代码块解决线程安全问题 public class TicketRunnableImp implements Runnable { private int ticket = 100; Object o=new Object(); @Override public void run() { while (true) { synchronized (o){ if (ticket > 0) { System.out.println(Thread.currentThread().getName() + "-->正在售第"+ticket+"张票!"); ticket--; } } } } }
实现原理:使用了一个锁对象.这个锁对象也叫同步锁,对象锁,对象监视器。先进入同步代码块的线程会先获取到锁对象,其他线程不能进入代码块,只有先进入同步代码块的线程
执行完毕同步代码块释放锁对象后,其他线程才有机会获取到锁对象进入同步代码块。也就是每次只有一个线程进入到同步代码块执行逻辑。
优点:实现了线程安全
缺点:程序不停的判断锁,获取锁,释放锁,效率会降低
二、同步方法
package com.synchronized2; // 买票示例 // 使用同步方法解决线程安全问题 public class TicketRunnableImp implements Runnable { private int ticket = 100; Object o = new Object(); @Override public void run() { while (true) { func(); } } // 同步方法的锁对象是调用者对象(Runnable对象) public synchronized void func(){ if (ticket > 0) { System.out.println(Thread.currentThread().getName() + "-->正在售第" + ticket + "张票!"); ticket--; } } }
实现原理:和同步代码块实现原理基本一致,同步方法中的锁对象是线程实现类的对象。
三、静态方法
package com.staticSyn; // 买票示例 // 使用同步方法解决线程安全问题 public class TicketRunnableImp implements Runnable { private static int ticket = 100; Object o = new Object(); @Override public void run() { while (true) { func(); } } // 静态方法的锁对象是本类的class属性 public synchronized static void func(){ if (ticket > 0) { System.out.println(Thread.currentThread().getName() + "-->正在售第" + ticket + "张票!"); ticket--; } } }
四、Lock锁
方式一
package com.lock1; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; // 买票示例 // 使用同步代码块解决线程安全问题 public class TicketRunnableImp implements Runnable { private int ticket = 100; Object o = new Object(); Lock lock=new ReentrantLock(); @Override public void run() { while (true) { lock.lock(); if (ticket > 0) { System.out.println(Thread.currentThread().getName() + "-->正在售第" + ticket + "张票!"); ticket--; } lock.unlock(); } } }
方式二
package com.lock2; import java.util.concurrent.locks.Lock; import java.util.concurrent.locks.ReentrantLock; // 买票示例 // 使用同步代码块解决线程安全问题 public class TicketRunnableImp implements Runnable { private int ticket = 100; Object o = new Object(); Lock lock = new ReentrantLock(); @Override public void run() { while (true) { lock.lock(); try { if (ticket > 0) { System.out.println(Thread.currentThread().getName() + "-->正在售第" + ticket + "张票!"); ticket--; } } catch (Exception e) { System.out.println(e); } finally { lock.unlock(); } } } }
五、上述几种方式的测试类
package com.lock2; public class DemoTicket { public static void main(String[] args) { TicketRunnableImp t=new TicketRunnableImp(); Thread t1=new Thread(t); Thread t2=new Thread(t); Thread t3=new Thread(t); t1.start(); t2.start(); t3.start(); } }
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· AI 智能体引爆开源社区「GitHub 热点速览」
· 从HTTP原因短语缺失研究HTTP/2和HTTP/3的设计差异
· 三行代码完成国际化适配,妙~啊~