使用Rx的計算和最大化線程處理并行任務(wù)

歡迎Follow我的GitHub, 關(guān)注我的簡書. 其余參考Android目錄.

Threads

本文的合集已經(jīng)編著成書,高級Android開發(fā)強(qiáng)化實(shí)戰(zhàn)惩琉,歡迎各位讀友的建議和指導(dǎo)屈呕。在京東即可購買:https://item.jd.com/12385680.html

Android

隨著手機(jī)的發(fā)展, 其性能已經(jīng)與電腦越來越接近, 也會有一些復(fù)雜耗時的并行任務(wù)需要處理, 對于異步與并行, RxAndroid是我們的最佳選擇. 那么讓我來使用實(shí)例介紹一下吧.

在計算調(diào)度器Schedulers.computation()中, 可以并行處理任務(wù), 核數(shù)是Rx根據(jù)手機(jī)CPU定制的, 在我的華為P8手機(jī)(8核)中, 使用的是8個線程. 但是根據(jù)Java線程的最佳配置而言, 8核最佳是9個線程, 即線程數(shù)等于核數(shù)+1.

本文源碼的GitHub下載地址


配置

RxAndroid+ButterKnife, 我是ButterKnife的粉絲.

    compile 'com.jakewharton:butterknife:7.0.1'
    compile 'io.reactivex:rxjava:1.1.0'
    compile 'io.reactivex:rxandroid:1.1.0'

計算線程

MAX是并行執(zhí)行的任務(wù)數(shù). 使用flatMap逐個分發(fā)到計算線程computation中, 執(zhí)行耗時任務(wù)intenseCalculation.

    // 計算線程并行, 8核
    public void computePara(View view) {
        mTvComputeValue.setText("計算中");
        Observable.range(MIN, MAX)
                .flatMap(i -> Observable.just(i)
                                .subscribeOn(Schedulers.computation()) // 使用Rx的計算線程
                                .map(this::intenseCalculation)
                )
                .observeOn(AndroidSchedulers.mainThread())
                .subscribe(this::computeTag);
    }

使用intenseCalculation模擬耗時任務(wù).

    // 模擬耗時計算
    private int intenseCalculation(int i) {
        try {
            tag("Calculating " + i + " on " + Thread.currentThread().getName());
            Thread.sleep(randInt(100, 500));
            return i;
        } catch (InterruptedException e) {
            throw new RuntimeException(e);
        }
    }

最大線程

最大線程數(shù) = CPU核數(shù) + 1

由于CPU的核數(shù)是8, 因此我們選擇9個線程. 創(chuàng)建執(zhí)行器executor, 使用執(zhí)行器創(chuàng)建Rx的調(diào)度器Scheduler, 處理異步任務(wù).

    // 定制線程并行, 9核
    public void customPara(View view) {
        int threadCt = Runtime.getRuntime().availableProcessors() + 1;
        mTvCustomValue.setText(String.valueOf("計算中(" + threadCt + "線程)"));

        ExecutorService executor = Executors.newFixedThreadPool(threadCt);
        Scheduler scheduler = Schedulers.from(executor);

        Observable.range(MIN, MAX)
                .flatMap(i -> Observable.just(i)
                                .subscribeOn(scheduler)
                                .map(this::intenseCalculation)
                ).observeOn(AndroidSchedulers.mainThread())
                .subscribe(this::customTag);
    }

高版本計算CPU核數(shù)的方式.

Runtime.getRuntime().availableProcessors()

低版本, 參考.

    private int getNumCoresOldPhones() {
        //Private Class to display only CPU devices in the directory listing
        class CpuFilter implements FileFilter {
            @Override
            public boolean accept(File pathname) {
                //Check if filename is "cpu", followed by a single digit number
                if (Pattern.matches("cpu[0-9]+", pathname.getName())) {
                    return true;
                }
                return false;
            }
        }

        try {
            //Get directory containing CPU info
            File dir = new File("/sys/devices/system/cpu/");
            //Filter to only list the devices we care about
            File[] files = dir.listFiles(new CpuFilter());
            //Return the number of cores (virtual CPU devices)
            return files.length;
        } catch (Exception e) {
            //Default to return 1 core
            return 1;
        }
    }

循環(huán)賽模式

