国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

高并發(fā)場景下,6種解決SimpleDateFormat類的線程安全問題方法

這篇具有很好參考價(jià)值的文章主要介紹了高并發(fā)場景下,6種解決SimpleDateFormat類的線程安全問題方法。希望對(duì)大家有所幫助。如果存在錯(cuò)誤或未考慮完全的地方,請(qǐng)大家不吝賜教,您也可以點(diǎn)擊"舉報(bào)違法"按鈕提交疑問。

摘要:解決SimpleDateFormat類在高并發(fā)場景下的線程安全問題可以有多種方式,這里,就列舉幾個(gè)常用的方式供參考。

本文分享自華為云社區(qū)《【高并發(fā)】更正SimpleDateFormat類線程不安全問題分析的錯(cuò)誤》,作者: 冰 河 。

解決SimpleDateFormat類在高并發(fā)場景下的線程安全問題可以有多種方式,這里,就列舉幾個(gè)常用的方式供參考,大家也可以在評(píng)論區(qū)給出更多的解決方案。

1.局部變量法

最簡單的一種方式就是將SimpleDateFormat類對(duì)象定義成局部變量,如下所示的代碼,將SimpleDateFormat類對(duì)象定義在parse(String)方法的上面,即可解決問題。

package io.binghe.concurrent.lab06;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
/**
 * @author binghe
 * @version 1.0.0
 * @description 局部變量法解決SimpleDateFormat類的線程安全問題
 */
public class SimpleDateFormatTest02 {
 //執(zhí)行總次數(shù)
 private static final int EXECUTE_COUNT = 1000;
 //同時(shí)運(yùn)行的線程數(shù)量
 private static final int THREAD_COUNT = 20;
 public static void main(String[] args) throws InterruptedException {
 final Semaphore semaphore = new Semaphore(THREAD_COUNT);
 final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT);
 ExecutorService executorService = Executors.newCachedThreadPool();
 for (int i = 0; i < EXECUTE_COUNT; i++){
 executorService.execute(() -> {
 try {
 semaphore.acquire();
 try {
 SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
 simpleDateFormat.parse("2020-01-01");
 } catch (ParseException e) {
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }catch (NumberFormatException e){
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }
 semaphore.release();
 } catch (InterruptedException e) {
 System.out.println("信號(hào)量發(fā)生錯(cuò)誤");
 e.printStackTrace();
 System.exit(1);
 }
 countDownLatch.countDown();
 });
 }
 countDownLatch.await();
 executorService.shutdown();
 System.out.println("所有線程格式化日期成功");
 }
}

此時(shí)運(yùn)行修改后的程序,輸出結(jié)果如下所示。

所有線程格式化日期成功

至于在高并發(fā)場景下使用局部變量為何能解決線程的安全問題,會(huì)在【JVM專題】的JVM內(nèi)存模式相關(guān)內(nèi)容中深入剖析,這里不做過多的介紹了。

當(dāng)然,這種方式在高并發(fā)下會(huì)創(chuàng)建大量的SimpleDateFormat類對(duì)象,影響程序的性能,所以,這種方式在實(shí)際生產(chǎn)環(huán)境不太被推薦。

2.synchronized鎖方式

將SimpleDateFormat類對(duì)象定義成全局靜態(tài)變量,此時(shí)所有線程共享SimpleDateFormat類對(duì)象,此時(shí)在調(diào)用格式化時(shí)間的方法時(shí),對(duì)SimpleDateFormat對(duì)象進(jìn)行同步即可,代碼如下所示。

package io.binghe.concurrent.lab06;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
/**
 * @author binghe
 * @version 1.0.0
 * @description 通過Synchronized鎖解決SimpleDateFormat類的線程安全問題
 */
