深入理解 Java 多線程、Lambda 表達式及線程安全最佳實踐

来源:https://www.cnblogs.com/xiaowange/p/18076372
-Advertisement-
Play Games

Java 線程 線程使程式能夠通過同時執行多個任務而更有效地運行。 線程可用於在不中斷主程式的情況下在後臺執行複雜的任務。 創建線程 有兩種創建線程的方式。 擴展Thread類 可以通過擴展Thread類並覆蓋其run()方法來創建線程: public class MyThread extends ...


Java 線程

線程使程式能夠通過同時執行多個任務而更有效地運行。

線程可用於在不中斷主程式的情況下在後臺執行複雜的任務。

創建線程

有兩種創建線程的方式。

  1. 擴展Thread類

可以通過擴展Thread類並覆蓋其run()方法來創建線程:

public class MyThread extends Thread {
  public void run() {
    System.out.println("This code is running in a thread");
  }
}
  1. 實現Runnable介面

另一種創建線程的方式是實現Runnable介面:

public class MyRunnable implements Runnable {
  public void run() {
    System.out.println("This code is running in a thread");
  }
}

運行線程

  1. 擴展Thread類

如果類擴展Thread類,則可以通過創建類的實例並調用其start()方法來運行線程:

public class Main {
  public static void main(String[] args) {
    MyThread myThread = new MyThread();
    myThread.start();
    System.out.println("This code is outside of the thread");
  }
}
  1. 實現Runnable介面

如果類實現了Runnable介面,則可以通過將類的實例傳遞給Thread對象的構造函數,然後調用線程的start()方法來運行線程:

public class Main {
  public static void main(String[] args) {
    MyRunnable myRunnable = new MyRunnable();
    Thread thread = new Thread(myRunnable);
    thread.start();
    System.out.println("This code is outside of the thread");
  }
}

區分“擴展”和“實現”線程

主要區別在於,當一個類擴展Thread類時,無法擴展任何其他類,但通過實現Runnable介面,可以擴展另一個類,例如:

class MyClass extends OtherClass implements Runnable

併發問題

因為線程與程式的其他部分同時運行,所以無法知道代碼將按照什麼順序運行。當線程和主程式同時讀取和寫入相同的變數時,值是不可預測的。由此導致的問題稱為併發問題。

示例

一個變數amount值不可預測的代碼示例:

public class Main extends Thread {
  public static int amount = 0;

  public static void main(String[] args) {
    Main thread = new Main();
    thread.start();
    System.out.println(amount);
    amount++;
    System.out.println(amount);
  }

  public void run() {
    amount++;
  }
}

為避免併發問題,最好儘可能少地線上程之間共用屬性。如果需要共用屬性,一種可能的解決方案是在使用線程可以更改的任何屬性之前,使用線程的isAlive()方法檢查線程是否已完成運行。

示例

使用isAlive()防止併發問題:

public class Main extends Thread {
  public static int amount = 0;

  public static void main(String[] args) {
    Main thread = new Main();
    thread.start();
    // 等待線程完成
    while (thread.isAlive()) {
      System.out.println("Waiting...");
    }
    // 更新amount並列印其值
    System.out.println("Main program: " + amount);
    amount++;
    System.out.println("Main program: " + amount);
  }

  public void run() {
    amount++;
  }
}

線程池

線程池是一種管理線程的資源。它允許您創建並維護一組可重用的線程。使用線程池可以提高應用程式的性能和效率。

線程安全

線程安全是指多個線程可以訪問和修改數據而不導致數據損壞。使數據線程安全的一種方法是使用同步。同步是一種機制,它允許線程一次一個地訪問共用數據。

常見的線程安全問題

  • 競態條件:當多個線程同時訪問共用數據並嘗試對其進行更改時,就會發生競態條件。這可能導致數據損壞。
  • 原子性:原子操作是指不可分割的操作。當多個線程嘗試同時執行原子操作時,可能會導致數據損壞。
  • 可見性:當一個線程對共用數據進行更改時,其他線程必須能夠看到這些更改。

避免線程安全問題

  • 使用同步
  • 使用不可變對象
  • 使用原子操作

Java Lambda表達式

Lambda表達式簡介

Lambda表達式是在Java 8中引入的。Lambda表達式是一小段代碼塊,它接受參數並返回一個值。Lambda表達式類似於方法,但它們不需要名稱,並且可以直接在方法體中實現。

Lambda表達式的語法

最簡單的Lambda表達式包含一個參數和一個表達式:

參數 -> 表達式

要使用多個參數,請將它們放在括弧中:

(參數1, 參數2) -> 表達式

表達式是有限制的。它們必須立即返回一個值,並且不能包含變數、賦值或if或for等語句。為了執行更複雜的操作,可以使用帶有花括弧的代碼塊。如果Lambda表達式需要返回一個值,那麼代碼塊應該有一個return語句。

(參數1, 參數2) -> { 代碼塊 }

Lambda表達式的使用

