Oracle C++ Call Interface – 3. Niečo málo o Statementoch
 x   TIP: Přetáhni ikonu na hlavní panel pro připnutí webu
Reklama
Reklama

Oracle C++ Call Interface – 3. Niečo málo o StatementochOracle C++ Call Interface – 3. Niečo málo o Statementoch

 

Oracle C++ Call Interface – 3. Niečo málo o Statementoch

Google       Google       15. 10. 2007       10 720×

Naposledy keď som sa s vami lúčil, spomínal som akési statementy. V dnešnom dieli teda načrtnem, čo statement je, ako pomocou statementu vykonávať SQL command, ako na transakcie a zavŕšim to tým, že vám ukážem, ako na bindovanie premenných.

Reklama
Reklama

Skôr ako načrtnem význam Statement triedy, tak si urobíme jednu drobnú prípravu. Na naše experimentálne účely v tomto dieli budeme potrebovať jeden kus tabuľky a jeden kus storovanej procedúry. V prvom dieli som spomínal, aby ste si preklikali web rozhranie. Dúfam, že tieto pojmy nebude treba nijak obzvlášť predstavovať. Ešte vám poradím si vytvoriť pre naše študijné experimentovanie nového užívateľa, napríklad OCCIUSER. Je to čisto z dôvodu prehľadnosti. Totižto užívateľ SYSTEM má vo svojej schéme už množstvo tabuliek a pohlaď cez object browser je neprehľadný. Počítam s tým, že viete čo robiť s nasledujúcim výpisom a že ho nemusím nejak rozoberať. Predsa len o tento článok väčšinou zavadia ľudia, ktorý majú aspoň základy relačných databáz.

Najprv si vytvoríme tabuľku, s ktorou počas článku budeme pracovať:


CREATE TABLE osoby (
   meno       VARCHAR2(20),
   priezvisko VARCHAR2(20),
   vek        NUMBER(6)
);

Vytvoríme si procedúru, ktorá bude mať jeden vstupný a druhy výstupný parameter. Tuto procedúru použijeme v závere tohto článku.


CREATE OR REPLACE PROCEDURE myProc ( p1 IN NUMBER, p2 OUT NUMBER ) AS
BEGIN
   p2  := 2*p1;
END;

INSERT, UPDATE, DELETE

Teraz si už môžeme predstaviť statement. Už vieme, ako sa na DB pripojiť. Nevieme však, ako nad DB vykonávať SQL príkazy. Práve na tento účel slúži trieda Statement. Inštancia triedy je vytváraná z aktuálneho Connection-u. Ako príklad si uvedieme INSERT. Do tabuľky OSOBY chceme zapísať dáta.

main.cpp
#include <iostream>
#include <occi.h>

using namespace std;
using namespace oracle::occi;

#define USER  "OCCIUSER"
#define PASS  "****"
#define TNS   "XE"

int main()
{
   Environment* env;
   Connection*  con;
   Statement*   stm;

   try {
      env = Environment::createEnvironment();
      con = env->createConnection(USER, PASS, TNS);

      stm = con->createStatement();
      stm->executeUpdate("INSERT INTO OSOBY(MENO, PRIEZVISKO, VEK) VALUES( 'Jozef', 'Taraba', 25)");

      env->terminateConnection(con);
      Environment::terminateEnvironment(env);

   } catch (SQLException& e) {
      cout << "ORACLE ERROR:" << e.getErrorCode() << " - " << e.getMessage() << endl;
   }

   return 0;
}

Tak ako som spomínal, inštancia Statementu je vytváraná z aktuálneho spojenia pomocou createStatement() funkcie. Potom je volaná jednoduchá funkcia executeUpdate(), ktorej sa predáva reťazec obsahujúci SQL príkaz. Funkcia executeUpdate() je šitá pre INSERT, UPDATE, DELETE a samozrejme aj na vykonávanie skriptov medzi blokmi BEGIN a END. Okrem tejto funkcie ďalej existuje executeQuery(), ktorá vracia inštanciu triedy ResultSet. Tu budeme neskôr použivať pri SELECToch. Ďalšia funkcia je execute(). Tá vracia Status, v akom sa Statement nachádza. Totižto Statement okrem toho, že slúži na vykonávanie SQL dotazov, nadobúda aj určité stavy, ale na to je ešte skoro.

COMMIT a ROLLBACK

Databázistom určite neuniklo, že v našom prvom príklade sa nikde nepoužil COMMIT a predsa k nemu došlo. Pri vzniku Statement triedy totižto ostáva nastavená možnosť auto commit. Ten zabezpečí, že po každom volaní executeUpdate() sa vykoná COMMIT. To nie je veľmi vhodné v prípadoch, kedy potrebujeme mať kontrolu nad transakciami. Táto vlastnosť sa dá vypnúť a zapnúť volaním funkcie setAutoCommit(). Teraz budeme musieť vykonávať COMMIT volaním funkcie commit(), ktorá sa však nevzťahuje na konkrétny Statement, ale na celý Connection. main.cpp