public class SimpleDateFormatTest03 {
 //執(zhí)行總次數(shù)
 private static final int EXECUTE_COUNT = 1000;
 //同時(shí)運(yùn)行的線程數(shù)量
 private static final int THREAD_COUNT = 20;
 //SimpleDateFormat對(duì)象
 private static SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
 public static void main(String[] args) throws InterruptedException {
 final Semaphore semaphore = new Semaphore(THREAD_COUNT);
 final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT);
 ExecutorService executorService = Executors.newCachedThreadPool();
 for (int i = 0; i < EXECUTE_COUNT; i++){
 executorService.execute(() -> {
 try {
 semaphore.acquire();
 try {
 synchronized (simpleDateFormat){
 simpleDateFormat.parse("2020-01-01");
 }
 } catch (ParseException e) {
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }catch (NumberFormatException e){
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }
 semaphore.release();
 } catch (InterruptedException e) {
 System.out.println("信號(hào)量發(fā)生錯(cuò)誤");
 e.printStackTrace();
 System.exit(1);
 }
 countDownLatch.countDown();
 });
 }
 countDownLatch.await();
 executorService.shutdown();
 System.out.println("所有線程格式化日期成功");
 }
}

此時(shí),解決問題的關(guān)鍵代碼如下所示。

synchronized (simpleDateFormat){
simpleDateFormat.parse("2020-01-01");
}

運(yùn)行程序,輸出結(jié)果如下所示。

所有線程格式化日期成功

需要注意的是,雖然這種方式能夠解決SimpleDateFormat類的線程安全問題,但是由于在程序的執(zhí)行過程中,為SimpleDateFormat類對(duì)象加上了synchronized鎖,導(dǎo)致同一時(shí)刻只能有一個(gè)線程執(zhí)行parse(String)方法。此時(shí),會(huì)影響程序的執(zhí)行性能,在要求高并發(fā)的生產(chǎn)環(huán)境下,此種方式也是不太推薦使用的。

3.Lock鎖方式

Lock鎖方式與synchronized鎖方式實(shí)現(xiàn)原理相同,都是在高并發(fā)下通過JVM的鎖機(jī)制來保證程序的線程安全。通過Lock鎖方式解決問題的代碼如下所示。

package io.binghe.concurrent.lab06;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
/**
 * @author binghe
 * @version 1.0.0
 * @description 通過Lock鎖解決SimpleDateFormat類的線程安全問題
 */
public class SimpleDateFormatTest04 {
 //執(zhí)行總次數(shù)
 private static final int EXECUTE_COUNT = 1000;
 //同時(shí)運(yùn)行的線程數(shù)量
 private static final int THREAD_COUNT = 20;
 //SimpleDateFormat對(duì)象
 private static SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
 //Lock對(duì)象
 private static Lock lock = new ReentrantLock();
 public static void main(String[] args) throws InterruptedException {
 final Semaphore semaphore = new Semaphore(THREAD_COUNT);
 final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT);
 ExecutorService executorService = Executors.newCachedThreadPool();
 for (int i = 0; i < EXECUTE_COUNT; i++){
 executorService.execute(() -> {
 try {
 semaphore.acquire();
 try {
 lock.lock();
 simpleDateFormat.parse("2020-01-01");
 } catch (ParseException e) {
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }catch (NumberFormatException e){
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }finally {
 lock.unlock();
 }
 semaphore.release();
 } catch (InterruptedException e) {
 System.out.println("信號(hào)量發(fā)生錯(cuò)誤");
 e.printStackTrace();
 System.exit(1);
 }
 countDownLatch.countDown();
 });
 }
 countDownLatch.await();
 executorService.shutdown();
 System.out.println("所有線程格式化日期成功");
 }
}

通過代碼可以得知,首先,定義了一個(gè)Lock類型的全局靜態(tài)變量作為加鎖和釋放鎖的句柄。然后在simpleDateFormat.parse(String)代碼之前通過lock.lock()加鎖。這里需要注意的一點(diǎn)是:為防止程序拋出異常而導(dǎo)致鎖不能被釋放,一定要將釋放鎖的操作放到finally代碼塊中,如下所示。

finally {
lock.unlock();
}

運(yùn)行程序,輸出結(jié)果如下所示。

所有線程格式化日期成功

