併發concurrent---1

来源:https://www.cnblogs.com/taojietaoge/archive/2019/01/18/10285607.html
-Advertisement-
Play Games

背景:併發知識是一個程式員段位升級的體現,同樣也是進入BAT的必經之路,有必要把併發知識重新梳理一遍。 說到併發concurrent,肯定首先想到了線程,創建線程有兩種方法:1、從Java.lang.Thread類派生一個新的線程類,重載它的run()方法;2、實現Runnalbe介面,重載Runn ...


背景:併發知識是一個程式員段位升級的體現,同樣也是進入BAT的必經之路,有必要把併發知識重新梳理一遍。

 

併發concurrent:

說到併發concurrent,肯定首先想到了線程,創建線程有兩種方法:1、從Java.lang.Thread類派生一個新的線程類,重載它的run()方法;2、實現Runnalbe介面,重載Runnalbe介面中的run()方法;建議使用方法二創建線程,因為,如果是通過擴展 Thread類的方法來創建線程,那麼這個自定義類就不能再去擴展其他的類,也就無法實現更加複雜的功能;而實現Runnable介面的方法來定義該類為線程類,這樣就可以避免Java單繼承所帶來的局限性,也更符合面向對象編程的思想,最重要的就是使用實現Runnable介面的方式創建的線程可以處理同一資源,從而實現資源的共用。

創建線程的兩種方法:

 1 package www.concurent.test;
 2 public class TraditionalThread {
 3 
 4     public static void main(String[] args) {
 5         //Thread1:
 6         Thread thread = new Thread() {
 7             @Override
 8             public void run() {
 9                 while(true) {
10                     try {
11                         Thread.sleep(1000);
12                     } catch (InterruptedException e) {
13                         e.printStackTrace();
14                     }
15                     System.out.println("thread1: "+Thread.currentThread().getName());
16                 }
17             }
18         }; 
19         thread.start();
20         
21         //Thread2:
22         //Runnable變數是線程要運行的代碼的宿主,更適合面向對象思想的線程方法
23         Thread thread2 = new Thread(new Runnable() {
24             @Override
25             public void run() {
26                 while(true) {
27                     try {
28                         Thread.sleep(1000);
29                     } catch (InterruptedException e) {
30                         e.printStackTrace();
31                     }
32                     System.out.println("thread2: "+Thread.currentThread().getName());
33                 }
34             }
35         });
36         thread2.start();
37     }
38 }

線程和Timer定時器很類似,下麵介紹了兩種和線程相似的定時器寫法:1、定時一天之後調用方法查詢天氣情況介面,然後每隔60秒後繼續調用該方法;2、定時每天00:39:32調用查詢天氣情況介面,通過Hutool工具和Timer定時器調用HTTP天氣狀況介面的返回結果如下截圖:(result2得到了天津的天氣狀況)

