您好,登錄后才能下訂單哦!
這篇文章主要介紹了怎么使用Java多線程實現簡易微信發紅包,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。
一、
首先我們先大致了解一下什么是多線程。(書上的解釋)
程序是一段靜態的代碼,它是應用軟件的藍本。進程是程序的一次動態執行過程,對應了從代碼加載執行,執行到執行完畢的一個完整的過程。
線程不是進程,線程是比進程更小的執行單位,一個進程在其執行過程中,可以產生多個線程形成多條執行線索,每條線索即每個線程也有它自身的產生,存在,消亡的過程,和進程共享操作系統的資源類似,線程間也可以共享進程中的某些內存單元,并利用這些共享單元來實現數據交換,實時通信與必要的同步操作,但與進程不同的是線程的中斷和恢復更加節省開支。線程是運行在進程中的“小進程”。
多線程是指一個應用程序中同時存在幾個執行體,按幾條不同的執行線索共同工作的情況。雖然看似是幾個事件同時發生,但其實計算機在任何給定時刻只能執行那些線程中的一個。為了建立這些線程在同步進行的感覺,Java虛擬機快速的把控制從一個線程切換到另一個線程。這些線程將被輪流執行,使得每個線程都有機會使用CPU資源。
二、
利用單線程實現的簡易微信發紅包
共寫有三種方法,其中第一種,第二種未設置范圍,紅包數和人數為一一對應,第三種增添了取值范圍以及計數器,人多紅包少有未搶到現象發生。
(1) 方法一
import java.util.Scanner; import com.sun.deploy.security.SelectableSecurityManager; import java.util.Random; public class 簡易微信發紅包 { public static void main(String[] args) { Scanner scanner=new Scanner(System.in); int n; double money; System.out.println("請輸入您想要發的紅包數量"); n=scanner.nextInt(); System.out.println("請輸入您發送的紅包金額"); money=scanner.nextDouble(); T2 t2=new T2(n,money); t2.Rob(); } } class T2 { public double remain;//有紅包被領取后的余額 int n;//紅包數量 T2(int n,double money) { this.remain=money; this.n=n; } int a=1; public void Rob() { while (n > 0) { double x2; if (n != 1) {//因為最后一個人領取金額為前面人領取紅包后剩下的,所以無需再進行隨機 x2 = process();//取隨機金額 while (judge(x2) != 1) {//判斷取到的隨機金額是否非法,即無法保證后來每個紅包領取者領到最低金額0.01 x2 = process();//若非法則重新取隨機金額 } remain = remain - x2;//當領取成功后余額減去領走的金額 n--;//確保每次判斷人數為所剩紅包數減1 System.out.println("紅包獲得者" + a + "獲得" + x2 + "元");//此處默認領取者順序為升序 a++;//控制輸出順序 } else { x2 = remain;//因為最后一個人領取金額為前面人領取紅包后剩下的,所以無需再進行隨機 String str = String.valueOf(x2); String str1 = String.format("%.2f", x2); x2 = Double.parseDouble(str1); System.out.println("紅包獲得者" + a + "獲得" + x2 + "元"); n--;//確保每次判斷人數為所剩紅包數減1 } } } public int judge(double x){//判斷函數 if(remain-x>(n-1)*0.01){//確保后來紅包領取者最少能領到最低金額0.01 return 1; } else return 0; } public double process() {//實現紅包金額隨機的函數 double x2; double x1; String str1; Random random = new Random();//隨機數為取0到1之間的任意double值 x1 = remain*random.nextDouble(); str1= String.format("%.2f",x1);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 x2=Double.parseDouble(str1);//再將字符串型數據轉換成double型 while(x2==0){//如果所取金額非法則回爐重造 x1 = remain*random.nextDouble(); str1= String.format("%.2f",x1);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 x2=Double.parseDouble(str1);//再將字符串型數據轉換成double型 } return x2; } }
程序運行結果如下
(2) 方法二
import java.util.Random; import java.util.Scanner; public class 簡易微信發紅包2 { public static void main(String[] args) { Scanner scanner=new Scanner(System.in); double money=0;//紅包總金額 int n;//紅包個數 System.out.println("請輸入您想要發的紅包數量"); n=scanner.nextInt(); System.out.println("請輸入您發送的紅包金額"); money=scanner.nextDouble(); if(money/n==0.01){//當所發金額剛好為每人0.01元時 T6 t6=new T6(money,n); t6.Rob(); }else{ T5 t5=new T5(money,n); t5.Rob(); } } } class T5{ double remain; int n; T5(double money,int n){ this.remain=money; this.n=n; } int a=1; public void Rob(){ double max;//最大可領紅包金額 double x1;//隨機金額 double x2;//所得金額 while(n>0) { if (n != 1) {//前n-1個紅包領取者領的紅包為隨機金額紅包 max = remain - (n - 1) * 0.01;//最大可領紅包金額為剩下的人都獲得最小金額0.01 Random random = new Random(); x1 = (double) random.nextInt((int) ((max - 0.01) * 100)); //用nextInt而不用nextDouble的原因是nextDouble無法設置seed //上式中max-0.01,下面的x2+0.01即解決了隨機數取0導致紅包獲得者沒搶到錢的問題 x1 /= 100.0; x2 = x1 + 0.01; remain = remain - x2; n--; System.out.println("紅包獲得者" + a + "獲取金額為:" + String.format("%.2f", x2) + "元"); a++; } else {//最后一人領的紅包為前n-1個人領完后剩下的紅包 System.out.println("紅包獲得者" + a + "獲取金額為:" + String.format("%.2f", remain) + "元"); n--; } } } } class T6 { double remain; int n; T6(double money,int n){ this.remain=money; this.n=n; } public void Rob(){ for(int i=1;i<=n;i++){ System.out.println("紅包獲得者"+i+"獲得了0.01元"); } } }
程序運行結果如下:
(3) 方法三
import java.util.Random; import java.util.Scanner; public class 簡易微信發紅包3 { public static void main(String[] args) { int p,n; double money; System.out.println("請輸入您發送的紅包金額"); Scanner scanner=new Scanner(System.in); money=scanner.nextDouble(); System.out.println("請輸入您發送的紅包數量"); n=scanner.nextInt(); System.out.println("請輸入參與搶紅包的人數"); p=scanner.nextInt(); T7 t7=new T7(money,n,p); t7.Rob(); } } class T7 { double money; int n,p; int count =0;//計數器 double remain; T7(double money,int n,int p){ this.money=money;//總金額 this.n=n;//紅包數 this.p=p;//搶紅包人數 this.remain=money;//所剩金額 } public void Rob() { for(int i=1;i<=p;i++) { double x1, x2, d; String s1, s2; Random random = new Random(); d = money / (n - 1);//設置范圍讓每次所得金額不超過總數的1/(n-1),這樣也就避免了一次取得過大導致后面搶的紅包不能保證每個最少0.01 x1 = d * random.nextDouble(); s1 = String.format("%.2f", x1);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 x1 = Double.parseDouble(s1);//再將字符串型數據轉換成double型 while (x1 == 0 || x1 == money / (n - 1)) { x1 = d * random.nextDouble(); s1 = String.format("%.2f", x1);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 x1 = Double.parseDouble(s1);//再將字符串型數據轉換成double型 } s2 = String.format("%.2f", remain);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 remain = Double.parseDouble(s2);//再將字符串型數據轉換成double型 if (count < n - 1) {//前n-1個紅包金額為隨機金額 System.out.println( "紅包搶奪者"+i+ "搶到了" + s1 + "元"); remain -= x1; count++; } else if (count == n - 1) {//第n個為前n-1個紅包搶完所剩金額 System.out.println( "紅包搶奪者"+i+ "搶到了" + s2 + "元"); count++; } else if (count > n - 1) {//紅包被搶完后再來的 System.out.println( "紅包搶奪者"+i+ "哎呀,手慢了!沒搶到!"); count++; } } } }
程序運行結果如下:
三、
利用多線程實現的簡易微信發紅包
那么如何創建多線程呢?
1.通過繼承thread類創建多線程
JDK中提供了一個線程類Thread,通過繼承Thread類,并重寫Thread類中的run()方法便可實現多線程。
在Thread類中,提供了一個start()方法用于啟動新線程,線程啟動后,系統會自動調用run()方法,如果子類重寫了該方法便會執行子類中的方法。
run()方法中就是寫能夠被線程執行的程序。如果直接調用則相當于普通方法,必須使用start()方法,才能啟動線程,然后再由JVM去調用該線程的run()方法。
創建并啟動多線程的步驟
①定義Thread類的子類,并重寫該類的run方法,其方法體代表線程需要完成的任務。因此常把run方法稱為線程執行體。
②創建Thread子類的實例,即創建線程對象。
③用線程對象的start方法來啟動該線程。
2.通過實現Runnable接口創建多線程
通過繼承Thread類實現了多線程,但是這種方式有一定的局限性。因為Java中只支持單繼承,一個類一旦繼承了某個父類就無法再繼承Thread類。
Thread類提供了另外一個構造方法Thread(Runnable target),其中Runnable是一個接口,它只有一個run()方法。
當通過Thread(Runnable target))構造方法創建線程對象時,只需為該方法傳遞一個實現了Runnable接口的實例對象,這樣創建的線程將調用實現了Runnable接口中的run()方法作為運行代碼,而不需要調用Thread類中的run()方法。
創建并啟動多線程的步驟
①定義Runnable接口的實現類,并重寫該接口的run方法,該run方法的方法體同樣是該線程的線程執行體。
②創建Runnable實現類的實例,并以此為實例作為Thread的參數來創建Thread對象,該Thread對象才是真正的線程對象。
當多個線程使用同一個共享資源時,可以將處理共享資源的代碼放置在一個代碼塊中,使用synchronized關鍵字來修飾,被稱作同步代碼塊:
sychronized(lock){
操作共享資源代碼塊
}
其中:lock是一個鎖對象,它是同步代碼塊的關鍵。當線程執行同步代碼塊時,首先會檢查鎖對象的標志位,默認情況下,標志位為1,此時線程會執行同步代碼塊,同時將鎖對象的標志位置為0。當一個新的線程執行到這段同步代碼塊時,由于鎖對象的標志位為0,新線程會發生阻塞,等待當前線程執行完同步代碼塊后,鎖對象的標志位被置為1,新線程才能進入同步代碼塊執行其中的代碼。循環往復,直到共享資源被處理完為止。
同步代碼塊可以有效解決線程的安全問題,當把共享資源的操作放在synchronized定義的區域內時,便為這些操作加了同步鎖。
在方法前面同樣可以使用synchronized關鍵字來修飾,被修飾的方法為同步方法,它能實現和同步代碼塊同樣的功能,具體語法格式如下:
synchronized 返回值類型 方法名 {}
被synchronized修飾的方法在某一時刻只允許一個線程訪問,訪問該方法的其它線程都會發生阻塞,直到當前線程訪問完畢后,其它線程才有機會執行方法。
另外
public final String getName():獲取線程的名稱。
public static Thread currentThread():返回當前正在執行的線程對象,這樣就可以獲取任意方法所在的線程名稱。
Thread.currentThread().getName()
現將上面的單線程改成多線程實現
本篇文章多線程的創建以及實現用Runnable接口實現
(1)
import java.util.Scanner; import com.sun.deploy.security.SelectableSecurityManager; import java.util.Random; public class 微信發紅包多線程 { public static void main(String[] args) { Scanner scanner=new Scanner(System.in); int n; double money; System.out.println("請輸入您想要發的紅包數量"); n=scanner.nextInt(); System.out.println("請輸入您發送的紅包金額"); money=scanner.nextDouble(); T3 t3=new T3(n,money);//創建runnable實現類的實例 for (int j = 1; j <= n; j++) { new Thread(t3, "紅包獲得者" + j).start();//以上面創建的實例作為Thread的參數來創建Thread對象,并為Thread對象指定一個名字,用線程對象的start方法來啟動該線程。 } } } class T3 implements Runnable {//實現runnable接口 public double remain;//有紅包被領取后的余額 int n;//紅包數量 public synchronized void run() {//同步方法,在某一時刻只允許一個線程訪問,防止數據錯亂 Rob(); } T3(int n, double money) { this.remain = money; this.n = n; } int a = n; public void Rob() { double x2; if (n != 1) {//因為最后一個人領取金額為前面人領取紅包后剩下的,所以無需再進行隨機 x2 = process();//取隨機金額 while (judge(x2) != 1) {//判斷取到的隨機金額是否非法,即是否能保證后來每個紅包領取者領到最低金額0.01 x2 = process();//若非法則重新取隨機金額 } remain = remain - x2;//當領取成功后余額減去領走的金額 n--; //確保每次判斷人數為紅包數減一 } else { x2 = remain;//因為最后一個人領取金額為前面人領取紅包后剩下的,所以無需再進行隨機 String str = String.valueOf(x2); String str1 = String.format("%.2f", x2); x2 = Double.parseDouble(str1); } Thread th = Thread.currentThread();//返回當前正在執行的線程對象 String th_name = th.getName();//獲取線程的名稱 System.out.println(th_name + "搶到" + x2 + "元"); } public int judge(double x) {//判斷函數 if (remain - x > (n - 1) * 0.01) {//確保后來紅包領取者能領到最低金額0.01 return 1; } else return 0; } public double process() {//實現紅包金額隨機的函數 double x2; double x1; String str1; Random random = new Random();//隨機數為取0到1之間的任意double值 x1 = remain * random.nextDouble(); str1 = String.format("%.2f", x1);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 x2 = Double.parseDouble(str1);//再將字符串型數據轉換成double型 while (x2 == 0) {//如果所取金額非法則回爐重造 x1 = remain * random.nextDouble(); str1 = String.format("%.2f", x1);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 x2 = Double.parseDouble(str1);//再將字符串型數據轉換成double型 } return x2; } }
程序運行結果如下:
(2)
import java.util.Random; import java.util.Scanner; public class 簡易微信發紅包多線程2 { public static void main(String[] args) { Scanner scanner=new Scanner(System.in); double money=0;//紅包總金額 int n;//紅包個數 System.out.println("請輸入您想要發的紅包數量"); n=scanner.nextInt(); System.out.println("請輸入您發送的紅包金額"); money=scanner.nextDouble(); if(money/n==0.01){//當所發金額剛好為每人0.01元時 T4 t4=new T4(money,n); for(int i=1;i<=n;i++) { new Thread(t4,"紅包獲得者"+i).start(); } }else{ T1 t1=new T1(money,n); for(int i=1;i<=n;i++) { new Thread(t1,"紅包獲得者"+i).start(); } } } } class T1 implements Runnable{ double remain; int n; T1(double money,int n){ this.remain=money; this.n=n; } @Override public synchronized void run() { Rob(); } public void Rob(){ double max;//最大可領紅包金額 double x1;//隨機金額 double x2;//所得金額 if(n!=1) {//前n-1個紅包領取者領的紅包為隨機金額紅包 max=remain-(n-1)*0.01;//最大可領紅包金額為剩下的人都獲得最小金額0.01 Random random=new Random(); x1=(double)random.nextInt((int) ((max-0.01)*100)); //用nextInt而不用nextDouble的原因是nextDouble無法設置seed //上式中max-0.01,下面的x2+0.01即解決了隨機數取0導致紅包獲得者沒搶到錢的問題 x1/=100.0; x2=x1+0.01; remain=remain-x2; n=n-1; Thread th=Thread.currentThread();//獲取當前線程 String th_name=th.getName();//獲取線程名字 System.out.println(th_name+"獲取金額為:"+String.format("%.2f", x2)+"元"); } else {//最后一人領的紅包為前n-1個人領完后剩下的紅包 Thread th=Thread.currentThread();//獲取當前線程 String th_name=th.getName();//獲取線程名字 System.out.println(th_name+"獲取金額為:"+String.format("%.2f", remain)+"元"); } } } class T4 implements Runnable{ double remain; int n; T4(double money,int n){ this.remain=money; this.n=n; } public synchronized void run() { Rob(); } public void Rob(){ Thread th=Thread.currentThread();//獲取當前線程 String th_name=th.getName();//獲取線程名字 System.out.println(th_name+"獲取金額為:"+String.format("%.2f", remain/n)+"元"); } }
程序運行結果如下:
(3)
import java.util.Random; import java.util.Scanner; public class 簡易微信發紅包多線程3 { public static void main(String[] args) { int p,n; double money; System.out.println("請輸入您發送的紅包金額"); Scanner scanner=new Scanner(System.in); money=scanner.nextDouble(); System.out.println("請輸入您發送的紅包數量"); n=scanner.nextInt(); System.out.println("請輸入參與搶紅包的人數"); p=scanner.nextInt(); HH hh=new HH(money,n); for (int i=1;i<=p;i++){ new Thread(hh,"第"+i+"個人").start(); } } } class HH implements Runnable{ double money; int n; int count =0;//計數器 double remain; HH(double money,int n){ this.money=money;//總金額 this.n=n;//紅包數 this.remain=money;//所剩金額 } @Override public synchronized void run() { Rob(); } public void Rob(){ double x1,x2,d; String s1,s2; Thread th=Thread.currentThread();//獲取當前線程 String th_name=th.getName();//獲取線程名字 Random random=new Random(); d=money/(n-1);//設置范圍讓每次所得金額不超過總數的1/(n-1),這樣也就避免了一次取得過大導致后面搶的紅包不能保證每個最少0.01 x1=d*random.nextDouble(); s1=String.format("%.2f",x1);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 x1 = Double.parseDouble(s1);//再將字符串型數據轉換成double型 while(x1==0||x1==money/(n-1)){ x1=d*random.nextDouble(); s1=String.format("%.2f",x1);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 x1 = Double.parseDouble(s1);//再將字符串型數據轉換成double型 } s2= String.format("%.2f",remain);//轉化成字符串型用字符串型的format進行格式化處理,紅包金額最多取到小數點后兩位 remain = Double.parseDouble(s2);//再將字符串型數據轉換成double型 if (count<n-1){//前n-1個紅包金額為隨機金額 System.out.println(th_name+"搶到了"+s1+"元"); remain-=x1; count++; }else if (count==n-1){//第n個為前n-1個紅包搶完所剩金額 System.out.println(th_name+"搶到了"+s2+"元"); count++; }else if (count>n-1){//紅包被搶完后再來的 System.out.println(th_name+"哎呀,手慢了!沒搶到!"); count++; } } }
程序運行結果如下:
感謝你能夠認真閱讀完這篇文章,希望小編分享的“怎么使用Java多線程實現簡易微信發紅包”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關注億速云行業資訊頻道,更多相關知識等著你來學習!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。