此種方式同樣會(huì)影響高并發(fā)場景下的性能,不太建議在高并發(fā)的生產(chǎn)環(huán)境使用。

4.ThreadLocal方式

使用ThreadLocal存儲(chǔ)每個(gè)線程擁有的SimpleDateFormat對(duì)象的副本,能夠有效的避免多線程造成的線程安全問題,使用ThreadLocal解決線程安全問題的代碼如下所示。

package io.binghe.concurrent.lab06;
import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
/**
 * @author binghe
 * @version 1.0.0
 * @description 通過ThreadLocal解決SimpleDateFormat類的線程安全問題
 */
public class SimpleDateFormatTest05 {
 //執(zhí)行總次數(shù)
 private static final int EXECUTE_COUNT = 1000;
 //同時(shí)運(yùn)行的線程數(shù)量
 private static final int THREAD_COUNT = 20;
 private static ThreadLocal<DateFormat> threadLocal = new ThreadLocal<DateFormat>(){
 @Override
 protected DateFormat initialValue() {
 return new SimpleDateFormat("yyyy-MM-dd");
 }
 };
 public static void main(String[] args) throws InterruptedException {
 final Semaphore semaphore = new Semaphore(THREAD_COUNT);
 final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT);
 ExecutorService executorService = Executors.newCachedThreadPool();
 for (int i = 0; i < EXECUTE_COUNT; i++){
 executorService.execute(() -> {
 try {
 semaphore.acquire();
 try {
 threadLocal.get().parse("2020-01-01");
 } catch (ParseException e) {
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }catch (NumberFormatException e){
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }
 semaphore.release();
 } catch (InterruptedException e) {
 System.out.println("信號(hào)量發(fā)生錯(cuò)誤");
 e.printStackTrace();
 System.exit(1);
 }
 countDownLatch.countDown();
 });
 }
 countDownLatch.await();
 executorService.shutdown();
 System.out.println("所有線程格式化日期成功");
 }
}

通過代碼可以得知,將每個(gè)線程使用的SimpleDateFormat副本保存在ThreadLocal中,各個(gè)線程在使用時(shí)互不干擾,從而解決了線程安全問題。

運(yùn)行程序,輸出結(jié)果如下所示。

所有線程格式化日期成功

此種方式運(yùn)行效率比較高,推薦在高并發(fā)業(yè)務(wù)場景的生產(chǎn)環(huán)境使用。

另外,使用ThreadLocal也可以寫成如下形式的代碼,效果是一樣的。

package io.binghe.concurrent.lab06;
import java.text.DateFormat;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
/**
 * @author binghe
 * @version 1.0.0
 * @description 通過ThreadLocal解決SimpleDateFormat類的線程安全問題
 */
public class SimpleDateFormatTest06 {
 //執(zhí)行總次數(shù)
 private static final int EXECUTE_COUNT = 1000;
 //同時(shí)運(yùn)行的線程數(shù)量
 private static final int THREAD_COUNT = 20;
 private static ThreadLocal<DateFormat> threadLocal = new ThreadLocal<DateFormat>();
 private static DateFormat getDateFormat(){
 DateFormat dateFormat = threadLocal.get();
 if(dateFormat == null){
 dateFormat = new SimpleDateFormat("yyyy-MM-dd");
 threadLocal.set(dateFormat);
 }
 return dateFormat;
 }
 public static void main(String[] args) throws InterruptedException {
 final Semaphore semaphore = new Semaphore(THREAD_COUNT);
 final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT);
 ExecutorService executorService = Executors.newCachedThreadPool();
 for (int i = 0; i < EXECUTE_COUNT; i++){
 executorService.execute(() -> {
 try {
 semaphore.acquire();
 try {
 getDateFormat().parse("2020-01-01");
 } catch (ParseException e) {
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }catch (NumberFormatException e){
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }
 semaphore.release();
 } catch (InterruptedException e) {
 System.out.println("信號(hào)量發(fā)生錯(cuò)誤");
 e.printStackTrace();
 System.exit(1);
 }
 countDownLatch.countDown();
 });
 }
 countDownLatch.await();
 executorService.shutdown();
 System.out.println("所有線程格式化日期成功");
 }
}

