一步一步寫線程之一簡單的開始

一、多線程

多線程不管是在前面的文章分析中還是在網上還是大書籍上,學習C/C++多線程編程是無法繞過的,即使使用別人封裝好的框架,包括使用STL的一些庫,如果僅僅是簡單的應用,可能也就過去了。不過,稍微復雜的一些應用,其實就需要對多線程的應用有深層次的理解。
尤其在使用一些封裝庫時,特別要注意庫的一些注意事項,往往這些注意點是應用BUG的重要原因。比如std::async中std::future返回值如果控制不好,異步就可能變成了同步。所以說多線程要想寫好,不但要會用一些庫一些API,更要懂得這些庫的底層原理,直到這個庫調用的不同的平臺,封裝的是哪個OS的系統函數。如果再展開到并發和并行,更得需要了解相關OS的相關進程線程的調度知識等等。這些都是寫多線程的開發者需要一步步學習的目標。不過在這里,只就時論事,只講多線程,與OS相關的知識,用到就展開分析一下。
這個系列更側重于線程的應用,而非基礎知識的普及,所以如果在示例的一些源碼代碼中不清楚的基礎的用法,需要去查一下前面的多線程的基礎知識的文章或者自己行在網上查找一些相關知識。

二、線程的運用

多線程的應用其實是一個非常重要且普遍的應用。協程的編程目前看來應該是被排除到了更高的應用層上,也就是說,想用好協程,更多的是純上層應用了。線程被遷移到了一個中間層的庫應用。說得再深入一些,協程的普及,可能會讓開發者寫并發變得簡單但同樣大幅降低了技術門檻,這意味著什么,以前不明白的,這兩年應該都懂了。
多線程的應用一般會和異步IO共同使用,有的人在網上說多線程和異步有什么不同,這就說汽車和鋼鐵有什么不同一樣,這本來就是兩回事。說多了,扯回來。
多線程的應用非常廣泛,圖像處理、數據讀寫、IO通信和時間控制等等,都可以用到。但正如已經分析過的一樣,用的方法不一樣,那么產生的結果也不一樣。單純的多線程讀寫操作不涉及到數據互斥沒有什么難度。舉個簡單的例子,寫十個線程,讀十個不同的文件,這和寫普通的非線程代碼沒有本質不同,可能只是套上了十個線程的殼子。但是如果這十個線程需要同時把讀到數據寫到一個緩沖區內,根據不同情況來覆蓋或者追加相關數據,這就需要謹慎的處理了。否則數據很可能不是丟失就是多存儲了。
另外,如果只有五個線程來處理十個文件呢?如果這十個文件大小不一,有的非常大,有的非常小?那么小文件讀取完成后,這個線程不就空閑了么?多線程不還是單線程么。再細分一下,大文件可不可以切成段,在別的線程讀完成文件后去讀這些分段的大文件來加速讀取的速度呢?這都是多線程需要解決的現實問題。
很多問題,在學習多線程編程時,是很難想到的或者說遇到實際問題不知道怎么下手,這其實就是學無法致用的一個典型的表現。在這個系列中,就把這些多線程的用法與實際的情況結合起來,進行一一的分析。

三、例程

做為簡單的開始,本篇不做多么復雜的例程,只是把剛剛提到std::async遇到的同步問題展現一下以及相關的處理辦法:

#include <iostream>       
#include <future>         
#include <chrono>int getData(int x) {std::cout << "async thread run start!"<<std::endl;std::this_thread::sleep_for(std::chrono::milliseconds(6000));std::cout << "async thread run end!" << std::endl;return x;
}int main()
{std::async(std::launch::async, getData, 100);std::cout << "go to ..."<<std::endl;return 0;
}

運行結果:

async thread run start!
async thread run end!
go to ...

你會發現運行的結果并不是期房的兩個線程并行執行,而執行完async中的線程,才回頭執行主線程的函數。什么原因呢,在std::async中有下面的話:
“If the std::future obtained from std::async is not moved from or bound to a reference, the destructor of the std::future will block at the end of the full expression until the asynchronous operation completes, essentially making code such as the following synchronous”
在std::future的析構函數中:
“these actions will not block for the shared state to become ready, except that they may block if all of the following are true:
the shared state was created by a call to std::async,
the shared state is not yet ready, and
the current object was the last reference to the shared state.”
在下面還有一句話,意思是說只在運行策略為“ std::launch::async ”才會產生這種情況。
所以通過上述兩個說明就明白了,在std::async中,std::future的臨時變量會在沒有被移動或者引用情況下一直阻塞到異步計算完成,所以這句話也就給了解決方法:

