abclinuxu.cz AbcLinuxu.cz itbiz.cz ITBiz.cz HDmag.cz HDmag.cz abcprace.cz AbcPráce.cz
AbcLinuxu hledá autory!
Inzerujte na AbcPráce.cz od 950 Kč
Rozšířené hledání
×
    dnes 02:44 | Nová verze

    Byla vydána listopadová aktualizace aneb nová verze 1.107 editoru zdrojových kódů Visual Studio Code (Wikipedie). Přehled novinek i s náhledy a videi v poznámkách k vydání. Ve verzi 1.107 vyjde také VSCodium, tj. komunitní sestavení Visual Studia Code bez telemetrie a licenčních podmínek Microsoftu.

    Ladislav Hagara | Komentářů: 0
    včera 19:22 | Zajímavý článek

    Pornhub zveřejnil podrobné statistiky za rok 2025. V části věnované zařízením a technologiím se lze dočíst, že 87 % přenášených dat směrovalo na telefony, 2 % na tablety a 11 % na desktopy. Operační systém Linux běžel na 6,3 % desktopů. O 22,4 % více než před rokem. Firefox má na desktopu 8,4 % podíl.

    Ladislav Hagara | Komentářů: 2
    včera 13:11 | Pozvánky

    Chcete vědět, co se odehrálo ve světě techniky za poslední měsíc? Nebo si popovídat o tom, co zrovna bastlíte? Pak dorazte na prosincovou Virtuální Bastlírnu s mikrofonem a kamerou, nalijte si něco k pití a ponořte se s strahovskými bastlíři do diskuze u virtuálního piva o technice i všem možném okolo. O čem budou tentokrát strahováci referovat? Téměř každý už si všiml významného zdražení RAM a SSD, jsou zde ale i příjemnější zprávy. Průša uvádí

    … více »
    bkralik | Komentářů: 0
    včera 12:55 | Bezpečnostní upozornění

    Národní úřad pro kybernetickou a informační bezpečnost (NÚKIB) podporuje vyjádření partnerů ze Spojeného království, kteří upozorňují na škodlivé aktivity společností Anxun Information Technology (též „I-S00N“) (pdf) a Beijing Integrity Technology (též „Integrity Tech“) působících v kyberprostoru a sídlících v Čínské lidové republice (ČLR). Tyto společnosti jsou součástí komplexního ekosystému soukromých subjektů v ČLR,

    … více »
    Ladislav Hagara | Komentářů: 8
    včera 04:55 | IT novinky

    Společnost Pebble představila (YouTube) prsten s tlačítkem a mikrofonem Pebble Index 01 pro rychlé nahrávání hlasových poznámek. Prsten lze předobjednat za 75 dolarů.

    Ladislav Hagara | Komentářů: 6
    včera 04:22 | IT novinky

    Společnost JetBrains v listopadu 2021 představila nové IDE s názvem Fleet. Tento týden oznámila jeho konec. Od 22. prosince 2025 již nebude možné Fleet stáhnout.

    Ladislav Hagara | Komentářů: 2
    9.12. 21:22 | Nová verze

    Byl vydán Mozilla Firefox 146.0. Přehled novinek v poznámkách k vydání a poznámkách k vydání pro vývojáře. Řešeny jsou rovněž bezpečnostní chyby. Nový Firefox 146 bude brzy k dispozici také na Flathubu a Snapcraftu.

    Ladislav Hagara | Komentářů: 0
    9.12. 17:00 | IT novinky

    Před rokem převzala Digitální a informační agentura (DIA) vlastnictví a provoz jednotné státní domény gov.cz. Nyní spustila samoobslužný portál, který umožňuje orgánům veřejné moci snadno registrovat nové domény státní správy pod doménu gov.cz nebo spravovat ty stávající. Proces nové registrace, který dříve trval 30 dní, se nyní zkrátil na několik minut.

    Ladislav Hagara | Komentářů: 7
    9.12. 11:33 | IT novinky

    IBM kupuje za 11 miliard USD (229,1 miliardy Kč) firmu Confluent zabývající se datovou infrastrukturou. Posílí tak svoji nabídku cloudových služeb a využije růstu poptávky po těchto službách, který je poháněný umělou inteligencí.

    Ladislav Hagara | Komentářů: 0
    9.12. 01:55 | IT novinky

    Nejvyšší správní soud (NSS) podruhé zrušil pokutu za únik zákaznických údajů z e-shopu Mall.cz. Incidentem se musí znovu zabývat Úřad pro ochranu osobních údajů (ÚOOÚ). Samotný únik ještě neznamená, že správce dat porušil svou povinnost zajistit jejich bezpečnost, plyne z rozsudku dočasně zpřístupněného na úřední desce. Úřad musí vždy posoudit, zda byla přijatá opatření přiměřená povaze rizik, stavu techniky a nákladům.

    Ladislav Hagara | Komentářů: 13
    Jaké řešení používáte k vývoji / práci?
     (34%)
     (47%)
     (19%)
     (17%)
     (22%)
     (15%)
     (24%)
     (16%)
     (18%)
    Celkem 451 hlasů
     Komentářů: 18, poslední 2.12. 18:34
    Rozcestník

    Java Native Interface: propojujeme Javu a C/C++ – 4

    22. 12. 2010 | Luboš Doležel | Programování | 3086×

    Doposud jsme si hráli jen s objekty, které nám předalo JNI. Je na čase nabýt samostatnosti a začít s třídami a instancemi pracovat samostatně. Také začneme řešit javovské výjimky.

    Obsah

    Načítáme třídy

    link

    Základem je funkce FindClass, která nám vrátí objekt jclass představující načtenou třídu. Pokud považujeme jclass za ekvivalent java.lang.Class, pak funkce FindClass odpovídá metodě Class.forName(). Volání je vcelku jednoduché:

    jclass jlInteger = env->FindClass("java/lang/Integer");
    

    Nejprve si řekneme, že jlInteger v naší ukázce je lokální reference, takže podle toho s ní musíme zacházet. Druhá věc je podoba, v jaké předáváme jméno třídy. V případě FindClass jméno třídy odpovídá v podstatě jménu souboru bez přípony .class. Mělo by stačit si pamatovat základní pravidla: místo teček jsou lomítka (podadresáře ve struktuře balíčků) a podtřídy se oddělují znakem $. Máme li například třídu Pokus v balíčku cz.abclinuxu a ta má v sobě enum Nesmysly, označení Nesmysly je cz/abclinuxu/Pokus$Nesmysly. První anonymní třída v druhé anonymní třídě ve třídě pokus by pak byla cz/abclinuxu/Pokus$2$1. Pokud tvoříte něco složitějšího a váháte, je nejsnazší se prostě podívat na jméno souboru.

    A teď, co s tím? Než se vrhneme na vytváření instancí, podíváme se na dodatečné metody pro práci s třídami. První věc je obdoba javovského operátoru instanceof:

    extern jstring myString; // Někde máme schovaný řetězec...
    
    jclass jlString = env->FindClass("java/lang/String");
    assert(env->IsInstanceOf(myString, jlString) == JNI_TRUE);
    

    Další věc je obdoba metody Class.isAssignableFrom(): IsAssignableFrom. Praktický rozdíl oproti IsInstanceOf spočívá v tom, že IsAssignableFrom pracuje nad dvěma třídami a nepotřebuje instanci.

    jclass jlObject = env->FindClass("java/lang/Object");
    jclass jlString = env->FindClass("java/lang/String");
    
    assert(env->IsAssignableFrom(jlString, jlObject) == JNI_TRUE);
    

    Pak ještě máme tyto funkce:

    • GetSuperclass – k předané třídě vrátí rodiče této třídy; pokud rodič neexistuje (například u java.lang.Object), vrátí NULL.
    • DefineClass – obdoba ClassLoader.defineClass(): načte předaný bajtkód jako novou třídu. To už je taková "vyšší dívčí".

    Vytváříme instance

    link

    K čistému vytvoření instance slouží funkce AllocObject – té dáme jclass a z druhé strany vyleze jobject. Obráceně funguje GetObjectClass, které si vezme jobject a vrátí odpovídající jclass (obdoba Object.getClass()). Dá se očekávat, že při vytváření objektu většinou chceme hned zavolat nějaký ten konstruktor. Od toho slouží trojice funkcí NewObject, NewObjectA a NewObjectV – ty vytvoří objekt a o zavolání konstruktoru se postarají.

    Signatury metod

    link

    Ale než začneme volat konstruktory, je tu ještě jedna funkce, kterou musíme znát: GetMethodID. GetMethodID budeme používat, ať už chceme volat konstruktor nebo obyčejnou metodu. GetMethodID přebírá jclass, název metody (v případě konstruktoru používáme <init>) a její signaturu. Vrací pak jMethodID (nebo NULL a výjimku navrch), které potřebujeme, kdykoliv k nějakému volání metody dochází.

    jmethodID GetMethodID(JNIEnv *env, jclass clazz, const char *name, const char *sig);

    Název metody name je docela jasná věc, se signaturou už je to horší. Takže se ještě naučíme tvořit signatury – taková signatura vypadá následovně: (argumenty)návratový_typ. Návratovým typem konstruktorů je void. K vytváření signatur využijeme tabulku, jež jsem ukázal ve druhém díle tohoto seriálu (tabulka primitivních datových typů, sloupeček identifikátor), kterou si pomyslně rozšíříme o tyto dvě speciality:

    • Identifikátor návratového typu void je V
    • Identifikátor instance objektu je Lidentifikátor_třídy;, přičemž identifikátorem třídy se myslí to samé, co jsme na začátku tohoto článku předávali funkci FindClass. Příklad pro java.lang.String: Ljava/lang/String;. Nezapomínejte na středník!

    Argumenty dáváme přímo za sebou, metoda void metoda(int i1, String s2, float f3) tedy bude mít signaturu (ILjava/lang/String;F)V. Následujícím způsobem tedy najdeme konstruktor Integer(String s) (třída java.lang.Integer):

    jclass jlInteger = env->FindClass("java/lang/Integer");
    jmethodID konstruktor = env->GetMethodID(jlInteger, "<init>", "(Ljava/lang/String;)V");
    

    Voláme konstruktor

    link

    Konečně už máme vše potřebné pro vytvoření objektu. Jaký je ale rozdíl mezi NewObject, NewObjectA a NewObjectV?

    • jobject NewObject(JNIEnv *env, jclass clazz, jmethodID methodID, ...)

      Funkce s variabilním počtem argumentů. Hned po ID metody tedy začneme předávat argumenty.

    • jobject NewObjectA(JNIEnv *env, jclass clazz, jmethodID methodID, jvalue *args)

      Přebírá argumenty jako odkaz na začátek pole objektů jvalue. jvalue je ve skutečnosti céčkový union, takže jeho použití vypadá následovně:

      jvalue args[3];
      
      args[0].i = 5;
      args[1].l = myString;
      args[2].f = 3.0f;
      
      // a můžeme předat args
      

      Využijeme při tvorbě různých wrapperů.

    • jobject NewObjectV(JNIEnv *env, jclass clazz, jmethodID methodID, va_list args)

      Funkce, která přebírá seznam variabilních argumentů :-) Pokud děláme nějaký jednoduchý wrapper kolem NewObject a naše funkce taktéž přebírá variabilní počet argumentů, bezpracně jej můžeme poslat dál v podobě va_list. Používání va_list už je mimo rozsah tohoto článku. Vztah mezi NewObjectV a NewObject je jako mezi vprintf a printf.

    Takže konečně vytvořme objekt:

    extern jstring myString;
    
    jclass jlInteger = env->FindClass("java/lang/Integer");
    jmethodID konstruktor = env->GetMethodID(jlInteger, "<init>", "(Ljava/lang/String;)V");
    jobject ii = env->NewObject(jlInteger, konstruktor, myString);
    

    A máme na světě svůj první vlastnoruční objekt, respektive lokální referenci na něj! Náš kód je obdobou tohoto javovského kódu:

    Integer ii = new Integer(myString);
    

    Na volání ostatních metod se podíváme příště, tentokrát nám ještě zbývá čas na výjimky.

    Výjimky

    link

    Seznam možných výjímek po volání JNI najdete v dokumentaci. Výjimky pak ještě může házet volání libovolné javovské metody. Protože je JNI céčkové API a C výjimky nemá, musíme výjimky ošetřovat ručně. Prvním krokem je vědět, jestli k výjimce došlo. Jednoduchá funkce ExceptionCheck nám v tom udělá jasno.

    
    jclass jlInteger = env->FindClass("java/lang/Integer");
    if (env->ExceptionCheck() == JNI_TRUE)
    {
    	cerr << "Pohroma!!!\n";
    	// ...?
    }
    

    Často ale chceme s výjimkou rovnou pracovat. V takovém případě můžeme ExceptionCheck rovnou přeskočit a přikročit k ExceptionOccurred, které nám vrátí nějakého potomka java.lang.Throwable. Nesmíme ale zapomínat JNI říci, že jsme se o výjimku postarali. Tentokrát jsme už chytřejší než v předchozí ukázce a zavoláme ExceptionClear. Před vymazáním pamatáky po výjimce ještě ale použijeme (ladící) funkci ExceptionDescribe, která na stderr vypíše popis výjimky a její zásobník volání. Ušetří nám to volání oblíbeného printStackTrace().

    jclass jlInteger = env->FindClass("java/lang/Integer");
    if (jthrowable ex = env->ExceptionOccurred())
    {
    	cerr << "Pohroma:\n";
    	env->ExceptionDescribe();
    	env->ExceptionClear();
    }
    

    Dobrá rada: jakmile máte daný jthrowable "v ruce", okamžitě (respektive hned po ExceptionDescribe, pokud jej používáte) volejte ExceptionClear. Může se stát, že v obsluze výjimky budete např. najednou chtít zavolat nějakou javovskou metodu, dáte takový kód před ExceptionClear a pak se budete divit, jaktože se to chová divně a kolabuje to. Pokud máte v JNI výjimku, musíte ji nejprve odstranit, než budete dělat něco dalšího!

    Házíme výjimky

    link

    Poslední věcí dnešního článku je výjimky nejen chytat, ale i házet. Na to existují dvě funkce: Throw a ThrowNew. První si od nás vezme libovolný jthrowable a po ukončení naší nativní funkce poletí javovským kódem výjimka. Druhá funkce usnadňuje házení výjimek tím, že výjimku pro nás ještě vytvoří. Předpokladem pro užití ThrowNew je to, že třída výjimky, kterou chceme hodit, má konstruktor, který má jediný argument řetězec s popisem chyby. V ostatních případech musíme konstrukci výjimky provádět sami. Příklad se ThrowNew:

    jclass jlException = env->FindClass("java/lang/Exception");
    env->ThrowNew(jlException, "Stala se strašlivá chyba!");
    

    Popis chyby předáváme jako řetězec v UTF-8. Tento kód je obdobou javovského throw new Exception("Stala se strašlivá chyba!");.

    Co bude příště

    link

    Příště začneme volat metody a pokud zbyde čas, tak se podíváme i na přístup ke členským proměnným.

           

    Hodnocení: 100 %

            špatnédobré        

    Nástroje: Tisk bez diskuse

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

    Komentáře

    Vložit další komentář

    22.12.2010 06:19 petr_p | skóre: 59 | blog: pb
    Rozbalit Rozbalit vše Re: Java Native Interface: propojujeme Javu a C/C++ – 4

    V řetězci signatury konstruktoru chybí první a třetí argument:

    jmethodID konstruktor = env->GetMethodID(jlClass, "<init>", "(Ljava/lang/String;)V");
    jobject ii = env->NewObject(jlInteger, konstruktor, 5, myString, 3.0f);
    Luboš Doležel (Doli) avatar 22.12.2010 11:04 Luboš Doležel (Doli) | skóre: 98 | blog: Doliho blog | Kladensko
    Rozbalit Rozbalit vše Re: Java Native Interface: propojujeme Javu a C/C++ – 4
    Díky za upozornění, tohle místo jsem přepisoval a přepsal jsem ho blbě. Původně mělo jít o new Integer(myString). Opravím to.
    ISSN 1214-1267   www.czech-server.cz
    © 1999-2015 Nitemedia s. r. o. Všechna práva vyhrazena.