Objective-C封裝std::priority_queue<>實現優先隊列

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

回憶下什么是優先隊列###

講到隊列一般人都知道,先進先出嘛,就和排隊買東西一樣,先來的人排在前面,買完就從隊列里出去了。那什么是優先隊列呢,假設我們生活在一個特別尊老愛幼的社會,每次排隊買東西的時候,都要按照年齡作為優先級比較的參照,年紀大的在最前面,年紀小的在其次,青壯年排在后面,老爺爺老奶奶買完了,才輪到小孩兒,小孩兒們正買著辣條呢呢,忽然又來了個老奶奶,大家于是很懂禮貌的讓老奶奶排到了第一個,等老奶奶搶完了超市里的土雞蛋離開隊伍后,才輪到剛剛正準備買辣條的小孩子繼續買。

用OC封裝std::priority_queue###

STL中的priority_queue是C++基于heap實現的優先隊列模板類,其魯棒性和性能已經經過了無數開發者的考驗。所以我們放心大膽的用吧。

首先定義一下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是優先隊列中所要管理的對象的基類,目前先實現以NSUInteger做為比較優先級的類型,有需要的可以擴展其他的基類出來。
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++類,用來重載()運算符,實現兩個QueueIntNodeObject對象的比較。

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

給priority_queue另外定義個名字,這個實在太長了。
下面就是實現PriorityQueue的幾個方法,每個方法對應的即是操作std::priority_queue的方法。當然別忘了再不使用std::priority_queue的時候delete掉,否則會有內存泄漏。

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

@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

然后再創建幾個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));

這段代碼執行后,會按年齡從大到小輸出person對象,達到了我們想要的按照年紀作為優先級參照輸出的效果。
并且,我在其中加入了輸出person對象引用計數的log,運行后發現,priority_queue在ARC下也很好的管理了計數,person對象在被push到queue中后,queue對其持強引用,引用計數加1,從queue中pop出來后,引用計數減1。所以我們依舊不用擔心如何在ARC中管理內存。

如果想實現更復雜的優先級的控制,只需要實現一個類似于QueueIntNodeObject的類和一個用于比較優先級的類即可。
是不是很簡單啦啦啦~~~

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

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,001評論 6 537
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,786評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,986評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,204評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,964評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,354評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,410評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,554評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,106評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,918評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,093評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,648評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,342評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,755評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,009評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,839評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,107評論 2 375

推薦閱讀更多精彩內容

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