Objective-C封裝std::priority_queue<>實現(xiàn)優(yōu)先隊列

原文地址:Objective-C封裝std::priority_queue<>實現(xiàn)優(yōu)先隊列
最近項目中需要用到優(yōu)先隊列,google了半天谋梭,發(fā)現(xiàn)Cocoa Foundation中竟然木有現(xiàn)成的好用的輪子可以拿來用。找了半天,也只有Core Foundation的CFBinaryHeap算是滿足需求孽椰,但是CFBinaryHeap需要自己管理釋放對象,而且不能實時更新heap中的值茎辐,再一看文檔中提供的方法叫挟,辣么多回調(diào)列在那里,做為一個前C++開發(fā)者昼激,我想我還不如用我熟悉一點的std::priority_queue來實現(xiàn)我的需求吧庇绽。

回憶下什么是優(yōu)先隊列###

講到隊列一般人都知道,先進先出嘛橙困,就和排隊買東西一樣瞧掺,先來的人排在前面,買完就從隊列里出去了凡傅。那什么是優(yōu)先隊列呢辟狈,假設(shè)我們生活在一個特別尊老愛幼的社會,每次排隊買東西的時候,都要按照年齡作為優(yōu)先級比較的參照哼转,年紀大的在最前面明未,年紀小的在其次,青壯年排在后面壹蔓,老爺爺老奶奶買完了趟妥,才輪到小孩兒,小孩兒們正買著辣條呢呢佣蓉,忽然又來了個老奶奶披摄,大家于是很懂禮貌的讓老奶奶排到了第一個,等老奶奶搶完了超市里的土雞蛋離開隊伍后偏螺,才輪到剛剛正準備買辣條的小孩子繼續(xù)買行疏。

用OC封裝std::priority_queue###

STL中的priority_queue是C++基于heap實現(xiàn)的優(yōu)先隊列模板類,其魯棒性和性能已經(jīng)經(jīng)過了無數(shù)開發(fā)者的考驗套像。所以我們放心大膽的用吧酿联。

首先定義一下std::priority_queue<>的包裝類:

----PriorityQueue.h----
@interface QueueIntNodeObject : NSObject

@property (nonatomic, assign) NSUInteger compareValue;

@end

@interface PriorityQueue : NSObject

@property (nonatomic, readonly) QueueIntNodeObject *topObject;

@property (nonatomic, readonly) NSUInteger count;

- (void)pushObject:(QueueNodeObject *)myObject;

- (void)popObject;

- (void)popAllObjects;

QueueIntNodeObject是優(yōu)先隊列中所要管理的對象的基類,目前先實現(xiàn)以NSUInteger做為比較優(yōu)先級的類型夺巩,有需要的可以擴展其他的基類出來贞让。
PriorityQueue是用來包裝std::priority_queue的wrapper。定義幾個常用的屬性和方法柳譬。

----PriorityQueue.mm----
#import "PriorityQueue.h"
#include <queue>

class QueueCompare {
public:
    bool operator()(QueueIntNodeObject *l, QueueIntNodeObject *r) const {
        if (l.compareValue < r.compareValue) {
            return true;
        }else {
            return false;
        }
    }
};

typedef std::priority_queue<QueueIntNodeObject *, std::vector<QueueIntNodeObject *>, QueueCompare> Queue;

#pragma mark - QueueIntNodeObject
@implementation QueueIntNodeObject
@end

#pragma mark - PriorityQueue
@interface PriorityQueue ()
@property (nonatomic) Queue *priority_queue;
@end

@implementation PriorityQueue
- (instancetype)init {
      self = [super init];
      if (self) {
          _priority_queue = new Queue();
      }
      return self;
}

- (void)dealloc {
      delete _priority_queue;
      _priority_queue = NULL;
}

- (QueueIntNodeObject *)topObject {
       return !self.priority_queue->empty() ? self.priority_queue->top() : nil;
}

- (NSUInteger)count {
       return (NSUInteger)self.priority_queue->size();
}

- (void)popObject {
       if (!self.priority_queue->empty()) {
            self.priority_queue->pop();
        }
}

- (void)pushObject:(QueueIntNodeObject *)myObject {
        self.priority_queue->push(myObject);
}

- (void)popAllObjects {
        if (!self.priority_queue->empty()) {
              delete _priority_queue;
              _priority_queue = new Queue();
        }
}
@end

QueueCompare定義一個C++類喳张,用來重載()運算符,實現(xiàn)兩個QueueIntNodeObject對象的比較美澳。

typedef std::priority_queue<QueueIntNodeObject *, std::vector<QueueIntNodeObject *>, QueueCompare> Queue;

給priority_queue另外定義個名字销部,這個實在太長了。
下面就是實現(xiàn)PriorityQueue的幾個方法制跟,每個方法對應(yīng)的即是操作std::priority_queue的方法舅桩。當然別忘了再不使用std::priority_queue的時候delete掉,否則會有內(nèi)存泄漏雨膨。

我們再來看段實例代碼擂涛,以前面舉的排隊的例子,先定義一個排隊的人的對象聊记,對象有兩個屬性撒妈,名稱和年紀:

@interface Person : QueueIntNodeObject

@property (nonatomic, copy) NSString *name;

- (instancetype)initWithName:(NSString *)name age:(NSUInteger)age;

@end
@implementation SeatInfo
- (instancetype)initWithName:(NSString *)name age:(NSUInteger)age {
        if (self = [super init]) {
            self.name = name;
            self.compareValue = age;
        }
        return self;
}
@end

然后再創(chuàng)建幾個Person對象,放到隊列管理去排监。

