abclinuxu.cz AbcLinuxu.cz itbiz.cz ITBiz.cz HDmag.cz HDmag.cz abcprace.cz AbcPráce.cz
Inzerujte na AbcPráce.cz od 950 Kč
Rozšířené hledání
×
    dnes 14:22 | IT novinky

    VASA-1, výzkumný projekt Microsoftu. Na vstupu stačí jediná fotka a zvukový záznam. Na výstupu je dokonalá mluvící nebo zpívající hlava. Prý si technologii nechá jenom pro sebe. Žádné demo, API nebo placená služba. Zatím.

    Ladislav Hagara | Komentářů: 1
    dnes 04:44 | Nová verze

    Nová čísla časopisů od nakladatelství Raspberry Pi: MagPi 140 (pdf) a HackSpace 77 (pdf).

    Ladislav Hagara | Komentářů: 0
    dnes 01:00 | Nová verze

    ESPHome, tj. open source systém umožňující nastavovat zařízení s čipy ESP (i dalšími) pomocí konfiguračních souborů a připojit je do domácí automatizace, například do Home Assistantu, byl vydán ve verzi 2024.4.0.

    Ladislav Hagara | Komentářů: 0
    včera 22:11 | IT novinky Ladislav Hagara | Komentářů: 0
    včera 20:55 | Nová verze

    Neziskové průmyslové konsorcium Khronos Group vydalo verzi 1.1 specifikace OpenXR (Wikipedie), tj. standardu specifikujícího přístup k platformám a zařízením pro XR, tj. platformám a zařízením pro AR (rozšířenou realitu) a VR (virtuální realitu). Do základu se z rozšíření dostalo XR_EXT_local_floor. Společnost Collabora implementuje novou verzi specifikace do platformy Monado, tj. open source implementace OpenXR.

    Ladislav Hagara | Komentářů: 2
    včera 17:22 | Nová verze

    Byla vydána nová verze 0.38.0 multimediálního přehrávače mpv (Wikipedie) vycházejícího z přehrávačů MPlayer a mplayer2. Přehled novinek, změn a oprav na GitHubu. Požadován je FFmpeg 4.4 nebo novější a také libplacebo 6.338.2 nebo novější.

    Ladislav Hagara | Komentářů: 13
    včera 17:11 | Nová verze

    ClamAV (Wikipedie), tj. multiplatformní antivirový engine s otevřeným zdrojovým kódem pro detekci trojských koní, virů, malwaru a dalších škodlivých hrozeb, byl vydán ve verzích 1.3.1, 1.2.3 a 1.0.6. Ve verzi 1.3.1 je mimo jiné řešena bezpečnostní chyba CVE-2024-20380.

    Ladislav Hagara | Komentářů: 2
    včera 12:11 | IT novinky

    Digitální a informační agentura (DIA) oznámila (PDF, X a Facebook), že mobilní aplikace Portál občana je ode dneška oficiálně venku.

    Ladislav Hagara | Komentářů: 10
    včera 05:11 | Komunita

    #HACKUJBRNO 2024, byly zveřejněny výsledky a výstupy hackathonu města Brna nad otevřenými městskými daty, který se konal 13. a 14. dubna 2024.

    Ladislav Hagara | Komentářů: 2
    17.4. 17:55 | IT novinky

    Společnost Volla Systeme stojící za telefony Volla spustila na Kickstarteru kampaň na podporu tabletu Volla Tablet s Volla OS nebo Ubuntu Touch.

    Ladislav Hagara | Komentářů: 3
    KDE Plasma 6
     (68%)
     (10%)
     (2%)
     (20%)
    Celkem 563 hlasů
     Komentářů: 4, poslední 6.4. 15:51
    Rozcestník

    Dotaz: QT5 zaslani dat potomkovi; pomocí signálů

    6.1.2014 01:55 exo
    QT5 zaslani dat potomkovi; pomocí signálů
    Přečteno: 573×
    Ahoj. Můžete mi někdo poradit? Mám aplikaci s hlavním vláknem a vláknem1. V prvním vlákně běží GUI, ve druhém příkazový řádek. Pomocí GUI potřebuji ovlivňovat příkazy které bude příkazový řádek zpracovávat. Existuje nějaká možnost, jak poslat data od předka do potomka? V přdkovi mám signál haveData(QString), ten vyvolam ve chvili, kdy mam pro potomka nejaka data. V potomkovi mam pak slot který má daný signál zachytit a zpracovat. connect(parent, SIGNAL(haveData(QString)), this, SLOT(insertInput(QString))); Jenže ten na signál nereaguje. V konstruktoru dítěte mám na předka odkaz: explicit mcmd(QMainWindow *parent);

    Nenapadá mne už co dělám špatně :-(, nicméně je divné, že QTCreator všechny signály při zápisu předbízí/doplňujě, ale v tomto případě ne.

    Odpovědi

    Beda0 avatar 6.1.2014 08:19 Beda0 | skóre: 29
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů

    Zběžně to vypadá, že parent je QMainWindow, který nezná signál haveData(QString) -  QMainWindow nahradit třídou, která obsahuje ten signál (a dědí z QMainWindow)

    7.1.2014 02:30 exo
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Díky za tip... nakonec jsem se rozhodnul vytvořit samostatnou třídu na zaslání signálu, kterou jsem předal jak do konstruktoru mého cmd programu, tak do mainwindow.

    Signál se již předbízí, opět jsem ale bohužel dopadnul stejně :-( Signál je vyslán, ale cmd jej nezachytí... zkusím uvést příklad...
    //*********************
    (MainWindow.cpp)
    //*********************
    
    ...
    //Konstruktor
    MainWindow::MainWindow(QWidget *parent) :
        QMainWindow(parent),
        ui(new Ui::MainWindow),
        mycommandline(0)
    {
        ui->setupUi(this);
        connect(ui->pushButton, SIGNAL(clicked()), this, SLOT(startCommandline()));
        connect(ui->pushButton_2, SIGNAL(clicked()), this, SLOT(executeCommand()));
    }
    
    ...
    void MainWindow::startCommandline()
    {
    
        myiface = new minterface();
        mycommandline = new mcmd(myiface);
        mycommandline->start();                      // spustíme vlákno
    }
    
    void MainWindow::executeCommand()
    {
        myiface->emitSignal("ping 192.168.1.1");    //nefunguje - vola funkci emit v myiface
        emit myiface->haveData("ping 192.168.1.1"); //taky ne :-(
    }
    
    
    
    
    
    //*********************
    (mcmd.cpp)
    //*********************
    
    ...
    mcmd::mcmd(minterface *parent)
    {
        commandline = new QProcess();
        connect(parent, SIGNAL(haveData(QString)), this, SLOT(insertInput(QString)));
    }
    
    ...
    void mcmd::run()
    {
    
        commandline->start("cmd");
        commandline->waitForReadyRead();
    }
    
    void mcmd::insertInput(QString m_input)  //sem signal nikdy nedorazi :-(
    {
        input = m_input.toLatin1();
        commandline->write(input);
        commandline->write("\n");
        commandline->waitForReadyRead();
    }
    
    void mcmd::getOutput()                   //odeslani odtud do predka funguje bez problemu...
    {
        result = commandline->readAll();
        emit sendingData(QString(result));
    }
    
    
    7.1.2014 02:35 exo
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Omlouvám se všem. Jsem jelito... funkci insertInput(QString m_input) jsem zapoměl definovat jako slot :-(. Už to něco dělá :-). Akorát mi teď zamrzává GUI, což znamená, že ještě něco dělám špatně. Ale s tím už se snad nějak porvu...
    Beda0 avatar 7.1.2014 20:02 Beda0 | skóre: 29
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů

    Pokud je aplikace jednovláknová, tak pravděpodobně GUI čeká na zbytek aplikace. Sloty/signály samy nezaručí to, že když se vyšlou, tak program běží paralelně na dvou místech. Pokud je vyhozen signál, tak aplikace hledá kde se má přijmout - spustí se kód slotu a až po jeho doběhnutí se zase vrátí na místo odkud byl signál vyslán.

    7.1.2014 20:14 exo
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Nee. cmdčko se pouští v samostatném vlákně. Tím že jsem udělal samostatnou třídu na výměnu dat minterface() jsem si kupodivu pomohl. (abych si vzájemně neovlivňoval vlákna, vytvořil jsem minterface až v novém vláknu potomka a předal odkaz na něj zpět do předka - ale asi špatně)

    GUI se zasekne ve chvíli kdy CMDčku pomocí signálu pošlu další příkaz k vykonání (předchozí příkazy které mu dávám při inicializaci vlákna proběhnout krásně paralelně...).

    Pak začne aplikace hlásit něco o probému s thready, že se cmd pouští v jiném vlákně a celé to zatuhne... Ještě s tím budu chvílku bojovat a pokud se mi to nepodaří tak se vás tu zkusím znovu zeptat. Beztak to bude zase nějaká úplná bobost :-).
    8.1.2014 13:00 MadCatX
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    A máte ta vlákna udělaná dobře? V Qt nelze jen zdědit z třídy QThread, jako se to dělá třeba v Javě.
    8.1.2014 14:40 Ivan
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Ono to "jde" akorat se to "nedoporucuje". I kdyz vlastne dokumentace to porad jeste "doporucuje".

    Zkuste zadat do google "qthread you're doing it wrong" and "qthread You were not doing so wrong". Tyhle clanky nastini oba pristupy. Vami zminovany clanek jsem kdysi pouzil a a trochu rozsiril:

    toEventQuery toEventQueryWorker

    Pridal jsem zniceni vlakna i parent objektu a asynchronni preruseni cinnosti bg vlakna.
    8.1.2014 18:07 MadCatX
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Ano, v Qt je to dost nešťastně zmatené. Z různých zdrojů jsem pochopil, že subclassovat QThread a overridnout run() je možné tehdy, pokud nepotřebujete v daném vlákně zpracovávat eventy. Na jednorázovou akci (zápis do DB, složitější výpočet) to funguje dobře. Z toho, co ale tazatel píše soudím, že chce v GUI vlákně odpálit nějaký signál a příslušnou akci provést ve vlákně jiném. V takovém případě je vhodné použít QThread + Worker řešení. Další hezký zápisek rozebírající různé možnosti a jejich funkci je třeba zde (http://blog.debao.me/2013/08/how-to-use-qthread-in-the-right-way-part-1/)

    Pokud máte zájem, můžu postnout nějaký jednoduchý příklad, jak to lze vyřešit, až se dostanu domů...
    10.1.2014 01:13 exo
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Ahoj. no... jinak řečeno, potřebuju vytvořit něco jako CMD simulátor. Prostě z gui potřebuju spouštět a řídit cmd příkazy. V případě jednoduchého a neměnného zadání není problém. Například pusť ping na 192.168.1.1. Zinicializuji nové vlákno a v něm vše vytvořím a spustím. Pokud ale dojde na to že potřebuji zadání měnit, už narážím na problém. A to vůbec nemluvím o tom, když potřebuji spustit příkaz, ktarý v půlce potřebuje uživatelskou interakci. Například "net use" který se v průběhu dotáže na heslo, nebo kopírování, které narazí na "je cílové umístění soubor nebo adresář?"

    V té chvíli potřebuji mít a) nezakousnuté gui b) možnost poslat mému cmd další znaky aby mohl pokračovat. Ono a) by možná nevadilo, jenže já chci mít gui ve chvíli provádění příkazu živé. Očekávám že uživatel... než se vyhodnotí příkaz může dělat něco jiného.

    Takže cílem je spustit příkaz a pracovat na něčem jiném. Mezi tím se bude v nějakém textboxu vypisovat výstup. Jakmile se cmd zasatví, buď že už příkaz doběhnul a nebo čeká na doplnění informací, pošlu mu další znaky a opět pokračuju v práci.

    Teď na to nebudu mít zhruba týden čas určitě se ale k problému co nejdříve vrátím a dám info jak se mi podařilo pokročit.

    Jak popisuju, připadalo mi logické vytvořit třídu hlavního okna, třídu interface a třídu cmd. V hlavním okně standardně pracuju a v jiném thredu inicializuji cmd. Do cmd pak potřebuju prlběžně zasílat příkazy a z něj vracet odezvu bez toho, aby mi zamrzlo GUI. Toť mým hlavním cílem :-).

    Pokud by jste k tomu měli ještě nějaké rady či nápady určitě za ně budu rád. Ty odkazy ještě určitě nastuduju.

    10.1.2014 09:13 Ivan
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    hmm docela zajimava uloha. takovy veci jako alokace pts a emulace terminalu (vt100) uz mas poreseny? to mi prijde jako slozitejsi problem nez ty vlakna. Zkus se podivat na libqterminal

    skunkOS avatar 10.1.2014 09:31 skunkOS | skóre: 27 | blog: Tak nějak
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Zmatené není v podstatě nic, jen dokumentace na několika místech. Prakticky NIKDY se QThread nemusí a nemá dědit, worker řešení se naopak má použít skoro vždy.

    Máme tu taky QtConcurrent::run(...) metodu, která se tazateli může hodit.

    http://qt-project.org/doc/qt-4.8/qtconcurrent.html
    http://martinrotter.github.io
    Petr Bravenec avatar 11.1.2014 09:03 Petr Bravenec | skóre: 43 | blog: Bravenec
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Zmatené není v podstatě nic, jen dokumentace na několika místech.
    Souhlas
    Prakticky NIKDY se QThread nemusí dědit...
    Souhlas
    ...a nemá dědit
    Pověra
    worker řešení se naopak má použít skoro vždy
    Pověra

    Abych nebyl tak úsečný - worker-like řešení s děděním QThread může fungovat stejně dobře a mít některé jiné výhody, například to, že inicializace a start vlákna se přesune z hlavního vlákna, kde "zavazí", někam jinam, kde lépe ctí princip zapouzdřenosti objektu. Shrnul bych to spíš do jiné věty:

    Přetěžovat QThread::run() je hloupost.
    Petr Bravenec - Hobrasoft s.r.o.
    Petr Bravenec avatar 10.1.2014 12:45 Petr Bravenec | skóre: 43 | blog: Bravenec
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Pročetl jsem si tady tu diskusi. Možná se úplně pletu, když tak mě opravte.

    Vlákna si můžete dělat jak chcete - neexistuje "špatná" nebo "dobrá" cesta. Články jako "you are doing it wrong" a "you are not doing it not so wrong" vám řeknou každý něco jiného. Kdejaká *jednovláknová* aplikace v Qt používá při svém běhu spousty různých vláken a když se nad tím zamyslím, vychází mi z toho, že samotné Qt to dělá "wrong" (nebo "not so wrong"?) - a funguje to.

    Teď možná kecám, ale nebudu to ověřovat - pokud posíláte signál z vlákna A do slotu ve vlákně B, slot je překvapivě prováděn ve vlákně A. Ověřit si to můžete celkem jednoduše: qDebug() << QThread::currentThread(); . To může vést k hodně velikému zmatení. Řešením je posílat signál přes frontu zpráv, například:

    connect(this, SIGNAL(sig1()), vlakno, SLOT(slot1()), Qt::QueuedConnection);

    Vede to občas k trochu pracnějšímu zápisu: z GUI volám slot ve vlákně a až tam vyvolám signál přes frontu propojený s jiným slotem ve vlákně. Pokud to takhle nefunguje, budu muset svůj přístup přehodnotit a přepsat nějaké aplikace :-)

    S vlákny v Qt není moc problémů při vytváření a používání. Ovšem na hotové peklo můžete narazit, až budete chtít vlákno ukončit. Potíž může být i v tom, že u spousty Qt objektů ani netušíte, že si startují vlastní vlákna (QNetworkAccessManager).
    Petr Bravenec - Hobrasoft s.r.o.
    Petr Bravenec avatar 11.1.2014 07:21 Petr Bravenec | skóre: 43 | blog: Bravenec
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Pečlivě jsem si prostudoval výše uvedený odkaz: http://blog.debao.me/2013/08/how-to-use-qthread-in-the-right-way-part-1/

    Lépe shrnutou problematiku vláken v Qt jsem dosud nečetl. Budu muset svůj přístup přehodnotit - ne, že by můj zaběhlý způsob nefungoval, ale po prostudování a pochopení těch příkladů můžu dělat stejné věci s menší námahou.

    Díky za ten odkaz. Jdu přepisovat aplikace :-)
    Petr Bravenec - Hobrasoft s.r.o.
    Petr Bravenec avatar 11.1.2014 08:42 Petr Bravenec | skóre: 43 | blog: Bravenec
    Rozbalit Rozbalit vše Re: QT5 zaslani dat potomkovi; pomocí signálů
    Tak jsem si prostudoval ještě svoje aplikace - přepisování nemá smysl. Zjednoduším tím zápis vlákna za cenu složitějšího zápisu hlavního procesu. V hlavním procesu se mi implementační detaily vlákna nelíbí, takže to budu dělat i nadále "téměř wrong", dědit QThread a start vlákna a související procedury dělat v konstruktoru vlákna.

    Dovolil bych si jen jednu poznámku, která v odkazu uvedeném výše není vůbec zmíněná. Uvedené příklady vše řeší přes frontu zpráv, kde v zásadě k problémům nedochází, pokud teda nemám třeba ten nápad posílat z vlákna do vlákna například reference. Občas je potřeba volat z jiného vlákna nějakou metodu přímo, třeba:

    worker->stop();

    Zde je třeba pamatovat na to, že ač je worker v jiném vlákně s vlastní frontou zpráv, takové přímé volání frontu zpráv obchází a probíhá v hlavním vlákně - zde je nutné používat mutexy, případně v metodě stop() vyvolat signál propojený se slotem, který teprve provede žádanou akci.

    Problematika vláken je prostě složitější, než je popsáno v onom odkazu a v článcích typu "wrong" a "not so wrong", a je velmi jednoduché střelit se do nohy, když se s vlákny zachází neopatrně.
    Petr Bravenec - Hobrasoft s.r.o.

    Založit nové vláknoNahoru

    Tiskni Sdílej: Linkuj Jaggni to Vybrali.sme.sk Google Del.icio.us Facebook

    ISSN 1214-1267   www.czech-server.cz
    © 1999-2015 Nitemedia s. r. o. Všechna práva vyhrazena.