5.DateTimeFormatter方式

DateTimeFormatter是Java8提供的新的日期時(shí)間API中的類,DateTimeFormatter類是線程安全的,可以在高并發(fā)場景下直接使用DateTimeFormatter類來處理日期的格式化操作。代碼如下所示。

package io.binghe.concurrent.lab06;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
/**
 * @author binghe
 * @version 1.0.0
 * @description 通過DateTimeFormatter類解決線程安全問題
 */
public class SimpleDateFormatTest07 {
 //執(zhí)行總次數(shù)
 private static final int EXECUTE_COUNT = 1000;
 //同時(shí)運(yùn)行的線程數(shù)量
 private static final int THREAD_COUNT = 20;
 private static DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd");
 public static void main(String[] args) throws InterruptedException {
 final Semaphore semaphore = new Semaphore(THREAD_COUNT);
 final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT);
 ExecutorService executorService = Executors.newCachedThreadPool();
 for (int i = 0; i < EXECUTE_COUNT; i++){
 executorService.execute(() -> {
 try {
 semaphore.acquire();
 try {
 LocalDate.parse("2020-01-01", formatter);
 }catch (Exception e){
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }
 semaphore.release();
 } catch (InterruptedException e) {
 System.out.println("信號(hào)量發(fā)生錯(cuò)誤");
 e.printStackTrace();
 System.exit(1);
 }
 countDownLatch.countDown();
 });
 }
 countDownLatch.await();
 executorService.shutdown();
 System.out.println("所有線程格式化日期成功");
 }
}

可以看到,DateTimeFormatter類是線程安全的,可以在高并發(fā)場景下直接使用DateTimeFormatter類來處理日期的格式化操作。

運(yùn)行程序,輸出結(jié)果如下所示。

所有線程格式化日期成功

使用DateTimeFormatter類來處理日期的格式化操作運(yùn)行效率比較高,推薦在高并發(fā)業(yè)務(wù)場景的生產(chǎn)環(huán)境使用。

6.joda-time方式

joda-time是第三方處理日期時(shí)間格式化的類庫,是線程安全的。如果使用joda-time來處理日期和時(shí)間的格式化,則需要引入第三方類庫。這里,以Maven為例,如下所示引入joda-time庫。

<dependency>
<groupId>joda-time</groupId>
<artifactId>joda-time</artifactId>
<version>2.9.9</version>
</dependency>

引入joda-time庫后,實(shí)現(xiàn)的程序代碼如下所示。

package io.binghe.concurrent.lab06;
import org.joda.time.DateTime;
import org.joda.time.format.DateTimeFormat;
import org.joda.time.format.DateTimeFormatter;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;
/**
 * @author binghe
 * @version 1.0.0
 * @description 通過DateTimeFormatter類解決線程安全問題
 */
public class SimpleDateFormatTest08 {
 //執(zhí)行總次數(shù)
 private static final int EXECUTE_COUNT = 1000;
 //同時(shí)運(yùn)行的線程數(shù)量
 private static final int THREAD_COUNT = 20;
 private static DateTimeFormatter dateTimeFormatter = DateTimeFormat.forPattern("yyyy-MM-dd");
 public static void main(String[] args) throws InterruptedException {
 final Semaphore semaphore = new Semaphore(THREAD_COUNT);
 final CountDownLatch countDownLatch = new CountDownLatch(EXECUTE_COUNT);
 ExecutorService executorService = Executors.newCachedThreadPool();
 for (int i = 0; i < EXECUTE_COUNT; i++){
 executorService.execute(() -> {
 try {
 semaphore.acquire();
 try {
 DateTime.parse("2020-01-01", dateTimeFormatter).toDate();
 }catch (Exception e){
 System.out.println("線程:" + Thread.currentThread().getName() + " 格式化日期失敗");
 e.printStackTrace();
 System.exit(1);
 }
 semaphore.release();
 } catch (InterruptedException e) {
 System.out.println("信號(hào)量發(fā)生錯(cuò)誤");
 e.printStackTrace();
 System.exit(1);
 }
 countDownLatch.countDown();
 });
 }
 countDownLatch.await();
 executorService.shutdown();
 System.out.println("所有線程格式化日期成功");
 }
}

