2012年2月24日金曜日

Objective-CでTweetからハッシュタグを抽出する

「直近のtweetからハッシュタグを10件抽出して候補とする」って機能を実装する為に、tweetの文字列からハッシュタグの一覧を抽出するプログラムを作りますた。

1文字ずつパースしてはいないので実行速度はあまり速くないと思います。
- (NSArray *)hashTagFromString:(NSString *)_string
{
 NSMutableArray *result = [NSMutableArray array];
 NSInteger length = [_string length];
 NSRange validRange = {0, length};
 while (YES) {
  NSRange range = [_string rangeOfString:@"(^#[^ ]+ | #[^ ]+ | #[^ ]+$|^#[^ ]+$)" options:NSRegularExpressionSearch range:validRange];
  if (range.location == NSNotFound) {
   break;
  }
  NSString *hashTag = [_string substringWithRange:range];
  NSString *trimedHashTag = [hashTag stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];
  [result addObject:trimedHashTag];
  validRange.location = range.location + range.length;
  validRange.length = length - validRange.location;
 }
 
 return result;
 
}
※twitterAPIにてinclude_entities=trueをオプションしていするとハッシュタグ一覧を取得することができます。
このエントリーをはてなブックマークに追加

2012年2月22日水曜日

NSOperationのdeallocが呼ばれなくてハマりました

ひさしぶりにメモリ関連でハマりました。
NSOperationのdeallocが呼び出されなくてメモリーリークしまクリスティ。
setCompletionBlock内でoperation自身を参照したのが原因のようです。「自分で自分のリテインカウントをインクリメントして保持」てきなことが起こっているのかな?

// オペレーションがqueueから削除された時の処理設定
 [anOperation setCompletionBlock:^{
  // delegateに通知
  if (_delegate) {
   [_delegate netWorkConnector:self requestOperationCompleted:_request];
  }
  
  // 保持しているrequestの解放
  [operations removeObject:anOperation];
  if ([operations count] <= 0) {
   [self.delegateOperationDictionary removeObjectForKey:pointerOfDelegate];
  }
  GNRLoggerDebug(@"-- operation  completed --");
 }];
 

operationのアドレス情報を別変数に入れることで解決しました。
// オペレーションがqueueから削除された時の処理設定
 NSNumber *p = [NSNumber numberWithInteger:(NSInteger)anOperation];
 [anOperation setCompletionBlock:^{
  // delegateに通知
  if (_delegate) {
   [_delegate netWorkConnector:self requestOperationCompleted:_request];
  }
  
  // 保持しているrequestの解放
  [operations removeObject:(id)[p integerValue]];
  if ([operations count] <= 0) {
   [self.delegateOperationDictionary removeObjectForKey:pointerOfDelegate];
  }
  GNRLoggerDebug(@"-- operation  completed --");
 }];
 

まぁ原因の根本はBlocksの知識がしょっぱいからに違いない。。。
このエントリーをはてなブックマークに追加

2012年2月20日月曜日

No previous prototype for function 'functionname'ってwarningの対処

スタティックなメソッドなのにstaticが付いてませんでした。
なので
static int hoge;
こんな感じにstaticをつけてやることで解消しました。
このエントリーをはてなブックマークに追加

2012年2月19日日曜日

viewのx座標だけ、y座標だけ、widthだけ、heightだけ変更したい

UIViewのwidthだけ変更したいのに
@interface UIView (extend)
aView.frame = CGRectMake(aView.frame.origin.x, aView.frame.origin.y, 100, aView.frame.size.height);
@end
って書くのまんどくせー。
ってことで下記のようなUIViewの拡張メソッドを用意して使っています。

@interface UIView (extend)
- (void)setX:(NSInteger)x;
- (void)setY:(NSInteger)y;
- (void)setWidth:(NSInteger)width;
- (void)setHeight:(NSInteger)height;
- (void)setOrigin:(CGPoint)point;
- (void)setSize:(CGSize)size;
@end
 


#import "UIView+extend.h"


@implementation UIView (extend)



- (void)setX:(NSInteger)x

{

 self.frame = CGRectMake( x, self.frame.origin.y, self.frame.size.width, self.frame.size.height );

}

- (void)setY(NSInteger)y

{

 self.frame = CGRectMake( self.frame.origin.x, y, self.frame.size.width, self.frame.size.height );

}

- (void)setWidth:(NSInteger)width

{

 self.frame = CGRectMake( self.frame.origin.x, self.frame.origin.y, self, view.frame.size.height );

}

- (void)setHeight:(NSInteger)height

{

 self.frame = CGRectMake( view.frame.origin.x, view.frame.origin.y, view.frame.size.width, height );

}



- (void)setOrigin:(CGPoint)point {

 self.frame = CGRectMake(point.x, point.y, self.frame.size.width, self.frame.size.height);

}

- (void)setSize:(CGSize)size {

 self.frame = CGRectMake(self.frame.origin.x, self.frame.origin.y, size.width, size.height);

}



- (void)setViewSizeHalf:(UIView*)view {

 view.frame = CGRectMake(view.frame.origin.x, view.frame.origin.y, view.frame.size.width/2, view.frame.size.height/2);

}

@end
 

#import "Hoge.h"
#import "UIView+extend.h"

@implementation Hoge
- (id)init
{
 if (self = [super init]) {
  UIView *aView = [[[UIView alloc] init] autorelease];
  [aView setX:100];
 }
 return self;
}

@end
 
実績の無いソースコードなのでコピペじゃコンパイル通らないかもしれません。
このエントリーをはてなブックマークに追加

2012年2月17日金曜日

tabBarControllerに配置したnavigatioController内でviewControllerをpushした際にタブバーを非表示

タブターコントローラー(UITabBarController)に配置したナビゲーションコントローラー(UINavigatioController)内でviewControllerをpushした際にタブバー(tabbar)を非表示にするにはviewControllerのinitに下記コードを記述します。
-(id)init
{
 if( self=[super init] ){
  self.hidesBottomBarWhenPushed = YES;
 
 }
 return self;
}
 
ナビゲーションバー見たいにいつでも表示/非表示できないん?
このエントリーをはてなブックマークに追加

xcode4 + lionで自作テンプレートを配置する場所

ファインダーのタスクバーメニューから[移動]->[フォルダへ移動]を選択して
下記のパスを入力して[移動]をクリック
/Users/hoge/Library/Developer/Xcode

hogeの部分はアイコンがついてるフォルダ名です。
このエントリーをはてなブックマークに追加

2012年2月16日木曜日

iOS5でwent isFinished=YES without being started by the queue it is inの対処

iOS5以前はNSOperationをキャンセルする為にcancelメソッド内でisFinishedYESを設定すればよかったのですが、iOS5になってから「went isFinished=YES without being started by the queue it is in」っていうlogがコンソールに表示されるようになってしまいました。
どうやら
「実行されてないのに終了しちゃだめね♥」
といっているようです。これを回避する為に実装を下記のように修正しました。
 
[self setValue:[NSNumber numberWithBool:YES] forKey:@"isFinished"];
 
 
[self setValue:[NSNumber numberWithBool:YES] forKey:@"isExecuting"];
[self setValue:[NSNumber numberWithBool:NO] forKey:@"isExecuting"];
[self setValue:[NSNumber numberWithBool:YES] forKey:@"isFinished"];
 
これがあるべき姿かどうかはわかりませんが、キャンセルしてもlogに「went isFinished=YES without being started by the queue it is in」は表示されなくなれました。
めでたし、めでたし。。。なのか?
このエントリーをはてなブックマークに追加