通過Hutool工具和Timer定時器調用HTTP天氣狀況介面

 1 import java.util.Calendar;
 2 import java.util.Date;
 3 import java.util.Timer;
 4 import java.util.TimerTask;
 5 import cn.hutool.http.HttpUtil;
 6 
 7 public class TraditionalTimerTest {
 8     //時間間隔
 9      private static final long PERIOD_DAY = 24 * 60 * 60 * 1000;
10     //Timer 定時器
11     public static void main(String[] args) {
12         Calendar cl = Calendar.getInstance();
13         cl.set(Calendar.HOUR_OF_DAY, 0);
14         cl.set(Calendar.MINUTE, 39);
15         cl.set(Calendar.SECOND, 32);
16         Date date = cl.getTime();
17         Date dateNow = new Date();
18         //如果第一次執行定時任務的時間 小於 當前的時間
19         //此時要在 第一次執行定時任務的時間 加一天,以便此任務在下個時間點執行。如果不加一天,任務會立即執行。
20         if (date.before(dateNow)) {
21             Calendar clAdd = Calendar.getInstance();
22             clAdd.setTime(dateNow);
23             clAdd.add(Calendar.DAY_OF_MONTH, 1);
24              date = clAdd.getTime();
25         }
26         //Timer1:
27         new Timer().schedule(new TimerTask() {
28             @Override
29             public void run() {
30                 System.out.println("hello");
31                 //Hutool調用http介面
32                 String result1 = HttpUtil.get("http://t.weather.sojson.com/api/weather/city/101030100");
33                 System.out.println("result1: "+result1);
34             }
35             //一天之後調用方法查詢天氣情況介面,然後每隔60秒後繼續調用該方法
36         },PERIOD_DAY , 1000*60);
37         //Timer2:
38         new Timer().schedule(new TimerTask() {
39             @Override
40             public void run() {
41                 //Hutool調用http介面
42                 String result2 = HttpUtil.get("http://t.weather.sojson.com/api/weather/city/101030100");
43                 System.out.println("result2: " + result2);
44             }
45             //定時每天00:39:32調用查詢天氣情況介面
46         }, date , PERIOD_DAY);
47     }
48 
49 }

 如果是單個線程調用都還ok,要是有多個線程同時調用那就會出現併發產生;比如有一個方法Output()是經過charAt(i)獲取字元串i的字元並且列印再控制台,然後線程A和線程B同時調用Output()方法,此時就會出現線程不安全問題(如銀行取錢和轉賬同時進行),也就是併發;執行結果發現,線程A為執行完畢線程B就開始執行了,為了能後保證當有一個線程來執行某個方法時,其他的線程不能進來執行該方法,實現排他性,可以通過synchronized和ReentrantLock來實現線程同步;二者其實區別不大,synchronized由於是底層JVM實現的互斥,因此效率會高一些,而ReentrantLock的功能則比synchronized更多,比如定時獲取某個鎖,多個等待條件等,另外synchronized 會讓線程阻塞,ReentrantLock會讓線程等待,但是從行為效果上來看是一樣的;下麵有個例子:併發結果如截圖顯示,理想狀態是列印“huawei”或者“isoftstone”,但是由於併發列印出來諸如此類“ishuaweoftstoni”結果。

FYI:

 1 import java.util.concurrent.locks.Lock;
 2 import java.util.concurrent.locks.ReentrantLock;
 3 
 4 public class MyThreadSynchronized {
 5     public static void main(String[] args) {
 6         //要想調用內部類的對象,必須有外部類的實例對象
 7         new MyThreadSynchronized().init();   // 外部類的實例對象
 8     }
 9     public void init() {
10         final Outputer outputer = new Outputer();
11         //thread1:
12         new Thread(new Runnable() {
13             @Override
14             public void run() {
15                 while(!false) {
16                     try {
17                         Thread.sleep(100);
18                     } catch (InterruptedException e) {
19                         e.printStackTrace();
20                     }
21                     outputer.output3("huawei");
22                 }
23             }
24         }).start();
25         
26         //thread2:
27         new Thread(new Runnable() {
28             @Override
29             public void run() {
30                 while(!false) {
31                     try {
32                         Thread.sleep(100);
33                     } catch (InterruptedException e) {
34                         e.printStackTrace();
35                     }
36                     outputer.output("isoftstone");
37                 }
38             }
39         }).start();
40     }
41     //當有一個線程來執行某個方法時,其他的線程不能進來執行該方法,排他性、獨一無二;
42     //使用synchronized/lock同步,且線程用的同步鎖是同一個同步對象,可用this互斥或方法.class
43     //synchronized由於是底層JVM實現的互斥,因此效率會高一些
44     //ReentrantLock的功能則比synchronized更多,比如定時獲取某個鎖,多個等待條件
45     //synchronized 會讓線程阻塞,ReentrantLock會讓線程等待,但是從行為效果上來看是一樣的;
46 class Outputer{
47     //內部類  靜態方法中不能new內部類的實例對象
48         //synchronized:
49         public synchronized void output(String name) {
50             for(int i = 0; i<name.length(); i++) {
51                 System.out.print(name.charAt(i));
52             }
53             System.out.println();// switch line
54         }
55         
56         //線程不安全
57         public  void output3(String name) {
58             for(int i = 0; i<name.length(); i++) {
59                 System.out.print(name.charAt(i));
60             }
61             System.out.println();// switch line
62         }
63         
64         //lock:
65         public void ouputlock(String name) {
66             Lock lock = new ReentrantLock();
67             lock.lock();   // 上鎖同步
68             try {
69                 for (int i = 0; i < name.length(); i++) {
70                     System.out.print(name.charAt(i));
71                 }
72                 System.out.println();// switch line
73             } finally {
74                 lock.unlock();  // 解鎖
75             }
76         }
77         
78 }
79     
80 }

 


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