auto d = std::async(std::launch::async, getData, 100);

只需增加上面的返回值處理即可,延長一下std::future的生命周期至期望的位置即可。用別人的東西,就得服別人的套路,一個不小心,就會吃虧上當。不過話又說回來,想省事,吃點虧也應該,對吧。

四、總結

從簡單開始,不忘初心,朝著實際應用不斷的前進。多線程編程屬于那種入門容易,寫好難,精通更難的一種編程技術,真正的生產上,大牛們被憋住當場打臉的情況也不少見。總之一句話,多看書,多實踐,多應用。小心寫代碼,防御性編程,少引入BUG比如事后調試定位要更容易做到一些。

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/210708.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/210708.shtml
英文地址,請注明出處:http://en.pswp.cn/news/210708.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

VBA_MF系列技術資料1-237

MF系列VBA技術資料 為了讓廣大學員在VBA編程中有切實可行的思路及有效的提高自己的編程技巧&#xff0c;我參考大量的資料&#xff0c;并結合自己的經驗總結了這份MF系列VBA技術綜合資料&#xff0c;而且開放源碼&#xff08;MF04除外&#xff09;&#xff0c;其中MF01-04屬于定…

[Linux] 用LNMP網站框架搭建論壇

一、nginx在其中工作原理 原理&#xff1a; php-fpm.conf是控制php-fpm守護進程 它是php.ini是一個php解析器 工作過程&#xff1a; 1.當客戶端通過域名請求訪問時&#xff0c;Nginx會找到對應的虛擬主機 2. Nginx將確定請求。 對于靜態請求&#xff0c;Nginx會自行處理…

結構體和位段

結構體&#xff1a; C語言中&#xff0c;我們之前使用的都是C語言中內置的類型&#xff0c;比如整形&#xff08;int&#xff09;、字符型&#xff08;char&#xff09;、單精度浮點型&#xff08;float&#xff09;等。但是我們知道&#xff0c;我們現實世界中&#xff0c;還…

聊聊spring.mvc.servlet.load-on-startup

