wxPHP线程应用程序崩溃与0xc0000005异常

时间:2015-07-04 15:22:46

标签: php multithreading nullpointerexception wxwidgets wxphp

我正在尝试使用wxPHP编写一个简单的应用程序,该应用程序使用exec('ping')命令从文件中ping IP。

我希望结果逐一显示在GUI中,因为它们会在后台进行操作,因此我在一个线程中执行exec('ping')命令。

但我得到了非常奇怪的错误和行为。问题是我无法重现错误。有时该程序完全正常。有时它会在线程中崩溃。有时它在发送事件时崩溃。等

以下是从命令行运行程序时出现的错误:

C:\Users\SH\Desktop>wxphp pinger.wxphp
PHP Notice:  Undefined property: wxHtmlWindow::$parent in Unknown on line 0
PHP Fatal error:  Call to a member function setPingResults() on null in Unknown
on line 0

C:\Users\SH\Desktop>wxphp pinger.wxphp
PHP Notice:  Undefined property: wxHtmlWindow::$parent in Unknown on line 0
PHP Fatal error:  Call to a member function setPingResults() on null in Unknown
on line 0

C:\Users\SH\Desktop>wxphp pinger.wxphp
PHP Notice:  Undefined property: wxHtmlWindow::$parent in Unknown on line 553649674
PHP Fatal error:  Call to a member function setPingResults() on null in Unknown
on line 553649674

这是我的代码:

<?php

if(!extension_loaded('wxwidgets'))
{
    dl('wxwidgets.' . PHP_SHLIB_SUFFIX);
}

define('EVT_PINGDONE',wxNewEventType());


class myPing extends wxThread
{
    function __construct($parent)
    {
        parent::__construct(wxTHREAD_JOINABLE);

        $this->parent =  $parent;
    }

    function Entry()
    {
        $addresses = file($this->parent->btnBrowse->GetPath(),FILE_IGNORE_NEW_LINES|FILE_SKIP_EMPTY_LINES);
        foreach($addresses as $k => $address){

            $results = exec("ping $address");

            $this->parent->setPingResults($results);

            $evt = new wxCommandEvent(EVT_PINGDONE);
            $this->parent->QueueEvent($evt);
        }

        $this->parent->onThreadDone();

        return;
    }
}

class mythFrame extends wxFrame {

    function __construct( $parent=null ){
        parent::__construct ( $parent, wxID_ANY, 'Pinger', wxDefaultPosition, new wxSize( 600,400 ), wxDEFAULT_FRAME_STYLE|wxTAB_TRAVERSAL );

        $this->SetSizeHints( wxDefaultSize, wxDefaultSize );

        $bSizer2 = new wxBoxSizer( wxHORIZONTAL );

        $this->button = new wxButton( $this, wxID_ANY, "Ping", wxDefaultPosition, new wxSize(300, 40), 0 );
        $bSizer2->Add($this->button, 0, wxALL|wxEXPAND, 5);

        $bSizer1 = new wxBoxSizer( wxVERTICAL );

        $this->btnBrowse = new wxFilePickerCtrl( $this, wxID_ANY, 'C:\Users\SH\Desktop\stuff\servers.txt', "Select a file", "*.*", wxDefaultPosition, wxDefaultSize, wxFLP_DEFAULT_STYLE );
        $bSizer1->Add( $this->btnBrowse, 0, wxALL|wxEXPAND|wxALIGN_CENTER_HORIZONTAL, 5 );

        $this->html = new wxHtmlWindow( $this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0 );
        $bSizer1->Add( $this->html, 1, wxALL|wxALIGN_CENTER_HORIZONTAL|wxEXPAND, 5 );

        $bSizer1->Add( $bSizer2, 0, wxALL|wxALIGN_CENTER_HORIZONTAL, 5 );

        $this->SetSizer( $bSizer1 );
        $this->Layout();

        $this->Centre( wxBOTH );

        // Connect Events
        $this->button->Connect( wxEVT_COMMAND_BUTTON_CLICKED, array($this, "hndlrButton") );
        $this->Connect(wxEVT_TIMER, array($this, "onTimer"));
        $this->Connect(EVT_PINGDONE, array($this, 'updateText'));

        $this->pingThread = new myPing($this);
        $this->m_timer = new wxTimer($this);
        $this->pingResults = '';

    }

    function updateText($event){

        $this->html->AppendToPage($this->pingResults.'<br>');
    }

    function onTimer(){
        if($this->threadDone == true){
            $this->m_timer->Stop();
            $this->button->Enable();
            $this->btnBrowse->Enable();
            $this->html->AppendToPage('<br>Finished.<hr>');

            while($this->pingThread->IsRunning()){}
            $this->pingThread->Delete();

            $this->pingThread = new myPing($this);
        }
    }

    function onThreadDone(){
        $this->threadDone = true;
    }

    function setPingResults($results){
        $this->pingResults = $results;
    }


    function hndlrButton( $event ){
        $this->threadDone = false;
        $this->m_timer->start(3000);
        $this->button->Disable();
        $this->btnBrowse->Disable();

        $this->pingThread->Create();
        $this->pingThread->Run();
    }

}

$myFrame = new mythFrame();
$myFrame->show();
wxEntry();

?>

这是我在Windows错误日志中遇到的错误:

Faulting application name: wxphp.exe, version: 5.6.9.0, time stamp: 0x55765568
Faulting module name: ntdll.dll, version: 6.1.7601.18247, time stamp: 0x521eaf24
Exception code: 0xc0000005
Fault offset: 0x0000000000052f86
Faulting process id: 0x3c4
Faulting application start time: 0x01d0b6675e88094f
Faulting application path: C:\Program Files\wxPHP\php\wxphp.exe
Faulting module path: C:\Windows\SYSTEM32\ntdll.dll
Report Id: 9cd6fe5e-225a-11e5-b316-f46d04f70903

错误模块根据我是否调用php函数或在线程exec()中调用Entry()而更改。但是异常代码始终是0xc0000005

错误一致的一个奇怪的事情是

之类的错误
Undefined property wxHtmlWindow::$parent 

如果你阅读了这些代码,你会发现根本就没有这样的电话。实际上$parent只能在线程中访问。我在线程中遇到了与其他函数类似的错误。似乎应用程序没有意识到它在线程中并且线程中的$this指向主框架对象。

我想我的代码或者wxPHP可能会出现严重错误。这个程序昨晚没有工作,然后今天早上开始工作。然后我添加了一个图标和一些背景颜色,它突然再次给我错误。

其他信息

  • 不能继续工作:

    Windows 7-64bit - wxphp-3.0.2.0-php5.6-x64.exe

    Windows 7-32bit - wxphp-3.0.2.0-php5.4-x86

  • 与wxPHP捆绑在一起的示例thread.wxphp应用程序 包裹也不起作用。
  • 但是我的代码和示例应用程序在Linux上运行良好(Linux Mint-32bit - php5-wxwidgets_3.0.2.0_i386)。
  • 当我从帖子中删除exec()QueueEvent()Entry()我没有收到任何错误。

有没有人知道这里发生了什么?

1 个答案:

答案 0 :(得分:1)

我是wxPHP的维护者,我建议使用PHP pthreads扩展来进行多线程处理,因为这个扩展可以正确处理PHP zendengine的线程。 wxThread类由wxPHP源生成器自动包装,可能需要进一步开发才能正常工作。