私はAppleの開発に不慣れで(約3か月)、現在はすべてのプロジェクトのARCでXcode4.5.1を使用しています。私は自分のアプリ(また、 ARC )でmjpegビデオを表示しようとしていました。mjpegコーディングは独立したjpeg画像に基づいているため、NSImageWellを使用してmjpegフローのすべてのフレームを表示しました。しかし、フレームを切り替えると、メモリリークに気づきました。
.hファイル:
#import <Cocoa/Cocoa.h>
@interface AMCImagiaTestViewController : NSViewController
- (void) startVideo;
- (void) stopVideo;
@end
.mファイル:
#import "AMCImagiaTestViewController.h"
#include "AMCCommonLib.h" // My UNIX-like system call library. Used in the thread to call socket functions. Don't mind.
#import "TTImage.h" // This is NSImage. I only add an NSLog() in -dealloc method
/* forward declaration for a POSIX thread which receives jpeg data.
This thread receives jpeg images sent by a process of a Linux PC.
The socket content is simple enough: standard jpeg data started with
FFD8 and end with FFD9 */
void *threadReceiveFrameData(void *arg);
@interface AMCImagiaTestViewController ()
@property (nonatomic, assign) IBOutlet NSImageView *imageVideoFrame; // the outlet to the image in nib file
@property (nonatomic, assign) pthread_t hdlThread;
/* here are some properties used in thread but I can close or release if I kill it outside its thread loop */
@property (nonatomic, assign) int sockFd;
@property (nonatomic, assign) BOOL isThreadRunning;
@property (nonatomic, retain) NSData *dataForImage;
@property (nonatomic, retain) TTImage *tempImage;
@end
@implementation AMCImagiaTestViewController
... // some non-important methods
- (void) startVideo
{
[self stopVideo];
/* Start the receiving thread */
pthread_create(&_hdlThread, NULL, threadReceiveFrameData, (__bridge void*)self);
}
- (void) stopVideo
{
if (_isThreadRunning)
{
pthread_cancel(_hdlThread);
usleep(100000);
pthread_join(_hdlThread, NULL);
_hdlThread = 0;
if (_sockFd)
{
simpleSocketClose(_sockFd);
_sockFd = 0;
}
}
}
...
@end // end of @implementation AMCImagiaTestViewController
// Now comes to the POSIX thread
void *threadReceiveFrameData(void *arg)
{
@autoreleasepool {
AMCImagiaTestViewController *ctrl;
struct sockaddr_in sockMessage;
uint8_t buff[60001];
ssize_t dataLen;
memset(buff, 0, sizeof(buff));
ctrl = (__bridge AMCImagiaTestViewController*) arg;
pthread_setcancelstate(PTHREAD_CANCEL_ENABLE, NULL);
ctrl.isThreadRunning = YES;
/* socket(). create a UDP socket */
ctrl.sockFd = simpleSocketCreate_udp();
if (!ctrl.sockFd)
{
SYS_PERROR(); // a package of perror(). don't mind
goto THREAD_EXIT;
}
/* bind(). 22222 is the local port */
if (simpleSocketBind_udp(ctrl.sockFd, 22222) < 0)
{
SYS_PERROR();
goto THREAD_EXIT;
}
NSLog(@"pthread init OK");
do
{
/* my package of recvfrom().
The first NULL means receive from any IP.
The second NULL means never timeout (using the select() system call) */
dataLen = simpleSocketReceiveFrom_udp(ctrl.sockFd, NULL, &sockMessage, buff, sizeof(buff) - 1, NULL);
if (dataLen > 0)
{
//NSLog(@"received: %ld", dataLen);
ctrl.dataForImage = nil;
ctrl.dataForImage = [NSData dataWithBytes:buff
length:dataLen];
ctrl.tempImage = nil;
ctrl.tempImage = [[TTImage alloc] initWithData:ctrl.dataForImage];
/* MEMORY LEAK??? */
/* These two lines causes memory leaks.
I would explain it outside this code block */
[ctrl.imageVideoFrame setImage:nil];
[ctrl.imageVideoFrame setImage: ctrl.tempImage];
}
else
{
SYS_PERROR();
}
}
while (dataLen > 0);
THREAD_EXIT:
NSLog(@"pthread exit");
ctrl.isThreadRunning = NO;
if (ctrl.sockFd)
{
simpleSocketClose(ctrl.sockFd);
ctrl.sockFd = 0;
}
pthread_exit(NULL);
}
}
「MEMORYLEAK???」の説明 コードブロック内:
この行は、画像を表示するようにNSImageWellを設定します。jpegデータを送信し続けると、このアプリのメモリが増え続け、大幅に下がらないことに気づきました。[.. release]メソッドを作成しようとして「=nil」を追加しましたが、これは機能しませんでした。ただし、これらの2行をコメントアウトすると、メモリリークは発生しなくなりました(ビデオ機能も達成されませんでした...)
助言がありますか?または、この問題を調べるために非ARCモードに切り替える必要がありますか?どうもありがとう!!
新しい研究25年1月:私はARCなしで
別の簡単なプロジェクトを作成します
NSImageは、引き続きTTImage(TTはTesTを意味します)として次のようにサブクラス化されます。.m
ファイル:
@implementation TTImage
- (id)retain{
NSLog(@"%08x retain", (unsigned int)self);
return [super retain];
}
- (oneway void)release{
NSLog(@"release");
return [super release];
}
- (void)dealloc{
NSLog(@"%@ deallocated.", self);
return [super dealloc];
}
@end
これで、pthreadがAppデリゲートファイルで定義されました。
... // AppDelegate implementations
#define AMCRelease(obj) if(obj) {[(obj) release]; (obj) = nil;}
void *threadReceiveFrameData(void *arg)
{
@autoreleasepool
{
AMCAppDelegate *ctrl;
NSImage *lastImage;
struct sockaddr_in sockAddr;
uint8_t buff[60001];
ssize_t dataLen;
NSData *dataInThread = nil;
TTImage *imageInThread = nil;
ctrl = (AMCAppDelegate*)arg;
memset (buff, 0, sizeof(buff));
...
// the same initialization as before
NSLog(@"pthread init.");
do {
dataLen = simpleSocketReceiveFrom_udp(ctrl.sockFd, NULL, &sockAddr, buff, sizeof(buff) - 1, NULL);
if (dataLen > 0)
{
printf("\n");
AMCRelease(dataInThread);
dataInThread = [NSData dataWithBytes:buff length:dataLen];
NSLog(@"%d data rc: %ld", __LINE__, [dataInThread retainCount]); // Line 100
if (imageInThread)
{
NSLog(@"%d image pre rc: %ld", __LINE__, [imageInThread retainCount]); // Line 104
//lastImage = imageInThread;
}
AMCRelease(imageInThread);
imageInThread = [[TTImage alloc] initWithData:dataInThread];
NSLog(@"%d data rc: %ld", __LINE__, [dataInThread retainCount]); // Line 110
NSLog(@"%d image rc: %ld", __LINE__, [imageInThread retainCount]); // Line 111
//[ctrl.imageVideoFrame setImage:nil];
[ctrl.imageVideoFrame setImage:imageInThread];
NSLog(@"%d image rc: %ld", __LINE__, [imageInThread retainCount]); // Line 115
//NSLog(@"image last rc: %ld", [lastImage retainCount]);
}
else
{
SYS_PERROR();
}
} while (dataLen > 0);
THREAD_EXIT:
AMCRelease(dataInThread);
AMCRelease(imageInThread);
if (ctrl.sockFd)
{
simpleSocketClose(ctrl.sockFd);
ctrl.sockFd = 0;
}
pthread_exit(NULL);
}
}
次に、いくつかのjpegをアプリに送信します。出力は次のとおりです(すべての行のヘッダーは無視されます)。
pthread init.
100 data rc: 1
110 data rc: 2
111 image rc: 1
0067d890 retain <-- looks OK
115 image rc: 2
100 data rc: 1
104 image pre rc: 2
release
110 data rc: 2
111 image rc: 1
0067d890 retain <-- What??? Leaks???
0067d890 retain <-- What???
00632ba0 retain <-- looks OK
115 image rc: 2
100 data rc: 1
104 image pre rc: 2
release
110 data rc: 2
111 image rc: 1
00632ba0 retain <-- What???
00632ba0 retain <-- What???
0069d580 retain
115 image rc: 2