Person *s1 = [[Person alloc] initWithName:@"賈母" age:70];
NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)s1));
Person *s2 = [[Person alloc] initWithName:@"寶玉" age:16];
Person *s3 = [[Person alloc] initWithName:@"黛玉" age:15];
Person *s4 = [[Person alloc] initWithName:@"寶釵" age:17];
Person *s5 = [[Person alloc] initWithName:@"妙玉" age:18];
Person *s6 = [[Person alloc] initWithName:@"賈政" age:40];
Person *s7 = [[Person alloc] initWithName:@"鳳姐兒" age:20];
Person *s8 = [[Person alloc] initWithName:@"平兒" age:19];

PriorityQueue *queue = [[PriorityQueue alloc] init];
[queue pushObject:s1];
[queue pushObject:s2];
[queue pushObject:s3];
[queue pushObject:s4];
[queue pushObject:s5];
[queue pushObject:s6];
[queue pushObject:s7];
[queue pushObject:s8];
NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)s1));

while (queue.count) {
      Person *person = (Person *)[queue topObject];
      NSLog(@"%@", person.name);
      [queue popObject];
}
NSLog(@"Retain count is %ld", CFGetRetainCount((__bridge CFTypeRef)s1));

這段代碼執(zhí)行后狰右,會按年齡從大到小輸出person對象,達到了我們想要的按照年紀作為優(yōu)先級參照輸出的效果舆床。
并且棋蚌,我在其中加入了輸出person對象引用計數(shù)的log,運行后發(fā)現(xiàn),priority_queue在ARC下也很好的管理了計數(shù),person對象在被push到queue中后附鸽,queue對其持強引用,引用計數(shù)加1瞒瘸,從queue中pop出來后坷备,引用計數(shù)減1。所以我們依舊不用擔心如何在ARC中管理內(nèi)存情臭。

如果想實現(xiàn)更復雜的優(yōu)先級的控制省撑,只需要實現(xiàn)一個類似于QueueIntNodeObject的類和一個用于比較優(yōu)先級的類即可。
是不是很簡單啦啦啦~~~

文中代碼都提交到 github 啦:https://github.com/LicoC/PriorityQueue 俯在, 歡迎star ??

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末竟秫,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子跷乐,更是在濱河造成了極大的恐慌肥败,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,941評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件愕提,死亡現(xiàn)場離奇詭異馒稍,居然都是意外死亡,警方通過查閱死者的電腦和手機浅侨,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評論 3 395
  • 文/潘曉璐 我一進店門纽谒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人如输,你說我怎么就攤上這事鼓黔。” “怎么了不见?”我有些...
    開封第一講書人閱讀 165,345評論 0 356
  • 文/不壞的土叔 我叫張陵澳化,是天一觀的道長。 經(jīng)常有香客問我脖祈,道長肆捕,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,851評論 1 295
  • 正文 為了忘掉前任盖高,我火速辦了婚禮慎陵,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘喻奥。我一直安慰自己席纽,他們只是感情好,可當我...
    茶點故事閱讀 67,868評論 6 392
  • 文/花漫 我一把揭開白布撞蚕。 她就那樣靜靜地躺著润梯,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上纺铭,一...
    開封第一講書人閱讀 51,688評論 1 305
  • 那天寇钉,我揣著相機與錄音,去河邊找鬼舶赔。 笑死扫倡,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的竟纳。 我是一名探鬼主播撵溃,決...
    沈念sama閱讀 40,414評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼锥累!你這毒婦竟也來了缘挑?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,319評論 0 276
  • 序言:老撾萬榮一對情侶失蹤桶略,失蹤者是張志新(化名)和其女友劉穎语淘,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體删性,經(jīng)...
    沈念sama閱讀 45,775評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡亏娜,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了蹬挺。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片维贺。...
    茶點故事閱讀 40,096評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖巴帮,靈堂內(nèi)的尸體忽然破棺而出溯泣,到底是詐尸還是另有隱情,我是刑警寧澤榕茧,帶...
    沈念sama閱讀 35,789評論 5 346
  • 正文 年R本政府宣布垃沦,位于F島的核電站,受9級特大地震影響用押,放射性物質(zhì)發(fā)生泄漏肢簿。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,437評論 3 331
  • 文/蒙蒙 一蜻拨、第九天 我趴在偏房一處隱蔽的房頂上張望池充。 院中可真熱鬧,春花似錦缎讼、人聲如沸收夸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽卧惜。三九已至厘灼,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間咽瓷,已是汗流浹背设凹。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留茅姜,地道東北人围来。 一個月前我還...
    沈念sama閱讀 48,308評論 3 372
  • 正文 我出身青樓,卻偏偏與公主長得像匈睁,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子桶错,可洞房花燭夜當晚...
    茶點故事閱讀 45,037評論 2 355

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

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理航唆,服務(wù)發(fā)現(xiàn),斷路器院刁,智...
    卡卡羅2017閱讀 134,661評論 18 139
  • 3.1 Grand Central Dispatch(GCD)概要 3.1.1 什么是CGD Grand Cent...
    SkyMing一C閱讀 1,630評論 0 22
  • java筆記第一天 == 和 equals ==比較的比較的是兩個變量的值是否相等糯钙,對于引用型變量表示的是兩個變量...
    jmychou閱讀 1,501評論 0 3
  • 1.objective-c常見面試題:1、**OC **語言的基本特點OC 語言是 C 語言的一個超集,只是在 C...
    LZM輪回閱讀 965評論 0 3
  • 下面是我最近兩年學習OC中的一些基礎(chǔ)知識退腥,對于學習OC基礎(chǔ)知識的人可能有些幫助任岸,拿出來分享一下,還是那句話不喜勿噴...
    小小趙紙農(nóng)閱讀 2,594評論 1 7