java併發編程(5)併發程式測試

来源:http://www.cnblogs.com/zhangxinly/archive/2017/06/05/6939974.html
-Advertisement-
Play Games

併發程式測試 一、正確性測試 如:對一個自定義緩存的測試 ...


併發程式測試

 一、正確性測試

  如:對一個自定義緩存的測試

//自定義的緩存
public class SemaphoreBoundedBuffer <E> {
    private final Semaphore availableItems, availableSpaces;
    private final E[] items;
    private int putPosition = 0, takePosition = 0;

    public SemaphoreBoundedBuffer(int capacity) {
        if (capacity <= 0)
            throw new IllegalArgumentException();
        availableItems = new Semaphore(0);
        availableSpaces = new Semaphore(capacity);
        items = (E[]) new Object[capacity];
    }

    public boolean isEmpty() {
        return availableItems.availablePermits() == 0;
    }

    public boolean isFull() {
        return availableSpaces.availablePermits() == 0;
    }

    public void put(E x) throws InterruptedException {
        availableSpaces.acquire();
        doInsert(x);
        availableItems.release();
    }

    public E take() throws InterruptedException {
        availableItems.acquire();
        E item = doExtract();
        availableSpaces.release();
        return item;
    }

    private synchronized void doInsert(E x) {
        int i = putPosition;
        items[i] = x;
        putPosition = (++i == items.length) ? 0 : i;
    }

    private synchronized E doExtract() {
        int i = takePosition;
        E x = items[i];
        items[i] = null;
        takePosition = (++i == items.length) ? 0 : i;
        return x;
    }
}

//無需比較每次 生產者和消費者取出的值;只需要最終比較和即可
public class PutTakeTest extends TestCase {
    protected static final ExecutorService pool = Executors.newCachedThreadPool();
    protected CyclicBarrier barrier;
    protected final SemaphoreBoundedBuffer<Integer> bb;
    protected final int nTrials, nPairs;
    protected final AtomicInteger putSum = new AtomicInteger(0);
    protected final AtomicInteger takeSum = new AtomicInteger(0);
    public static void main(String[] args) throws Exception {
        new PutTakeTest(10, 10, 100000).test(); // sample parameters
        pool.shutdown();
    }
    public PutTakeTest(int capacity, int npairs, int ntrials) {
        this.bb = new SemaphoreBoundedBuffer<Integer>(capacity);
        this.nTrials = ntrials;
        this.nPairs = npairs;
        this.barrier = new CyclicBarrier(npairs * 2 + 1);   //初始化為 線程數量*2 + 1:生產者+消費者+主線程
    }
    void test() {
        try {
            for (int i = 0; i < nPairs; i++) {
                pool.execute(new Producer());
                pool.execute(new Consumer());
            }
            barrier.await(); // 等待所有線程初始化完成,完成後複位柵欄
            barrier.await(); // 等待所有線程執行完成
            assertEquals(putSum.get(), takeSum.get());      //執行比較,判斷線程安全性能
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
    static int xorShift(int y) {    //生成隨機數
        y ^= (y << 6);
        y ^= (y >>> 21);
        y ^= (y << 7);
        return y;
    }
    class Producer implements Runnable {
        public void run() {
            try {
                int seed = (this.hashCode() ^ (int) System.nanoTime());
                int sum = 0;
                barrier.await();                        //等待所有線程初始化完成
                for (int i = nTrials; i > 0; --i) {
                    bb.put(seed);
                    sum += seed;
                    seed = xorShift(seed);
                }
                putSum.getAndAdd(sum);
                barrier.await();                        //等待所有線程執行完成
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }
    }
    class Consumer implements Runnable {
        public void run() {
            try {
                barrier.await();                        //等待所有線程初始化完成
                int sum = 0;
                for (int i = nTrials; i > 0; --i) {
                    sum += bb.take();
                }
                takeSum.getAndAdd(sum);
                barrier.await();                        //等待所有線程執行完成
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        }
    }
}

 

 

 

 

 

 

 

 

 

 

 

  


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

-Advertisement-
Play Games
更多相關文章
  • 上一篇討論里我們介紹了幾種任務分配(Routing)模式。Akka提供的幾種現成智能化Routing模式大多數是通過對用戶屏蔽具體的運算Routee選擇方式來簡化Router使用,提高智能程度,所以我們提到Router的運算是一種無序的運算,消息之間絕對不容許任何形式的依賴,因為向Router發送的 ...
  • 一、thinkphp配置類型有哪些? 1.在thinkphp中,有6種配置。即慣例配置,應用配置、擴展配置、模塊配置、場景配置、動態配置。 2.慣例配置就是系統預設的配置。 3.應用配置就是我們自己開發創建的應用單獨的配置。 4.擴展配置主要是實現我們項目在開發過程中有一些擴展程式用到的配置。 5. ...
  • 顯式鎖 一、Lock與ReentrantLock Lock提供了一種無條件的、可輪詢的、定時的以及可中斷的鎖獲取操作,所有的加鎖和解鎖方法都是顯式的 ReentrantLock實現了Lock:並提供了和synchronized相同的記憶體語義;同時提供了可重入的加鎖語義 1.基本語義: 2.輪詢鎖與定 ...
  • Java運算符 運算符:是一種用於對數據進行運算,賦值,比較的特殊符號。 Java語言中運算符總共分為以下幾大類: 算術運算符:+ - * / ++ -- 示例如下代碼: 賦值運算符 首先要註意賦值運算符在Java中的優先順序是最低的,即在有其它運算符存在的運算式中永遠最後執行賦值運算符 示例代碼如下 ...
  • Python基礎 類變數和實例變數 寫在前面 如非特別說明,下文均基於Python3 大綱: 1. 類變數和實例變數 在 "Python Tutorial" 中對於類變數和實例變數是這樣描述的: Generally speaking, instance variables are for data ...
  • ​對於Java程式猿學習的建議 ​對於Java程式猿學習的建議 這一部分其實也算是今天的重點,這一部分用來回答很多群里的朋友所問過的問題,那就是LZ你是如何學習Java的,能不能給點建議? 今天LZ是打算來點乾貨,因此咱們就不說一些學習方法和技巧了,直接來談每個階段要學習的內容甚至是一些書籍。這一部 ...
  • 要深入學習註解,我們就必須能定義自己的註解,並使用註解,在定義自己的註解之前,我們就必須要瞭解Java為我們提供的元註解和相關定義註解的語法。 ...
  • 學習Java的同學註意了!!!學習過程中遇到什麼問題或者想獲取學習資源的話,歡迎加入Java學習交流群:159610322 我們一起學Java! 首先我們來講講:重載(Overloading) (1) 方法重載是讓類以統一的方式處理不同類型數據的一種手段。多個同名函數同時存在,具有不同的參數個數/類 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...