Lambda表達式通常作為參數傳遞給函數。在以下示例中,Lambda表達式作為參數傳遞給ArrayList的forEach()方法,以列印列表中的每個項:

import java.util.ArrayList;

public class Main {
  public static void main(String[] args) {
    ArrayList<Integer> numbers = new ArrayList<>();
    numbers.add(5);
    numbers.add(9);
    numbers.add(8

);
    numbers.add(1);
    numbers.forEach((n) -> { System.out.println(n); });
  }
}

Lambda表達式的存儲

如果變數的類型是僅具有一個方法的介面,那麼Lambda表達式可以存儲在變數中。Lambda表達式應該具有與該方法相同數量的參數和相同的返回類型。Java內置了許多這類介面,如Consumer介面(在java.util包中),它被列表使用。

import java.util.ArrayList;
import java.util.function.Consumer;

public class Main {
  public static void main(String[] args) {
    ArrayList<Integer> numbers = new ArrayList<>();
    numbers.add(5);
    numbers.add(9);
    numbers.add(8);
    numbers.add(1);
    Consumer<Integer> method = (n) -> { System.out.println(n); };
    numbers.forEach(method);
  }
}

Lambda表達式作為方法參數

要在方法中使用Lambda表達式,該方法應該有一個以單一方法介面作為其類型的參數。調用介面的方法將運行Lambda表達式。

interface StringFunction {
  String run(String str);
}

public class Main {
  public static void main(String[] args) {
    StringFunction exclaim = (s) -> s + "!";
    StringFunction ask = (s) -> s + "?";
    printFormatted("Hello", exclaim);
    printFormatted("Hello", ask);
  }

  public static void printFormatted(String str, StringFunction format) {
    String result = format.run(str);
    System.out.println(result);
  }
}

Lambda表達式的優勢

  • 簡化代碼
  • 提高可讀性
  • 增強代碼的表達力

Lambda 表達式是 Java 8 中引入的一項強大功能,可以簡化代碼並提高可讀性。它們是函數式編程的重要組成部分,可以用於各種任務,例如數據處理、事件處理和流處理。

最後

為了方便其他設備和平臺的小伙伴觀看往期文章:

微信公眾號搜索:Let us Coding,關註後即可獲取最新文章推送

看完如果覺得有幫助,歡迎 點贊、收藏、關註


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 近期看到了一個前阿裡資深開發的學術分析視頻: 高併發情況下,一個事務內有更新操作還有查詢操作,那是先更新好,還是先無鎖查詢好? 僅70秒的視頻,深感學問太深,但是海哥講的有待補充,於是寫下了這篇文章,作為補充。 鳴謝:前阿裡資深開發極海Channel的技術分享。 先說答案 這是個開放性的問題,必須看 ...
  • 前言 開發人員寫的SQL語句中經常會用到in,exists,not in,not exists 這類子查詢,通常,含in、exists的子查詢稱為半連接(semijoin),含not in、 not exists的子查詢被稱之為反連接,經常會有技術人員來評論in 與exists 效率孰高孰低的問題, ...
  • Android 輔助功能 -搶紅包(三) 本篇文章繼續講述輔助功能. 主要通過監聽通知欄紅包消息,來跳轉聊天頁面,並自動回覆對方"謝謝". 上篇文章我們講述了監聽notification, 跳轉聊天界面. 具體可查看: Android 輔助功能 -搶紅包(二) 1: 使用monitor抓取id. 打 ...
  • Android 輔助功能 -搶紅包 本篇文章主要介紹下通過android輔助功能的方式來實現類似搶紅包的功能. 1:許可權聲明 <uses-permission android:name="android.permission.BIND_ACCESSIBILITY_SERVICE" tools:ign ...
  • Android 輔助功能 -搶紅包(二) 本篇文章繼續講述輔助功能實現搶紅包的方案. 上篇文章主要講了下輔助功能的基本使用,本文涉及的一些基礎內容就不再贅述了. 有疑問的可以查看上篇文章: Android 輔助功能 -搶紅包 1: 添加微信監聽 修改xml文件,android:packageName ...
  • 前言 如果趕時間請直接使用目錄跳到解決問題的部分。 使用的項目使用vue腳手架生成。 npm init vue@latest 版本如下 "@vitejs/plugin-vue": "^5.0.4", "vue": "^3.4.21" 由於近期在學less,心想如果不能將其應用到vue項目中,無異於紙 ...
  • 這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 一、介紹 Promise,譯為承諾,是非同步編程的一種解決方案,比傳統的解決方案(回調函數)更加合理和更加強大 在以往我們如果處理多層非同步操作,我們往往會像下麵那樣編寫我們的代碼 doSomething(function(result) { ...
  • C語言中抽象函數與具體實現的命名與組織 在C語言的項目開發中,尤其是嵌入式系統和開源軟體項目里,合理地命名和組織抽象函數及其具體實現對於提高代碼的可讀性、可維護性和可擴展性至關重要。以下是關於如何在這些項目中有效地處理抽象和實現的一些建議: 抽象函數與具體實現的區分 API作為介面:API定義了一組 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...