-Advertisement-
Play Games
更多相關文章
  • 來認識更多的表單吧,增加知識面 我只創建了一個index.html幫助你認識它們 以下是代碼 1 <!DOCTYPE html> 2 <html> 3 <head> 4 <meta charset = "utf-8"> 5 </head> 6 <body> 7 <!--單按鈕輸入(單選)--> 8 ...
  • 先看看表單如何工作吧 請求 響應 簡要工作流程: 瀏覽器載入頁面 用戶輸入數據 用戶提交表單 伺服器響應 概念都清楚了,我們來寫表單吧 只有一個html文件 這是顯示 你可以向空白框框里寫一些東西,然後點擊提交 數據會發到web伺服器的contest.php里,當然了,你也可以自己寫一個php文件 ...
  • let命令的用法 是es6中的聲明一個變數的命令,只在它聲明的代碼塊中有效,出了這個代碼塊就會報錯。也非常適合 迴圈,在迴圈中i的值只在迴圈語句中生效,在外邊取不到的。 命令聲明的是一個全局的變數,i是指向全局的變數,只會輸出最後的值。而 只在迴圈語句塊裡面生效,每次迴圈都會重新聲明一個i的,所以每 ...
  • 如果這是第二次看到我的文章,歡迎右側掃碼訂閱我喲~ 👉 本文長度為2728字,建議閱讀8分鐘。 堅持原創,每一篇都是用心之作~ 前面聊完的2個章節「數據一致性」和「高可用」其實本質是一個通過提升複雜度讓整體更完善的方式。 接下去我們開始聊一些讓系統更簡單,更容易維護的東西——「易伸縮」,首當其衝的 ...
  •   需求變更是信息化過程中的家常便飯,而在變更過程中如何儘可能小的影響線上業務是比較頭疼的事情。舉個車聯網監控的例子:原終端設備上傳車輛的經緯度數據,新的終端設備支持同時上傳速度數據,而舊的車輛狀態表數據量超過億級,此時如果Alter table add column將會造成數據 ...
  • 一、問題描述 Tomcat下麵部署很多個java項目的war包,tomcat啟動一段時間後,發現cpu占用過高,整個界面卡死! 二、通過process explorer查看java進程下的線程 process explorer是Windows系統和應用程式監視工具。 process explorer ...
  • 字元串,是Java中最重要的類。這句肯定的推斷不是Java之父詹姆斯·高斯林說的,而是沉默王二說的,因此你不必懷疑它的準確性。 關於字元串,有很多的面試題,但我總覺得理論知識繞來繞去沒多大意思。你比如說:String cmower = new String("沉默王二");定義了幾個對象? 我總覺得 ...
  • 首先嘗試網友們的方法 按照網上大部分的教程仍無法解決自己的問題,不管是更改.project文件還是使用其他修改配置的方法,始終不能解決問題。 嘗試自己解決問題 想到了可能是eclipse版本的問題,我之前使用的是eclipse mars版本。我隨即去官網下載了最新版本的eclipse。下載後導入項目 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...