#include <iostream>
#include <occi.h>

using namespace std;
using namespace oracle::occi;

#define USER  "OCCIUSER"
#define PASS  "****"
#define TNS   "XE"

int main()
{
   Environment* env;
   Connection*  con;
   Statement*   stm;

   try {
      env = Environment::createEnvironment();
      con = env->createConnection(USER, PASS, TNS);

      stm = con->createStatement();
      stm->setAutoCommit(FALSE);

      stm->executeUpdate("INSERT INTO OSOBY(MENO, PRIEZVISKO, VEK) VALUES( 'Jan', 'Haras', 20)");
      con->rollback();

      stm->executeUpdate("INSERT INTO OSOBY(MENO, PRIEZVISKO, VEK) VALUES( 'Peter',    'Novak', 25)");
      stm->executeUpdate("INSERT INTO OSOBY(MENO, PRIEZVISKO, VEK) VALUES( 'Radoslav', 'Nemec', 19)");
      con->commit();

      env->terminateConnection(con);
      Environment::terminateEnvironment(env);

   } catch (SQLException& e) {
      cout << "ORACLE ERROR:" << e.getErrorCode() << " - " << e.getMessage() << ;endl;
   }

   return 0;
}

Potom, ako program spustíte a nazriete do databázy, zistíte že prvý insert nebol vykonaný. Presnejšie bol vykonaný no následne bol volaný rollback. Až ďalšie hodnoty boli úspešne vložené do DB. Tie už boli commit-nuté.

Parametrizované dotazy

Vo vyššie spomenutom príklade neustále voláme principiálne ten istý INSERT len s inými hodnotami. Na takomto spôsobe nie je nič zlé, len je to trošku otrava zakaždým vytvárať kompletný reťazec obsahujúci INSERT. V Oracle však existujú takzvané parametrizované dotazy, ktoré to podstatne uľahčia. Parametrizovaný dotaz môže v našom prípade vyzerať následovne:

INSERT INTO OSOBY (MENO, PRIEZVISKO, VEK) VALUES ( :1, :2, :3 )

V tomto dotaze budeme nahradzovať :1, :2 a :3, ktoré plnia funkciu parametrov, potrebnými hodnotami. To sa nazýva bindovanie.

main.cpp
#include <iostream>
#include <occi.h>

using namespace std;
using namespace oracle::occi;

#define USER  "OCCIUSER"
#define PASS  "****"
#define TNS   "XE"

int main()
{
   Environment* env;
   Connection*  con;
   Statement*   stm;

   try {
      env = Environment::createEnvironment();
      con = env->createConnection(USER, PASS, TNS);

      stm = con->createStatement();
      stm->setAutoCommit(FALSE);
      stm->setSQL("INSERT INTO OSOBY (MENO, PRIEZVISKO, VEK) VALUES ( :1, :2, :3 )");

      //Vlozime prvy zaznam
      stm->setString(1, "Martin");
      stm->setString(2, "Fabi");
      stm->setInt(3, 24);
      stm->executeUpdate();

      //Vlozime druhy zaznam
      stm->setString(1, "Jan");
      stm->setString(2, "Fabi");
      stm->setInt(3, 22);
      stm->executeUpdate();

      con->commit();

      env-&g;tterminateConnection(con);
      Environment::terminateEnvironment(env);

   } catch (SQLException& e) {
      cout << "ORACLE ERROR:" << e.getErrorCode() << " - " << e.getMessage() << endl;
   }

   return 0;
}

SQL dotaz pre Statement nastavíme volaním funkcie setSQL(). Jednotlivé hodnoty parametrov nastavíme volaním funkcii setString() a setInt() podľa potreby. Je potrebné dať si pozor na to, aký typ premennej bindujeme. Stĺpec VEK je číselný. Ak by sme bindovali textový reťazec, program by vyvolal výnimku. Keď už máme Statement pripravený a naplnený hodnotami, voláme executeUpdate() bez parametrov, ktorý sa postará o vykonanie dotazu.

Hodnoty môžeme nielen vkladať do SQL dotazov, ale taktiež môžeme ich aj získavať. Predstavme si situáciu, kedy máme databázovú procedúru myProc(), ktorá ma jeden vstupný a druhy výstupný parameter, do ktorého sa uloží dvojnásobok prvého parametra. Naša aplikácia zavolá túto procedúru s určitou hodnotou p1 a nakoniec získa výsledok p2.

main.cpp
#include <iostream>
#include <occi.h>

using namespace std;
using namespace oracle::occi;

#define USER  "OCCIUSER"
#define PASS  "****"
#define TNS   "XE"

