在Cocoa中读取带有进度条的文件

ben*_*ben 2 macos cocoa multithreading objective-c grand-central-dispatch

我想创建一个显示文件读取状态的进度条.我使用包含变量_progress的C++类Reader读取文件.

如何在没有在Reader类中编写任何ObjC代码的情况下告诉Cocoa使用reader._progress的值更新进度条?

任何帮助,将不胜感激.

ProgressController *pc = [[ProgressController alloc] init];
[pc showWindow:sender];


// Create the block that we wish to run on a different thread
void (^progressBlock)(void);
progressBlock = ^{
    [pc.pi setDoubleValue:0.0];
    [pc.pi startAnimation:sender];

    Reader reader("/path/to/myfile.txt");
    reader.read();

    while (reader._progress < 100.)
    {
        dispatch_async(dispatch_get_main_queue(), ^{
            [pc.pi setDoubleValue:reader._progress];
            [pc.pi setNeedsDisplay:YES];
        });
    }
}; // end of progressBlock

// Finally, run the block on a different thread
dispatch_queue_t queue = dispatch_get_global_queue(0, 0);
dispatch_async(queue, progressBlock);
Run Code Online (Sandbox Code Playgroud)

所以这是我的第二次尝试.

读者代码:

class PDBReader
{
public:
Reader(const char *filename);
Reader(string filename);
~Reader();

int read();

string _filename;
float _progress;

void setCallback(void (^cb)(double))
{
    if (_cb)
    {
        Block_release(_cb);
        _cb = Block_copy(cb);
    }
}
void (^_cb)(double);

protected:
private:
};



int Reader::read()
{
string buffer;
unsigned atomid = 0;
ifstream file;
file.open(_filename.c_str(), ifstream::in);

if (!file.is_open())
{
    return IOERROR;
}

file.seekg(0, ios_base::end);
float eof = (float) file.tellg();
file.seekg(0, ios_base::beg);

while (getline(file, buffer))
{
    _progress = (float) file.tellg() / eof * 100.;
    if (_cb)
    {
        _cb(_progress);
    }
        // some more parsing here...
    }
file.close();
return SUCCESS;
}

PDBReader::~PDBReader()
{
if (_cb)
{
    Block_release(_cb);
}
}
Run Code Online (Sandbox Code Playgroud)

和可可部分:

-(IBAction) test:(id) sender
{
ProgressController *pc = [[ProgressController alloc] init];
[pc showWindow:sender];

Reader reader("test.txt");

reader.setCallback(^(double progress) 
{
    dispatch_async(dispatch_get_main_queue(), ^{
        [pc.pi setDoubleValue:progress]; 
        [pc.pi setNeedsDisplay:YES];
    });
});

reader.read();
}
Run Code Online (Sandbox Code Playgroud)

谢谢你的帮助.

Ken*_*ses 7

仅仅因为你不想让Reader包括Objective-C代码并不意味着你只能从外面观察它.它可以通过传入的函数指针调用C函数.它可以使用更通用的仿函数(函数对象)机制.它甚至可以占用一块.

你绝对不想做那个while (reader._progress < 100.)循环.这是一个繁忙的循环.它将以尽可能快的方式快速更新进度.它将以100%的利用率固定CPU核心.实际上,它可能会将任务排队到主调度队列,而不是运行它们.

您只想在Reader更新其_progress成员时更新进度指示器,这需要Reader类的某种合作.