以下是我在NSTimer上的Objective-C类别,用于对NSTimers进行基于块的触发。我看不出它有什么问题,但我得到的是我传入schedule...
方法的块正在被释放,尽管我在其上调用了copy
。
我错过了什么?
typedef void(^NSTimerFiredBlock)(NSTimer *timer);
@implementation NSTimer (MyExtension)
+ (void)timerFired:(NSTimer *)timer
{
NSTimerFiredBlock blk = timer.userInfo;
if (blk != nil) {
blk(timer);
}
}
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)seconds
repeats:(BOOL)repeats
callback:(NSTimerFiredBlock)blk
{
return [NSTimer scheduledTimerWithTimeInterval:seconds
target:self
selector:@selector(timerFired:)
userInfo:[blk copy]
repeats:repeats];
}
@end
答案 0 :(得分:31)
我在http://orion98mc.blogspot.ca/2012/08/objective-c-blocks-for-fun.html
很棒的工作
NSTimer *timer = [NSTimer scheduledTimerWithTimeInterval:0.7
target:[NSBlockOperation blockOperationWithBlock:^{ /* do this! */ }]
selector:@selector(main)
userInfo:nil
repeats:NO
];
答案 1 :(得分:8)
你在github上有一个项目可以完成这项任务!
Cocoapod BlocksKit,允许您阻止一堆类......
#import "NSTimer+BlocksKit.h"
[NSTimer bk_scheduledTimerWithTimeInterval:1.0 block:^(NSTimer *time) {
// your code
} repeats:YES];
答案 2 :(得分:3)
以下是Mc.Stever's answer的Swift版本:
NSTimer.scheduledTimerWithTimeInterval(0.7, target: NSBlockOperation(block: {
/* do work */
}), selector: "main", userInfo: nil, repeats: false)
答案 3 :(得分:2)
你缺少的是,如果你传入的块在堆栈上,那么copy
将完全按照名称所说的 - 它将创建一个副本堆上的块。因此,您希望您传递的行为不会发生任何变化;没人会保留它。原件被解除分配后,副本将保持活动状态。
(除非:如果您不使用ARC,您还需要自动发布副本;您的意思是将非拥有引用传递为userInfo:
。否则副本将永远不会被释放)< / p>
答案 4 :(得分:0)
试试这个
typedef void(^NSTimerFiredBlock)(NSTimer *timer);
@interface NSObject (BlocksAdditions)
- (void)my_callBlock:(NSTimer *)timer;
@end
@implementation NSObject (BlocksAdditions)
- (void)my_callBlock:(NSTimer *)timer {
NSTimerFiredBlock block = (id)self;
block(timer);
}
@implementation NSTimer (MyExtension)
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)seconds
repeats:(BOOL)repeats
callback:(NSTimerFiredBlock)blk
{
blk = [[blk copy] autorelease];
return [NSTimer scheduledTimerWithTimeInterval:seconds
target:blk
selector:@selector(my_callBlock:)
userInfo:nil
repeats:repeats];
}
@end