循環(huán)賽模式(Round-Robin)是把數(shù)據(jù)分組, 按線程數(shù)分組, 每組9個, 一起發(fā)送處理. 這樣做, 可以減少Observable的創(chuàng)建, 節(jié)省系統(tǒng)資源, 但是會增加處理時間, 是空間和時間的綜合考慮.

int threadCt = Runtime.getRuntime().availableProcessors() + 1;
mTvGroupedValue.setText(String.valueOf("計算中(" + threadCt + "線程)"));

ExecutorService executor = Executors.newFixedThreadPool(threadCt);
Scheduler scheduler = Schedulers.from(executor);

final AtomicInteger batch = new AtomicInteger(0);

Observable.range(MIN, MAX)
        .groupBy(i -> batch.getAndIncrement() % threadCt)
        .flatMap(g -> g.observeOn(scheduler).map(this::intenseCalculation))
        .observeOn(AndroidSchedulers.mainThread())
        .subscribe(this::groupedTag);

這是Compute \ Max \ Group三種效果的時間對比, 可以發(fā)現(xiàn)Max的時間最優(yōu), 因?yàn)楸菴ompute多一個線程, 但是Group會更加節(jié)省資源一些. 根據(jù)所執(zhí)行的并行任務(wù)使用Rx吧.

效果

Demo

OK, that's all! Enjoy it!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市囊颅,隨后出現(xiàn)的幾起案子炉峰,更是在濱河造成了極大的恐慌畏妖,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,723評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件疼阔,死亡現(xiàn)場離奇詭異瓜客,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)竿开,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,485評論 2 382
  • 文/潘曉璐 我一進(jìn)店門谱仪,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人否彩,你說我怎么就攤上這事疯攒。” “怎么了列荔?”我有些...
    開封第一講書人閱讀 152,998評論 0 344
  • 文/不壞的土叔 我叫張陵敬尺,是天一觀的道長枚尼。 經(jīng)常有香客問我,道長砂吞,這世上最難降的妖魔是什么署恍? 我笑而不...
    開封第一講書人閱讀 55,323評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮蜻直,結(jié)果婚禮上盯质,老公的妹妹穿的比我還像新娘。我一直安慰自己概而,他們只是感情好呼巷,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,355評論 5 374
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著赎瑰,像睡著了一般王悍。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上餐曼,一...
    開封第一講書人閱讀 49,079評論 1 285
  • 那天压储,我揣著相機(jī)與錄音,去河邊找鬼源譬。 笑死集惋,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的瓶佳。 我是一名探鬼主播芋膘,決...
    沈念sama閱讀 38,389評論 3 400
  • 文/蒼蘭香墨 我猛地睜開眼鳞青,長吁一口氣:“原來是場噩夢啊……” “哼霸饲!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起臂拓,我...
    開封第一講書人閱讀 37,019評論 0 259
  • 序言:老撾萬榮一對情侶失蹤厚脉,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后胶惰,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體傻工,經(jīng)...
    沈念sama閱讀 43,519評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,971評論 2 325
  • 正文 我和宋清朗相戀三年孵滞,在試婚紗的時候發(fā)現(xiàn)自己被綠了中捆。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,100評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡坊饶,死狀恐怖泄伪,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情匿级,我是刑警寧澤蟋滴,帶...
    沈念sama閱讀 33,738評論 4 324
  • 正文 年R本政府宣布染厅,位于F島的核電站,受9級特大地震影響津函,放射性物質(zhì)發(fā)生泄漏肖粮。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,293評論 3 307
  • 文/蒙蒙 一尔苦、第九天 我趴在偏房一處隱蔽的房頂上張望涩馆。 院中可真熱鬧,春花似錦蕉堰、人聲如沸凌净。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,289評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽冰寻。三九已至,卻和暖如春皿渗,著一層夾襖步出監(jiān)牢的瞬間斩芭,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,517評論 1 262
  • 我被黑心中介騙來泰國打工乐疆, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留划乖,地道東北人。 一個月前我還...
    沈念sama閱讀 45,547評論 2 354
  • 正文 我出身青樓挤土,卻偏偏與公主長得像琴庵,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子仰美,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,834評論 2 345

推薦閱讀更多精彩內(nèi)容