1 / 98

Interoperabilità nei sistemi di gestione dei dati

Interoperabilità nei sistemi di gestione dei dati. Sommario. Introduzione Architettura di riferimento per ODBC,JDBC Flusso applicazioni ODBC architettura livelli di conformità il flusso generale di un’applicazione ODBC le primitive attraverso un esempio ODBC in Oracle JDBC architettura

mimis
Télécharger la présentation

Interoperabilità nei sistemi di gestione dei dati

An Image/Link below is provided (as is) to download presentation Download Policy: Content on the Website is provided to you AS IS for your information and personal use and may not be sold / licensed / shared on other websites without getting consent from its author. Content is provided to you AS IS for your information and personal use only. Download presentation by click this link. While downloading, if for some reason you are not able to download a presentation, the publisher may have deleted the file from their server. During download, if you can't get a presentation, the file might be deleted by the publisher.

E N D

Presentation Transcript


  1. Interoperabilità nei sistemi di gestione dei dati

  2. Sommario • Introduzione • Architettura di riferimento per ODBC,JDBC • Flusso applicazioni • ODBC • architettura • livelli di conformità • il flusso generale di un’applicazione ODBC • le primitive attraverso un esempio • ODBC in Oracle • JDBC • architettura • tipologie di driver • le classi principali • il flusso generale di un’applicazione ODBC • i metodi attraverso un esempio • JDBC in Oracle

  3. Introduzione • Il problema principale nelle architetture client-server è l’esistenza di client eterogenei: • Backend: database server • Frontend: gui/forms/interfaccie web • Nasce il problema di capire come le applicazioni client possano comunicare con il server: • come descrivere le query • come descrivere i risultati delle query • Prima soluzione: uso SQL • Svantaggi: cambiando server potrebbe essere necessario cambiare il client, in quanto lo specifico dialetto SQL potrebbe cambiare • Necessità di accedere i dati in modo interoperabile, cioè indipendente dallo specifico server considerato • se cambia il server non cambia l’applicazione • i tempi di sviluppo applicazioni client si riducono

  4. Introduzione • L’interoperabilità rappresenta il problema principale nello sviluppo di applicazioni eterogenee per sistemi distribuiti • Richiede la disponibilità di funzioni di adattabilità e di conversione che rendano possibile lo scambio di informazione tra sistemi, reti ed applicazioni, anche se eterogenei • l’interoperabilità è resa possibile dall’utilizzo di protocolli standard come FTP, SMTP/MIME, ecc. • In riferimento ai sistemi di gestione dei dati, l’interoperabilità ha richiesto lo sviluppo di standard adeguati, nella forma di API (Application Program Interface) • ODBC • JDBC

  5. Introduzione • Per comprendere la necessità di comunicare attraverso API con un DBMS, consideriamo come sia possibile in generale realizzare questa comunicazione: • embedded SQL • moduli SQL • call-level interface (CLI)

  6. Introduzione • Embedded SQL: • SQL “inserito” in un linguaggio di programmazione • statement processati da uno speciale precompilatore • visto in DB1 • può essere: • statico (statement noti a compile-time) • dinamico (statement generati a run-time) • meccanismo di interazione standard ma il codice dipende dal DBMS prescelto • cambiando DBMS l’applicazione deve essere nuovamente compilata • Esempio: per Java SQLJ

  7. Introduzione • Moduli SQL • un modulo consiste di un gruppo di procedure, chiamate da un linguaggio di programmazione ospite, ogni procedura contiene un singolo statement SQL • un modulo può essere interpretato come un oggetto di libreria legato al codice applicativo • questo collegamento dipende dall’implementazione: • le procedure possono essere compilate e linkate al codice applicativo • possono essere compilate e memorizzate nel DBMS e chiamate dal codice applicativo • possono essere interpretate • chiara separazione tra statement SQL e linguaggio di programmazione • meccanismo di interazione standard ma il codice dipende dal DBMS prescelto • cambiando DBMS l’applicazione deve essere nuovamente compilata

  8. Introduzione • Esempio: • in Oracle, moduli scritti in PL/SQL • compilati, memorizzati nel DBMS

  9. Introduzione • Call-level interface • libreria di funzioni del DBMS che possono essere chiamate dai programmi applicativi • simile alle tipiche librerie C • passi tipici: • l’applicazione effettua una chiamata ad una funzione CLI per connettersi al DBMS • l’applicazione crea uno statement SQL e lo inserisce in un buffer, quindi chiama una o più funzioni CLI per inviare lo statement al DBMS per l’esecuzione • se lo statement è di tipo SELECT, la funzione restituisce le tuple del risultato • se è di tipo INSERT, DELETE, UPDATE, la funzione restituisce il numero di tuple modificate • se è di tipo DDL, non viene restituito alcun valore significativo • l’applicazione effettua una chiamata ad una funzione CLI per disconnettersi dal DBMS

  10. Introduzione • Esempio: la CLI di Oracle si chiama OCI (Oracle Call Interface) • Esistono alcuni standard per CLI: • X/Open Specification SQL CLI • ISO/IEC SQL/CLI

  11. Introduzione • Tra i tre livelli di interazione precedenti, le CLI sembrano il meccanismo più adeguato per garantire interoperabilità: • indipendenti dal codice client • l’applicazione cambia interazione con il server chiamando in modo opportuno le funzioni di libreria • chiara distinzione tra interfaccia ed implementazione • una stessa applicazione binaria può funzionare considerando diversi DBMS • Necessità di standardizzazione • le applicazioni devono potere accedere DBMS diversi usando lo stesso codice sorgente • le applicazioni devono potere accedere DBMS diversi simultaneamente • ODBC, JDBC si possono vedere come CLI ottenute come risultato del processo di standardizzazione • ODBC: libreria C • JDBC: libreria Java

  12. Architettura di riferimento API API

  13. Applicazione • Un’applicazione è un programma che chiama specifiche funzioni API per accedere ai dati gestiti da un DBMS • Flusso tipico: • selezione sorgente dati (DBMS e specifico database) e connessione • sottomissione statement SQL per l’esecuzione • recupero risultati e processamento errori • commit o rollback della transazione che include lo statement SQL • disconnessione

  14. Driver Manager • È una libreria che gestisce la comunicazione tra applicazione e driver • risolve problematiche comuni a tutte le applicazioni • quale driver caricare, basandosi sulle informazione fornite dall’applicazione • caricamento driver • chiamate alle funzioni dei driver • l’applicazione interagisce solo con il driver manager

  15. Driver • Sono librerie dinamicamente connesse alle applicazioni che implementano le funzioni API • ciascuna libreria è specifica per un particolare DBMS • driver Oracle è diverso dal driver Informix • traducono le varie funzioni API nel dialetto SQL utilizzato dal DBMS considerato (o nell’API supportata dal DBMS) • il driver maschera le differenze di interazione dovute al DBMS usato, il sistema operativo e il protocollo di rete • Si occupano in particolare di: • iniziare transazioni • sottomettere statement SQL • inviano dati e recuperano dati • gestiscono errori

  16. DBMS • Il DBMS sostanzialmente rimane inalterato nel suo funzionamento • riceve sempre e solo richieste nel linguaggio supportato • esegue lo statement SQL ricevuto dal driver e invia i risultati

  17. Esempio • Nel seguito gli esempi si riferiranno alla tabella Impiegati definita in SQL come segue: CREATE TABLE Impiegati (Matr VARCHAR(4), Nome VARCHAR(20), Cognome VARCHAR(20), Manager VARCHAR(4), Stipendio NUMBER);

  18. ODBC (Open DataBase Connectivity) • Standard proposto da Microsoft nel 1991 • Supportata da praticamente tutti i sistemi di gestione dati relazionali • Offre all’applicazione un’interfaccia che consente l’accesso ad una base di dati non preoccupandosi di: • il particolare dialetto di SQL • il protocollo di comunicazione da usare con il DBMS • la posizione del DBMS (locale o remoto) • è possibile connettersi ad un particolare DB tramite una DSN (Data Source Name), che contiene tutti i parametri necessari alla connessione con il DB: • protocollo di comunicazione • tipo di sorgente dati (es: Oracle DBMS) • specifico database

  19. ODBC • Il linguaggio supportato da ODBC è un SQL ristretto, caratterizzato da un insieme minimale di istruzioni • Questa è una scelta obbligata se si vuole permettere un cambio di DBMS lasciando inalterati i client • Il linguaggio permette di eseguire query statiche per applicazioni di tipo tradizionale o dinamiche, per applicazioni interattive • Nel primo caso eventuali errori di SQL sono riportati al momento stesso della compilazione • Architettura conforme a quando descritto in generale • Funzioni implementate in C, quindi non completamente portabili • è compatibile con X/open SQL CLI e ISO SQL/CLI

  20. Architettura

  21. Architettura • Driver Manager: • su Windows, fornito con il sistema operativo • con Linux, deve essere installato • Driver: • forniti con il DBMS a cui si riferiscono • spesso installati e registrati a livello Driver Manager quando si installa il DBMS client

  22. Funzionamento di una query ODBC

  23. Come registrare una DSN? • I driver ODBC mettono in genere a disposizione tool di amministrazione che permettono di specificare tutti i dettagli relativi ad un DSN (Data Source Name) • DSN: Data Source Name, è la stringa che identifica: • tipo di driver • tipo di comunicazione • tipo di database • nome database • Dopo la registrazione, queste informazioni sono a disposizione del Driver Manager

  24. Vantaggi e svantaggi • Vantaggi • astrazione (almeno teorica) rispetto al DBMS utilizzato • diffusione: supportato almeno parzialmente da molti DBMS commerciali e non • non modifica il server • Svantaggi • per ogni coppia (piattaforma client, server) serve un driver specifico (strategia commerciale Microsoft) • gran parte del lavoro è demandata al driver, che diventa quindi cruciale per il funzionamento

  25. Livelli di conformità • DBMS diversi possono avere diverso potere espressivo • Inoltre i driver possono implementare solo una parte della libreria ODBC • I livelli di conformità permettono di specificare: • quali funzioni ODBC sono implementate dal driver • (livello di conformità dell’interfaccia) • quali statement SQL il DBMS è in grado di eseguire • (livello di conformità SQL)

  26. Livelli di conformità di interfaccia • Ogni livello di conformità è rappresentato da un insieme di funzioni API, per ciascuna delle quali viene specificata la modalità di chiamata (vincoli sui parametri) • L’applicazione può recuperare il livello di conformità del driver invocando una opportuna funzione ODBC (SQLGetInfo) • Sono stati definiti tre livelli: • Core: funzionalità di base • Livello 1: Core + funzionalità addizionali, supportate da tipici sistemi relazionali, come transazioni • Livello 2: Livello 1 + funzionalità di amministrazione, tra cui gestione cataloghi

  27. Livelli di conformità SQL • Ogni driver deve supportare una grammatica SQL-92 minima • Esistono altri livelli, che permettono di supportare un insieme maggiore di statement SQL • Entry level • intermediate level • full level • Possibilità di recuperare il livello con una opportuna funzione (SQLGetInfo)

  28. Tipi di dato • ODBC utilizza due insiemi di tipi di dato: • tipi di dato SQL, utilizzati nel database • tipi di dato C, da utilizzare nell’applicazione • Tipi di dato SQL: • ogni database definisce i propri tipi di dato mentre ODBC definisce identificatori di tipi di dato • il driver poi stabilisce come questi tipi di dato debbano essere mappati sugli identificatori ODBC • Esempio: SQL_CHAR identificatore per CHAR • Tipi di dato C: • ODBC definisce identificatori per i tipi di dato C (SQL_C_CHAR) • ODBC definisce anche un mapping da tipi di dato SQL a tipi di dato C

  29. Alcuni identificatori di tipi di dato SQL

  30. Alcuni identificatori di tipi di dato SQL

  31. Alcuni identificatori di tipi di dato C

  32. Alcuni identificatori di tipi di dato C

  33. Flusso applicazione ODBC (3.51)

  34. Flusso generale applicazione ODBC • Nel seguito vedremo solo le operazioni principali • Ogni funzione che vedremo restituisce un valore di tipo SQLRETURN, che può valere: • SQL_SUCCESS, se l’operazione è andata a buon fine • SQL_ERROR, se si è verificato qualche errore • SQL_SUCCESS_WITH_INFO, una sorta di warning • Nel seguito indicheremo solo i valori di ritorno diversi dai precedenti

  35. Passo 1: Connessione henv Ambiente • Caricamento Driver Manager • L’esecuzione di questo passo dipende dal sistema operativo • Allocazione ambiente: • buffer nel quale verranno registrate tutte le informazioni relative al driver e al DBMS verso il quale viene effettuata la connessione SQLHENV henv1; SQLAllocHandle(SQL_HANDLE_ENV,SQL_NULL_HANDLE,&henv1); • Possibilità di settare parametri d’ambiente (funzione SQLSetEnvAttr, che non vediamo) • versione ODBC utilizzata

  36. Connessione hdbc Buffer transito dati db • Definizione buffer di connessione: generazione buffer, atto a contenere informazioni relative alla connessione SQLHDBC hdbc1; SQLHENV henv1; SQLAllocHandle(SQL_HANDLE_DBC,henv1,&hdbc1); • Possibilità di settare parametri di connessione (funzione SQLSetConnectAttr, che non vediamo) • informazioni relative alla gestione di transazioni • Il driver non viene ancora caricato • verrà caricato solo al momento della connessione

  37. Connessione • Connessione al database: tre possibilità, vediamo la più semplice: SQLHDBC hdbc1; SQLConnect(hdbc,”DSN”, SQL_NTS, ”id_utente”, SQL_NTS, ”password”, SQL_NTS); /* SQL_NTS = lunghezza del parametro

  38. Passo 2: Inizializzazione hstmt1 Comando SQL • Allocazione buffer per statement SQL SQLHDBC hdbc1; SQLHSTMT hstmt1; SQLAllocHandle(SQL_HANDLE_STMT, hdbc1,&hstmt1); • Possibilità di settare parametri di statement (funzione SQLSetStmtAttr, che non vediamo) • informazioni relative alla gestione del cursore

  39. Esempio • SQLHENV henv; /* variabile ambiente • SQLHDBC hdbc; /* variabile buffer transito dati • SQLHSTMT hstmt; /* variabile comando SQL • SQLRETURN re; /* valore di ritorno • re = SQLAllocHandle(SQL_HANDLE_ENV,SQL_NULL_HANDLE, &henv); • if(re == SQL_SUCCESS) { • re = SQLAllocHandle(SQL_HANDLE_DBC,henv,&hdbc); • if(re == SQL_SUCCESS) { • re = SQLConnect(hdbc, • ”ContiCorrenti”, SQL_NTS, /*Riferimento al server • ”Pippo”, SQL_NTS, /*Utente • ”password”, SQL_NTS); /*Password dell’utente • if(re == SQL_SUCCESS) { • re = SQLAllocHandle(SQL_HANDLE_STMT,hdbc,&hstmt); • if(re == SQL_SUCCESS) { • ” ” ”.. Interrogazioni ed elaborazione dati ” ” ”. • }}}}

  40. Passo 3: Costruzione ed esecuzione statement SQL • Due modi principali di esecuzione: • diretta: l’applicazione definisce lo statement SQL (eventualmente preso come input), che viene ottimizzato ed eseguito in un unico passo a run-time • utile quando lo statement viene utilizzato una sola volta • preparata: l’applicazione definisce lo statement SQL che viene ottimizzato ed eseguito a run-time in due passi distinti • il piano di accesso viene generato e mantenuto per le successive esecuzioni • utile quando lo statement deve essere eseguito più volte • spesso si utilizzano parametri da settare dopo la preparazione e prima dell’esecuzione (non lo vediamo)

  41. Costruzione ed esecuzione di statement SQL • Esecuzione diretta: SQLHSTMT hstmt1; SQLCHAR* SQLStatement; SQLExectDirect(hstmt1,SQLStatement,SQL_NTS); • Esecuzione preparata: SQLHSTMT hstmt1; SQLCHAR* SQLStatement; SQLPrepare(hstmt1,SQLStatement,SQL_NTS); SQLExecute(hstmt1);

  42. Esempio esecuzione Diretta: SQLExectDirect(hstmt, ”SELECT* FROM Impiegati”); Preparata: SQLPrepare(hstmt,”SELECT*FROM Impiegati”,SQL_NTS); SQLExecute(hstmt); Per ogni successiva esecuzione non devo più invocare SQLPrepare

  43. Passo 4: Elaborazione risultato • Gli statement di tipo SELECT restituiscono un result set, mentre gli statement di tipo INSERT, DELETE, UPDATE restituiscono il numero delle tuple aggiornate • Per verificare se lo statement ha restituito un result set: SQLHSTMT hstmt1; SQLSMALLINT * number; SQLNumResultCols(hstmt1,&number); se restituisce 0, allora il result set è vuoto • Questa informazione può essere utile nel caso in cui lo statement sia costruito dinamicamente

  44. Elaborazione risultato • I campi delle tuple restituite devono essere associati a determinate variabili, nelle quali verranno inseriti i valori durante la scansione SQLHSTMT hstmt1; SQLSMALLINT ColumnNumber; SQLSMALLINT TargetType; SQLPOINTER TargetValuePtr; SQLINTEGER BufferLenght; SQLINTEGER* Info; SQLBindCol(hstmt1,ColumnNumber,TargetType, TargetValuePtr,BufferLenght,Info);

  45. Elaborazione risultato • ColumnNumber: numero progressivo colonna della tabella • TargetType: tipo C nel quale convertire il tipo SQL • TargetValuePtr: puntatore al campo nel quale inserire il valore estratto dalla tupla • BufferLenght: lunghezza in byte TargetValuePtr • Info: buffer (di tipo intero) nel quale l’istruzione SQLFetchinserirà informazioni circa il risultato dell’operazione di binding: • la lunghezza in byte del campo da inserire nella variabile • SQL_NULL_DATA: il campo contiene NULL • SQL_NO_DATA: nessuna tupla letta

  46. Esempio di binding SQLCHAR Matr[4]; SQLCHAR Nome[20]; SQLCHAR Cognome[20]; SQLCHAR Manager[4]; SQLINTEGER Stipendio, Matr_info,Nome_info,Cognome_info, Indirizzo_info,Manager_info,Stipendio_info; SQLRETURN re; SQLHSTMT hstmt; SQLBindCol(hstmt, 1, SQL_C_CHAR, Matr, sizeof(Matr), Matr_info); SQLBindCol(hstmt, 2, SQL_C_CHAR, Nome, sizeof(Nome), Nome_info); SQLBindCol(hstmt, 3, SQL_C_CHAR, Cognome, sizeof(Cognome), Cognome_info); SQLBindCol(hstmt, 4, SQL_C_CHAR, Manager, sizeof(Manager), Manager_info); SQLBindCol(hstmt, 5, SQL_C_CHAR, Stipendio, sizeof(Stipendio), Stipendio_info);

  47. Elaborazione risultato • Un cursore viene automaticamente creato quando viene eseguito uno statement che crea un result set • Simile all’SQL da programma • Un cursore è una sorta di puntatore che si sposta sulle tuple contenute nel result set • L’istruzione che ci permette di muoverci sulla tupla successiva è SQLHSTMT hstmt1; SQLFetch(hstmt1); • Questa istruzione può restituire: • SQL_SUCCESS: la tupla è stata correttamente letta • SQL_ERROR: si è verificato un errore • SQL_NO_DATA: non ci sono più tuple

  48. Elaborazione risultato • Se lo statement non ha creato alcun result set, SQLFetch provoca un errore • Al termine della scansione il cursore deve essere esplicitamente chiuso SQLHSTMT hstmt1; SQLCloseCursor(hstmt1);

  49. Esempio while((re= SQLFetch(hstmt)) != SQL_NO_DATA) { if(re ==SQL_ERROR..) {”.segnala errore ”.. }; if(re ==SQL_SUCCESS..) { printf(” Nome: %s %s, Manager: %s, Stipendio: %d ”, Nome,Cognome,Manager, Stipendio); } } SQLCloseCursor(hstmt);

  50. Elaborazione risultato • Se lo statement è di tipo INSERT, DELETE, UPDATE si può avere la necessità di determinare il numero di tuple aggiornate • nel momento in cui lo statement viene eseguito, il numero di tuple aggiornate viene mantenuto in una variabile di sistema • per recuperarlo: SQLHSTMT hstmt1; SQLSMALLINT number; SQLRowCount(hstmt1,&number);

More Related