int main()
{
   Environment* env;
   Connection*  con;
   Statement*   stm;
   int result = 0;

   try {
      env = Environment::createEnvironment();
      con = env->createConnection(USER, PASS, TNS);

      stm = con->createStatement();
      stm->setSQL("BEGIN myProc(:1,:2); END;");

      stm->setInt(1,5);
      stm->setInt(2,0);
      stm->executeUpdate();

      result = stm->getInt(2);
      cout << "P2=" << result << endl;

      env->terminateConnection(con);
      Environment::terminateEnvironment(env);

   } catch (SQLException& e) {
      cout << "ORACLE ERROR:" << e.getErrorCode() << " - " << e.getMessage() << endl;
   }

   return 0;
}

Volanie storovanej procedúry uzatvárame medzi BEGIN a END. Ako som v úvode písal, statementy možu vykonávať nielen jednoduché dotazy typu INSERT, UPDATE, DELETE, ale aj celé skripty. Funkcia setSQL() je ekvivalentom SQL command oknu, ktoré sa nachádza vo web rozhraní. Pred executeUpdate() musíme bindovať a tým inicializovať aj parameter :2, hoci je to výstupná premenná. V opačnom prípade by program skončil výnimkou. Po executeUpdate() nasleduje získanie výstupnej hodnoty. Ide o getInt() funkciu, ktorá je opakom funkcie setInt(). Opäť závisí na type premennej. Tieto funkcie konvertujú dátové typy medzi ORACLE XE a C++.

V ďalšom dieli si predstavíme streamy. Potom sa pozrieme na SELECTy pomocou ResultSet triedy, čím by sme sa mali dostať pomaličky k záveru relačného programovania OCCI.

×Odeslání článku na tvůj Kindle

Zadej svůj Kindle e-mail a my ti pošleme článek na tvůj Kindle.
Musíš mít povolený příjem obsahu do svého Kindle z naší e-mailové adresy kindle@programujte.com.

E-mailová adresa (např. novak@kindle.com):

TIP: Pokud chceš dostávat naše články každé ráno do svého Kindle, koukni do sekce Články do Kindle.

1 názor  —  1 nový  
Hlasování bylo ukončeno    
0 hlasů

Nové články

Obrázek ke článku Delphi 10.1.2 (Berlin Update 2) – na co se můžeme těšit

Delphi 10.1.2 (Berlin Update 2) – na co se můžeme těšit

Touto roční dobou, kdy je zem pokrytá barevným listím a prsty křehnou v mrazivých ránech, se obvykle těšíme na zbrusu novou verzi RAD Studia. Letos si však ale budeme muset počkat na Godzillu a Linux až do jara. Vezměme tedy za vděk alespoň updatem 2 a jelikož dle vyjádření pánů z Embarcadero se budou nové věci objevovat průběžně, pojďme se na to tedy podívat.

Reklama
Reklama
Obrázek ke článku Konference: Moderní datová centra pro byznys dneška se koná už 24. 11.

Konference: Moderní datová centra pro byznys dneška se koná už 24. 11.

Stále rostoucí zájem o cloudové služby i maximální důraz na pružnost, spolehlivost a bezpečnost IT vedou k výrazným inovacím v datových centrech. V infrastruktuře datových center hraje stále významnější roli software a stále častěji se lze setkat s hybridními přístupy k jejich budování i provozu.

Obrázek ke článku Konference: Mobilní technologie mají velký potenciál pro byznys

Konference: Mobilní technologie mají velký potenciál pro byznys

Firmy by se podle analytiků společnosti Gartner měly  rychle přizpůsobit skutečnosti, že mobilní technologie už zdaleka nejsou horkou novinkou, ale standardní součástí byznysu. I přesto - nebo možná právě proto - tu nabízejí velký potenciál. Kde tedy jsou ty největší příležitosti? I tomu se bude věnovat již čtvrtý ročník úspěšné konference Mobilní řešení pro business.

Obrázek ke článku Hackerský kongres přiveze v září do Prahy špičky světové kryptoanarchie

Hackerský kongres přiveze v září do Prahy špičky světové kryptoanarchie

Hackerský kongres HCPP16 pořádá od 30. září do 2. října nezisková organizace Paralelní Polis již potřetí, a to ve stejnojmenném bitcoinovém prostoru v pražských Holešovicích. Letos přiveze na třídenní konferenci přes 40 většinou zahraničních speakerů – lídrů z oblastí technologií, decentralizované ekonomiky, politických umění a aktivismu. Náměty jejich přednášek budou také hacking, kryptoměny, věda, svoboda nebo kryptoanarchie.

loadingtransparent (function() { var po = document.createElement('script'); po.type = 'text/javascript'; po.async = true; po.src = 'https://apis.google.com/js/plusone.js'; var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(po, s); })();
Hostujeme u Českého hostingu       ISSN 1801-1586       ⇡ Nahoru Webtea.cz logo © 20032016 Programujte.com
Zasadilo a pěstuje Webtea.cz, šéfredaktor Lukáš Churý