2012-07-19 10 views
8

Macでは、FSEventsウォッチャーを作成する必要があります。私はC++に慣れており、Objective-CではなくC++コードでFSEventsの通知を受け取る方法があります。いくつかのサンプルコードがありますか?それにインクルードする必要があるライブラリがありますか?FSEvents C++の例

私はすでにこのページに入っています。 http://developer.apple.com/library/mac/#featuredarticles/FileSystemEvents/_index.html

しかし、唯一のObjective Cがあるようです、私はそれ

答えて

13

のCPPのバージョンを持つことができますはい、それはあなたがカーネルキューのためになるはずですCで可能です。詳細はCHで見つけることができます

#include <errno.h>  // for errno 
#include <fcntl.h>  // for O_RDONLY 
#include <stdio.h>  // for fprintf() 
#include <stdlib.h>  // for EXIT_SUCCESS 
#include <string.h>  // for strerror() 
#include <sys/event.h> // for kqueue() etc. 
#include <unistd.h>  // for close() 

int main (int argc, const char *argv[]) 
{ 
    int kq = kqueue(); 
    // dir name is in argv[1], NO checks for errors here 
    int dirfd = open (argv[1], O_RDONLY); 

    struct kevent direvent; 
    EV_SET (&direvent, dirfd, EVFILT_VNODE, EV_ADD | EV_CLEAR | EV_ENABLE, 
      NOTE_WRITE, 0, (void *)dirname); 

    kevent(kq, &direvent, 1, NULL, 0, NULL); 

    // Register interest in SIGINT with the queue. The user data 
    // is NULL, which is how we'll differentiate between 
    // a directory-modification event and a SIGINT-received event. 
    struct kevent sigevent; 
    EV_SET (&sigevent, SIGINT, EVFILT_SIGNAL, EV_ADD | EV_ENABLE, 0, 0, NULL); 
    // kqueue event handling happens after the legacy API, so make 
    // sure it doesn eat the signal before the kqueue can see it. 
    signal (SIGINT, SIG_IGN); 

    // Register the signal event. 
    kevent(kq, &sigevent, 1, NULL, 0, NULL); 

    while (1) { 
     // camp on kevent() until something interesting happens 
     struct kevent change; 
     if (kevent(kq, NULL, 0, &change, 1, NULL) == -1) { exit(1); } 
     // The signal event has NULL in the user data. Check for that first. 
     if (change.udata == NULL) { 
      break; 
     } else { 
     // udata is non-null, so it's the name of the directory 
     printf ("%s\n", (char*)change.udata); 
     } 
    } 
    close (kq); 
    return 0; 
} 

は、ここでディレクトリを見に小さなサンプルです。 Mark Dalrympleによる「Advanced Mac OSX Programming」の16(kqueuesとFSEvents)追加情報は、* BSDドキュメンテーションのkqueuesにあります。

または、このAPIをFSEventsから使用してください(主にCベース)。

純粋なCコールバックでFSEventsイベントストリームを作成する。 -getCFRunLoopを使用してNSRunLoopからCFRunLoopを得る:

その後はい、ここであなたはおそらく実行ループハンドルを取得するためのObj-Cのラインを使用する必要があります

void FSEventStreamScheduleWithRunLoop (FSEventStreamRef streamRef, 
            CFRunLoopRef runLoop, 
            CFStringRef runLoopMode); 

を使用して実行ループに、このイベント・ストリームを添付

CFRunLoop* loopRef = [[NSRunLoop currentRunLoop] getCFRunLoop]; 

やイベントを開始します

純粋なCコール

CFRunLoop* loopRef = CFRunLoopGetCurrent(); 
を使用これで実行ループから

Boolean FSEventStreamStart (FSEventStreamRef streamRef); 

ストップで流れ

void FSEventStreamStop (FSEventStreamRef streamRef); 

と、イベントストリームし、それスケジュール解除:

void FSEventStreamInvalidate (FSEventStreamRef streamRef); 

void FSEventStreamUnscheduleFromRunLoop (FSEventStreamRef streamRef, 
            CFRunLoopRef runLoop, 
            CFStringRef runLoopMode); 

は、ストリーム(クリーンアップ)を無効

これはあなたにタート。

+0

私の実験では、kqueueはFSEventsと機能的に同等ではありません。 CFRunLoopビットの概要をお寄せいただきありがとうございます! – berkus