這里,需要注意的是:DateTime類是org.joda.time包下的類,DateTimeFormat類和DateTimeFormatter類都是org.joda.time.format包下的類,如下所示。

import org.joda.time.DateTime;
import org.joda.time.format.DateTimeFormat;
import org.joda.time.format.DateTimeFormatter;

運(yùn)行程序,輸出結(jié)果如下所示。

所有線程格式化日期成功

使用joda-time庫來處理日期的格式化操作運(yùn)行效率比較高,推薦在高并發(fā)業(yè)務(wù)場景的生產(chǎn)環(huán)境使用。

解決SimpleDateFormat類的線程安全問題的方案總結(jié)

綜上所示:在解決解決SimpleDateFormat類的線程安全問題的幾種方案中,局部變量法由于線程每次執(zhí)行格式化時(shí)間時(shí),都會(huì)創(chuàng)建SimpleDateFormat類的對(duì)象,這會(huì)導(dǎo)致創(chuàng)建大量的SimpleDateFormat對(duì)象,浪費(fèi)運(yùn)行空間和消耗服務(wù)器的性能,因?yàn)镴VM創(chuàng)建和銷毀對(duì)象是要耗費(fèi)性能的。所以,不推薦在高并發(fā)要求的生產(chǎn)環(huán)境使用。

synchronized鎖方式和Lock鎖方式在處理問題的本質(zhì)上是一致的,通過加鎖的方式,使同一時(shí)刻只能有一個(gè)線程執(zhí)行格式化日期和時(shí)間的操作。這種方式雖然減少了SimpleDateFormat對(duì)象的創(chuàng)建,但是由于同步鎖的存在,導(dǎo)致性能下降,所以,不推薦在高并發(fā)要求的生產(chǎn)環(huán)境使用。

ThreadLocal通過保存各個(gè)線程的SimpleDateFormat類對(duì)象的副本,使每個(gè)線程在運(yùn)行時(shí),各自使用自身綁定的SimpleDateFormat對(duì)象,互不干擾,執(zhí)行性能比較高,推薦在高并發(fā)的生產(chǎn)環(huán)境使用。

DateTimeFormatter是Java 8中提供的處理日期和時(shí)間的類,DateTimeFormatter類本身就是線程安全的,經(jīng)壓測,DateTimeFormatter類處理日期和時(shí)間的性能效果還不錯(cuò)(后文單獨(dú)寫一篇關(guān)于高并發(fā)下性能壓測的文章)。所以,推薦在高并發(fā)場景下的生產(chǎn)環(huán)境使用。

joda-time是第三方處理日期和時(shí)間的類庫,線程安全,性能經(jīng)過高并發(fā)的考驗(yàn),推薦在高并發(fā)場景下的生產(chǎn)環(huán)境使用。

?

點(diǎn)擊關(guān)注,第一時(shí)間了解華為云新鮮技術(shù)~文章來源地址http://www.zghlxwxcb.cn/news/detail-510937.html

到了這里,關(guān)于高并發(fā)場景下,6種解決SimpleDateFormat類的線程安全問題方法的文章就介紹完了。如果您還想了解更多內(nèi)容,請(qǐng)?jiān)谟疑辖撬阉鱐OY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點(diǎn)僅代表作者本人,不代表本站立場。本站僅提供信息存儲(chǔ)空間服務(wù),不擁有所有權(quán),不承擔(dān)相關(guān)法律責(zé)任。如若轉(zhuǎn)載,請(qǐng)注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實(shí)不符,請(qǐng)點(diǎn)擊違法舉報(bào)進(jìn)行投訴反饋,一經(jīng)查實(shí),立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費(fèi)用

