大家好,java實(shí)現(xiàn)多線程的幾種方式相信很多的網(wǎng)友都不是很明白,包括java 多線程實(shí)現(xiàn)方式也是一樣,不過沒有關(guān)系,接下來就來為大家分享關(guān)于java實(shí)現(xiàn)多線程的幾種方式和java 多線程實(shí)現(xiàn)方式的一些知識(shí)點(diǎn),大家可以關(guān)注收藏,免得下次來找不到哦,下面我們開始吧!
java多線程如何實(shí)現(xiàn)在多CPU上分布
一個(gè)JAVA線程就是一個(gè)OS線程,線程調(diào)度依賴于操作系統(tǒng),JVM層面不干預(yù)。
JAVA語言層面講,只是規(guī)范,并不要求具體如何實(shí)現(xiàn)。具體的實(shí)現(xiàn)模型有幾種,比如1:1,N:1,1:N,具體怎么選,JVM廠商自己的事兒。
在LINUX下,你可以用JNI來調(diào)用taskset,實(shí)現(xiàn)把線程分配到某一個(gè)CPU上。但是這么做的理由我現(xiàn)在還想不出來。
實(shí)際工作中,什么場景會(huì)用到多線程開發(fā)
最典型的應(yīng)用比如tomcat,tomcat內(nèi)部采用的就是多線程,上百個(gè)客戶端訪問同一個(gè)web應(yīng)用,tomcat接入后都是把后續(xù)的處理扔給一個(gè)新的線程來處理,這個(gè)新的線程最后調(diào)用到我們的servlet程序,比如doGet或者doPost方法。
如果不采用多線程機(jī)制,上百個(gè)人同時(shí)訪問一個(gè)web應(yīng)用的時(shí)候,tomcat就得排隊(duì)串行處理了,那樣客戶端根本是無法忍受那種訪問速度的。
還有就是需要異步處理的時(shí)候,需要使用多線程。比如taska和taskb要并行處理,單個(gè)線程只能串行處理,先做完taska然后再做taskb。如果想要多個(gè)task同時(shí)執(zhí)行的話,就必須為每個(gè)task分配一個(gè)線程,然后通過java虛擬機(jī)的線程調(diào)度,來同時(shí)執(zhí)行多個(gè)任務(wù)。比如你的CPU是多核心的話,就可以讓一個(gè)CPU執(zhí)行一個(gè)線程。如果只有一個(gè)CPU的話,底層是按照分時(shí)復(fù)用的原則,各個(gè)線程按照時(shí)間片來獲得CPU資源。
php多線程教程
PHP+shell實(shí)現(xiàn)多線程的方法
先寫個(gè)簡單的php代碼,這里為了讓腳本執(zhí)行時(shí)間更長,方便看效果,sleep一下,呵呵!先看下test.php的代碼:ls
PHP代碼:
for($i=0;$i<10;$i++){
echo$i;
sleep(10);
}
?>
在看下shell腳本的代碼,非常簡單
#!/bin/bash
foriin12345678910
do
/usr/bin/php-q/var/www/html/test.php&
done
注意到在請求php代碼的那行有一個(gè)&符號(hào)嗎,這個(gè)是關(guān)鍵,不加的話是不能進(jìn)行多線程的,&表示講服務(wù)推送到后臺(tái)執(zhí)行,因此,在shell的每次的循環(huán)中不必等php的代碼全部執(zhí)行完在請求下一個(gè)文件,而是同時(shí)進(jìn)行的,這樣就實(shí)現(xiàn)了多線程,下面運(yùn)行下shell看下效果,這里你將看到10個(gè)test.php進(jìn)程再跑,再利用linux的定時(shí)器,定時(shí)請求這個(gè)shell,在處理一些需要多線程的任務(wù),例如,批量下載時(shí),非常好用!
php中用WEB服務(wù)器實(shí)現(xiàn)多線程
假設(shè)我們現(xiàn)在運(yùn)行的是a.php這個(gè)文件.但是我在程序中又請求WEB服務(wù)器運(yùn)行另一個(gè)b.php,那么這兩個(gè)文件將是同時(shí)執(zhí)行的.(PS:一個(gè)鏈接請求發(fā)送之后,WEB服務(wù)器就會(huì)執(zhí)行它,而不管客戶端是否已經(jīng)退出)
有些時(shí)候,我們想運(yùn)行的不是另一個(gè)文件,而是本文件中的一部分代碼.該怎么辦呢?
其實(shí)可是通過參數(shù)來控制a.php來運(yùn)行哪一段程序.
下面看一個(gè)例子:
//a.php,b.php
PHP代碼:--------------------------------------------------------------------------------
functionrunThread()
{
$fp=fsockopen('localhost',80,$errno,$errmsg);
fputs($fp,"GET/b.php?act=b\r\n\r\n");//這里的第二個(gè)參數(shù)是HTTP協(xié)議中規(guī)定的請求頭
//不明白的請看RFC中的定義
fclose($fp);
}
functiona()
{
$fp=fopen('result_a.log','w');
fputs($fp,'Setin'.Date('h:i:s',time()).(double)microtime()."\r\n");
fclose($fp);
}
functionb()
{
$fp=fopen('result_b.log','w');
fputs($fp,'Setin'.Date('h:i:s',time()).(double)microtime()."\r\n");
fclose($fp);
}
if(!isset($_GET['act']))$_GET['act']='a';
if($_GET['act']=='a')
{
runThread();
a();
}
elseif($_GET['act']=='b')b();
?>
--------------------------------------------------------------------------------
打開result_a.log和result_b.log比較一下兩個(gè)文件的中訪問的時(shí)間.大家會(huì)發(fā)現(xiàn),這兩個(gè)的確是在不同線程中運(yùn)行的.有些時(shí)間完全一樣.
上面只是一個(gè)簡單的例子,大家可以改進(jìn)成其它形式.
既然PHP中也能多線程了,那么問題也來了,那就是同步的問題.我們知道PHP本身是不支持多線程的.所以更不會(huì)有什么像Java中synchronize的方法了.那我們該如何做呢.
1.盡量不訪問同一個(gè)資源.以避免沖突.但是可以同時(shí)像數(shù)據(jù)庫操作.因?yàn)閿?shù)據(jù)庫是支持并發(fā)操作的.所以在多線程的PHP中不要向同一個(gè)文件中寫入數(shù)據(jù).如果必須要寫的話,用別的方法進(jìn)行同步..如調(diào)用flock對文件進(jìn)行加鎖等.或建立臨時(shí)文件并在另外的線程中等待這個(gè)文件的消失while(file_exits('xxx'));這樣就等于這個(gè)臨時(shí)文件存在時(shí),表示其實(shí)線程正在操作
如果沒有了這個(gè)文件,說明其它線程已經(jīng)釋放了這個(gè).
2.盡量不要從runThread在執(zhí)行fputs后取這個(gè)socket中讀取數(shù)據(jù).因?yàn)橐獙?shí)現(xiàn)多線程,需要的用非阻塞模式.即在像fgets這樣的函數(shù)時(shí)立即返回..所以讀寫數(shù)據(jù)就會(huì)出問題.如果使用阻塞模式的話,程序就不算是多線程了.他要等上面的返回才執(zhí)行下面的程序.所以如果需要交換數(shù)據(jù)最后利用外面文件或數(shù)據(jù)中完成.實(shí)在想要的話就用socket_set_nonblock($fp)來實(shí)現(xiàn).
說了這么多,倒底這個(gè)有沒有實(shí)際的意義呢?在什么時(shí)候需要這種用這種方法呢?
答案是肯定的.大家知道.在一個(gè)不斷讀取網(wǎng)絡(luò)資源的應(yīng)用中,網(wǎng)絡(luò)的速度是瓶頸.如果采多這種形式就可以同時(shí)以多個(gè)線程對不同的頁面進(jìn)行讀取.
本人做的一個(gè)能從8848、soaso這些商城網(wǎng)站搜索信息的程序。還有一個(gè)從阿里巴巴網(wǎng)站上讀取商業(yè)信息和公司目錄的程序也用到了此技術(shù)。因?yàn)檫@兩個(gè)程序都是要不斷的鏈接它們的服務(wù)器讀取信息并保存到數(shù)據(jù)庫。利用此技術(shù)正好消除了在等待響應(yīng)時(shí)的瓶頸。
php模擬實(shí)現(xiàn)多線程的三種方法
PHP語言本身是不支持多線程的.總結(jié)了一下網(wǎng)上關(guān)于PHP模擬多線程的方法,總的來說,都是利用了PHP的好伙伴們本身所具有的多線程能力.PHP的好伙伴指的就是LINUX和APACHE啦,LAMP嘛.
另外,既然是模擬的,就不是真正的多線程.其實(shí)只是多進(jìn)程.進(jìn)程和線程是兩個(gè)不同的概念.好了,以下方法都是從網(wǎng)上找來的.
1.利用LINUX操作系統(tǒng)
for($i=0;$i<10;$i++){
echo$i;
sleep(5);
}
?>
上面存成test.php,然后寫一段SHELL代碼
#!/bin/bash
foriin12345678910
do
php-qtest.php&
done
2.利用fork子進(jìn)程(其實(shí)同樣是利用LINUX操作系統(tǒng))
declare(ticks=1);
$bWaitFlag=FALSE;///是否等待進(jìn)程結(jié)束
$intNum=10;///進(jìn)程總數(shù)
$pids=array();///進(jìn)程PID數(shù)組
echo("Startn");
for($i=0;$i<$intNum;$i++){
$pids[$i]=pcntl_fork();///產(chǎn)生子進(jìn)程,而且從當(dāng)前行之下開試運(yùn)行代碼,而且不繼承父進(jìn)程的數(shù)據(jù)信息
if(!$pids[$i]){
//子進(jìn)程進(jìn)程代碼段_Start
$str="";
sleep(5+$i);
for($j=0;$j<$i;$j++){$str.="*";}
echo"$i->".time()."$strn";
exit();
//子進(jìn)程進(jìn)程代碼段_End
}
}
if($bWaitFlag)
{
for($i=0;$i<$intNum;$i++){
pcntl_waitpid($pids[$i],$status,WUNTRACED);
echo"wait$i->".time()."n";
}
}
echo("Endn");
?>
3.利用WEBSERVER,PHP不支持多線程,APACHE可是支持的,呵呵.
假設(shè)我們現(xiàn)在運(yùn)行的是a.php這個(gè)文檔.但是我在程式中又請求WEB服務(wù)器運(yùn)行另一個(gè)b.php
那么這兩個(gè)文檔將是同時(shí)執(zhí)行的.(代碼同上)
當(dāng)然啦,也可以把需要多線程處理的部分交給JAVA去處理,然后在PHP里調(diào)用,哈哈.
system('javamultiThread.java');
?>
擴(kuò)展資料:PHP即“超文本預(yù)處理器”,是一種通用開源腳本語言。PHP是在服務(wù)器端執(zhí)行的腳本語言,與C語言類似,是常用的網(wǎng)站編程語言。PHP獨(dú)特的語法混合了C、Java、Perl以及PHP自創(chuàng)的語法。利于學(xué)習(xí),使用廣泛,主要適用于Web開發(fā)領(lǐng)域。
實(shí)現(xiàn)線程庫的兩種不同方法
java多線程的的兩種方法,要注意也有細(xì)微的差別!繼承thread類實(shí)現(xiàn)runnable接口(注意:共享數(shù)據(jù)的時(shí)候?qū)崿F(xiàn)runnable接口,thread類不支持)
python中的多線程和JAVA中的多線程有什么區(qū)別嗎
python是支持多線程的,但是python里的多線程是單cpu意義上的多線程,它和多cpu上的多線程有著本質(zhì)的區(qū)別,這是因?yàn)閜ython存在一個(gè)叫GlobalInterpreterLock(GIL)全局解釋器鎖。
在解釋器解釋執(zhí)行任何Python代碼時(shí),都需要先獲得這把鎖,也就是說在同一時(shí)刻內(nèi),只有一條線程可以在CPU中運(yùn)行。
但是python的多線程并不是毫無用處的。當(dāng)遇到I/O操作時(shí)會(huì)釋放這把GIL鎖,所以如果程序是一個(gè)IO密集型的程序,一個(gè)線程處在IO等待的時(shí)候另一個(gè)線程便可以取得鎖并在CPU中運(yùn)行,這時(shí)就發(fā)揮了多線程的作用。
但如果是純計(jì)算的程序,沒有I/O操作,那么只有取得GIL鎖的線程可以在CPU中運(yùn)行,其它的線程都處于等待狀態(tài),等待持有GIL鎖的線程的釋放鎖,也就相當(dāng)于單線程在跑(而且上下文切換也會(huì)有所開銷)。
Java方面,其提供了并發(fā)機(jī)制:一個(gè)進(jìn)程中可以并發(fā)多個(gè)線程,每條線程并行執(zhí)行不同的任務(wù)。因?yàn)榫€程運(yùn)行于多核CPU上,各線程可分布于CPU的各個(gè)核心,所以可以讓程序?qū)崿F(xiàn)真正的并發(fā)。
以上就是python和java多線程的區(qū)別,希望我的回答對你有所幫助。
關(guān)于java實(shí)現(xiàn)多線程的幾種方式到此分享完畢,希望能幫助到您。