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

时间:2012-04-04 15:59:37

标签: objective-c multithreading macos cocoa 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);

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

读者代码:

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);
}
}

可可部分:

-(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();
}

感谢您的帮助。

2 个答案:

答案 0 :(得分:7)

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

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

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

答案 1 :(得分:0)

最后!!!!!

当不把它放入队列时,它工作得很好。

ProgressController *pc = [[ProgressController alloc] init];
[pc showWindow:sender];
[pc.pi setUsesThreadedAnimation:YES];

Reader reader("test.txt");  
reader.setCallback(^(double progress) 
{
    [pc.pi setDoubleValue:progress]; 
    [pc.pi setNeedsDisplay:YES];
});

reader.read();

但是你为什么说“这很糟糕,因为它阻止了主线”? 因为基本上我的程序必须在执行任何其他操作之前等待文件被读取。 我在这里想到一些基本的优化吗?

非常感谢你的帮助。