相關(guān)文章

  • 【JavaEE】并發(fā)編程(多線程)線程安全問題&內(nèi)存可見性&指令重排序

    【JavaEE】并發(fā)編程(多線程)線程安全問題&內(nèi)存可見性&指令重排序

    目錄 第一個(gè)問題:什么是線程安全問題? 第二個(gè)問題:為什么會(huì)出現(xiàn)線程安全問題?? 第三個(gè)問題:如何解決多線程安全問題?? 第四個(gè)問題:產(chǎn)生線程不安全的原因有哪些?? 第五個(gè)問題:內(nèi)存可見性問題及解決方案? 第六個(gè)問題:指令重排序問題? 線程安全就是多線程

    2024年02月01日
    瀏覽(25)
  • 【Java|多線程與高并發(fā)】線程安全問題以及synchronized使用實(shí)例

    【Java|多線程與高并發(fā)】線程安全問題以及synchronized使用實(shí)例

    Java多線程環(huán)境下,多個(gè)線程同時(shí)訪問共享資源時(shí)可能出現(xiàn)的數(shù)據(jù)競爭和不一致的情況。 線程安全一直都是一個(gè)令人頭疼的問題.為了解決這個(gè)問題,Java為我們提供了很多方式. synchronized、ReentrantLock類等。 使用線程安全的數(shù)據(jù)結(jié)構(gòu),例如ConcurrentHashMap、ConcurrentLinkedQueue等

    2024年02月09日
    瀏覽(14)
  • Java面試題:SimpleDateFormat是線程安全的嗎?使用時(shí)應(yīng)該注意什么?

    在Java開發(fā)中,我們經(jīng)常需要獲取和處理時(shí)間,這需要使用到各種不同的方法。其中,使用SimpleDateFormat類來格式化時(shí)間是一種常見的方法。雖然這個(gè)類看上去功能比較簡單,但是如果使用不當(dāng),也可能會(huì)引發(fā)一些問題。 首先我們要明確一點(diǎn), SimpleDateFormat不是線程安全的 。

    2024年04月26日
    瀏覽(42)
  • 線程中并發(fā)安全問題(Sychronized關(guān)鍵字的底層原理)

    線程中并發(fā)安全問題(Sychronized關(guān)鍵字的底層原理)

    Sychronized的底層原理 ? sychronized 對(duì)象鎖采用互斥方式讓同一時(shí)刻至多只有一個(gè)線程能持有對(duì)象鎖,其他線程想獲取這個(gè)對(duì)象鎖只能被阻塞。 Monitor Sychronized的底層實(shí)現(xiàn)Monitor。 WaitSet:關(guān)聯(lián)調(diào)用了wait方法的線程,用于存儲(chǔ)處于等待狀態(tài)的線程。 EntryList:關(guān)聯(lián)了沒有獲得

    2024年02月16日
    瀏覽(22)
  • FunASR語音識(shí)別(解決-高并發(fā)線程問題)

    FunASR語音識(shí)別(解決-高并發(fā)線程問題)

    在我的另一個(gè)博客有介紹FunASR,并且進(jìn)行了語者分離,不過最近FunASR自帶了語者分離,挺好挺好,但是一直看社區(qū)就是大家都用python寫,會(huì)出現(xiàn)線程不安全問題,群里有大佬說使用多臺(tái)服務(wù)器,然后用nginx做代理,這不是妥妥土豪行為嗎,感覺很浪費(fèi) vad出現(xiàn)的問題 方案解決:

    2024年01月16日
    瀏覽(76)
  • (線程池)多線程使用場景--es數(shù)據(jù)批量導(dǎo)入、數(shù)據(jù)匯總、異步調(diào)用;如何控制某個(gè)方法允許并發(fā)訪問線程的數(shù)量;對(duì)ThreadLocal的理解及實(shí)現(xiàn)原理、源碼解析、ThreadLocal的內(nèi)存泄露問題

    (線程池)多線程使用場景--es數(shù)據(jù)批量導(dǎo)入、數(shù)據(jù)匯總、異步調(diào)用;如何控制某個(gè)方法允許并發(fā)訪問線程的數(shù)量;對(duì)ThreadLocal的理解及實(shí)現(xiàn)原理、源碼解析、ThreadLocal的內(nèi)存泄露問題

    CountDownLatch(閉鎖/倒計(jì)時(shí)鎖) 用來進(jìn)行線程同步協(xié)作,等待所有線程完成倒計(jì)時(shí)(一個(gè)或者多個(gè)線程,等待其他多個(gè)線程完成某件事情之后才能執(zhí)行) 其中構(gòu)造參數(shù)用來初始化等待計(jì)數(shù)值 await() 用來等待計(jì)數(shù)歸零 countDown() 用來讓計(jì)數(shù) 減一 多線程使用場景一:( es數(shù)據(jù)批量導(dǎo)

    2024年04月25日
    瀏覽(47)
  • 超賣等高并發(fā)秒殺場景的問題及解決方案

    超賣等高并發(fā)秒殺場景的問題及解決方案

    多線程并行運(yùn)行 多行代碼操作共享資源,但不具備原子性 例: 針對(duì)并發(fā)安全問題,最廣為人知的解決方案就是 加鎖 。 從實(shí)現(xiàn)思想上來說,鎖可以分為兩大類: 悲觀鎖 樂觀鎖 悲觀鎖是一種獨(dú)占和排他的鎖機(jī)制,保守地認(rèn)為數(shù)據(jù)會(huì)被其他事務(wù)修改,所以在整個(gè)數(shù)據(jù)處理過程

    2024年02月15日
    瀏覽(21)
  • 多線程進(jìn)階學(xué)習(xí)(高并發(fā)、線程池、使用場景)

    多線程進(jìn)階學(xué)習(xí)(高并發(fā)、線程池、使用場景)

    線程和進(jìn)程的區(qū)別? 程序由 指令 和 數(shù)據(jù) 組成,但這些指令要運(yùn)行,數(shù)據(jù)要讀寫,就必須將指令加載到CPU,數(shù)據(jù)加載到內(nèi)存。 在指令運(yùn)行過程中還需要用到磁盤,網(wǎng)絡(luò)等設(shè)備。 進(jìn)程就是用來加載指令、管理內(nèi)存、管理IO的 。 簡單來說, 當(dāng)一個(gè)程序被運(yùn)行,從磁盤加載這個(gè)

    2024年02月12日
    瀏覽(16)
  • 解決在使用 Elasticsearch(ES)多線程批量操作時(shí)導(dǎo)致并發(fā)一致性的問題?。? decoding=

    解決在使用 Elasticsearch(ES)多線程批量操作時(shí)導(dǎo)致并發(fā)一致性的問題??!

    先說一下什么是數(shù)據(jù)庫數(shù)據(jù)庫中 并發(fā)一致性 問題! 1、在并發(fā)環(huán)境下,事務(wù)的隔離性很難保證,因此會(huì)出現(xiàn)很多并發(fā)一致性問題。 數(shù)據(jù)丟失 T1 和 T2 兩個(gè)事務(wù)都對(duì)一個(gè)數(shù)據(jù)進(jìn)行修改,T1 先修改,T2 隨后修改,T2 的修改覆蓋了 T1 的修改。 讀臟數(shù)據(jù) T1 修改一個(gè)數(shù)據(jù),T2 隨后讀取

    2024年02月04日
    瀏覽(26)
  • Java多線程 - 線程安全和線程同步解決線程安全問題

    Java多線程 - 線程安全和線程同步解決線程安全問題

    線程安全問題指的是: 多個(gè)線程同時(shí)操作同一個(gè)共享資源的時(shí)候可能會(huì)出現(xiàn)業(yè)務(wù)安全問題,稱為線程安全問題。 舉例: 取錢模型演示 需求:小明和小紅是一對(duì)夫妻,他們有一個(gè)共同的賬戶,余額是10萬元。 如果小明和小紅同時(shí)來取錢,而且2人都要取錢10萬元,可能出現(xiàn)什么問

    2023年04月15日
    瀏覽(24)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請(qǐng)作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包