序 本文主要研究一下spring.mvc.servlet.load-on-startup spring.mvc.servlet.load-on-startup org/springframework/boot/autoconfigure/web/servlet/WebMvcProperties.java ConfigurationProperties(prefix "spring.mvc") public class WebMvcProperties {//.…

json精講

本文介紹json的規范及javascript和java對數據的交換讀取 1. json介紹1.1 json簡介1.2為什么使用 JSON&#xff1f; 2. json規范2.1基礎規范2.2 key值為-字符串、數字、布爾值2.3 key值為對象Object2.4 key值為數組2.5 json本身就是一個數組 3.javascript操作json3.1 javascript…

WPF(Windows Presentation Foundation) 的 Menu控件

WPF&#xff08;Windows Presentation Foundation&#xff09;的 Menu 是一種用于創建菜單的控件。菜單通常位于應用程序窗口的頂部&#xff0c;并提供了一組命令或選項&#xff0c;用于導航到不同的功能區域、執行特定的操作或訪問特定的功能。 Menu 控件是 WPF 中的一個容器…

2、關于使用ajax驗證繞過(實例2)

ajax原理我上一篇有寫過&#xff0c;參考&#xff1a;1、關于前端js-ajax繞過-CSDN博客 一、實例環境&#xff1a; 為手機上的某一割韭菜app 二、目的&#xff1a; 實現繞過手機驗證碼&#xff0c;找回密碼 三、工具&#xff1a; bp代理 四、驗證步驟如下&#xff1a; …

ECU安全學習網站和書籍介紹

ECU安全是指關注和保護汽車電子控制單元&#xff08;ECU&#xff09;的安全性和防護措施。ECU是現代汽車中的關鍵組件&#xff0c;它負責監控和控制車輛各種系統的運行&#xff0c;如發動機、制動、轉向等。ECU安全的重要性在于防止惡意攻擊者操控或干擾車輛的操作。 ECU安全涉…

hive自定義函數及案例

一.自定義函數 1.Hive自帶了一些函數&#xff0c;比如&#xff1a;max/min等&#xff0c;但是數量有限&#xff0c;自己可以通過自定義UDF來方便的擴展。 2.當Hive提供的內置函數無法滿足你的業務處理需要時&#xff0c;此時就可以考慮使用用戶自定義函數。 3.根據用戶自定義…

GitHub為Rust語言添加了供應鏈安全工具

GitHub的供應鏈安全特性包括咨詢數據庫、Dependabot警報和依賴關系圖現在可以用于Rust Cargo文件。 為了幫助Rust開發人員發現和防止安全漏洞&#xff0c;GitHub已經為快速增長的Rust語言提供了供應鏈安全特性套件。 這些特性包括GitHub Advisory Database&#xff0c;它已經有…

構建外賣系統:使用Django框架

在當今數字化的時代&#xff0c;外賣系統的搭建不再是什么復雜的任務。通過使用Django框架&#xff0c;我們可以迅速建立一個強大、靈活且易于擴展的外賣系統。本文將演示如何使用Django構建一個簡單的外賣系統&#xff0c;并包含一些基本的技術代碼。 步驟一&#xff1a;安裝…

shell的條件測試

shell 的條件測試 概述 條件測試是 shell 編程中非常重要的一個概念&#xff0c;它允許我們根據某個條件是否滿足&#xff0c;來選擇執行相應的任務。 條件測試的語法 shell 中的條件測試語法如下&#xff1a; [ 條件表達式 ]如果條件表達式為真&#xff0c;則返回 0&…

CentOS 7.9--離線安裝python3.9.18+virtualenv-20.25.0

# 想在centos6.x 上安裝新版本的python&#xff0c;但是擔心在用系統的環境被破壞&#xff0c;所以需要安裝python虛擬環境&#xff0c;然后就找到自用的aliyun主機先測試下離線安裝&#xff0c;在用6.X環境是沒有互聯網的&#xff0c;必須需要離線安裝。 1. 下載對應python源…

力扣解題之保姆教程:(1)兩數之和(代碼詳解)

題目描述 給定一個整數數組 nums 和一個整數目標值 target&#xff0c;請你在該數組中找出 和為目標值 target 的那 兩個 整數&#xff0c;并返回它們的數組下標。 假設每種輸入只會對應一個答案。但是&#xff0c;數組中同一個元素在答案里不能重復出現。你可以按任意順序返回…

Django模板

以下是一個簡單的Django模板示例&#xff1a; <!DOCTYPE html> <html><head><title>{{ title }}</title></head><body><h1>{{ heading }}</h1><p>{{ content }}</p></body> </html>一、模板的…

波奇學Linux:父子進程和進程狀態

vim編輯器&#xff0c;編寫一個程序模擬進程 在vim中查看sleep函數 底行模式輸入 寫個Makefile自動運行波奇學Linux:yum和vim-CSDN博客 運行程序 PID和PPID 查看進程目錄信息 實際有過濾出來有兩個&#xff0c;一個進程本身一個是grep程序&#xff0c;通過 -v grep過濾走含gre…

新版Android Studio 正則表達式匹配代碼注釋,刪除注釋,刪除全部注釋,IntelliJ IDEA 正則表達式匹配代碼注釋

正則表達式匹配代碼注釋 完整表達式拼接Android Studio 搜索匹配【IntelliJ IDEA 也是一樣的】 完整表達式拼接 (/*{1,2}[\s\S]?*/)|(//[\x{4e00}-\x{9fa5}].)|(<!-[\s\S]?–>)|(^\s\n)|(System.out.println.*) 表達式拆解&#xff0c;可以根據自己需求自由組合&#x…

Mybatis、Mybatis整合Spring的流程圖

Mybatis 注意MapperProxy里面有invoke方法&#xff0c;當進到invoker方法會拿到 二、mybatis整合Spring 1、當我們的拿到的【Dao】其實就是【MapperProxy】&#xff0c;執行Dao的方法時&#xff0c;會被MapperProxy的【Invoke方法攔截】 2、圖上已經標注了MapperProxy包含哪些…

力扣:200. 島嶼數量(Python3)

題目&#xff1a; 給你一個由 1&#xff08;陸地&#xff09;和 0&#xff08;水&#xff09;組成的的二維網格&#xff0c;請你計算網格中島嶼的數量。 島嶼總是被水包圍&#xff0c;并且每座島嶼只能由水平方向和/或豎直方向上相鄰的陸地連接形成。 此外&#xff0c;你可以假…

STM32-TIM定時器中斷

目錄 一、TIM&#xff08;Timer&#xff09;定時器簡介 二、定時器類型 2.1基本定時器結構 2.2通用定時器結構 2.3高級定時器結構 三、定時中斷基本結構 四、時序圖分析 4.1 預分頻器時序 4.2 計數器時序 4.3 計數器無預裝時序&#xff08;無影子寄存器&#xff09; …