Download Índice General - Instituto Wiener

Document related concepts
Transcript
INSTITUTO SUPERIOR TECNOLÓGICO
NORBERT WIENER
Manual del Alumno
ASIGNATURA: Lenguaje de Programación II
(Visual Basic II)
PROGRAMA: S3C
LIMA-PERU
2
Manual del Alumno
Índice General
Pag N°
1. Introducción y base de datos....................................................................................... 2
2. Control Ado ............................................................................................................... 4
3. El Control Ado Data Control 6.0 (OLEDB)................ .............................................. 8
4. El Control Ado Data Control 6.0 (OLEDB)(continuación)....................................... 44
5. El Control Ado Data Control 6.0 (OLEDB)(continuación)....................................... 50
6. Controles Enlazados .................................................................................................. 56
7. Uso de Múltiples Tablas ............................................................................................ 57
8. Uso de Múltiples Tablas (continuación)..................................................................... 58
9. El Data Environment ..................................................................................................63
10. Visual Basic – SQL ................................................................................................... 69
11. Controles no Enlazados ..............................................................................................70
12. Controles no Enlazados y Módulo de Clase................................................................78
13. Controles Active X......................................................................................................86
14. Funciones Api‟s.........................................................................................................91
3
Manual del Alumno
INTRODUCCION Y BASE DE DATOS
Acceso a Datos:
Los datos se consideran uno de los activos más valiosos de la empresa moderna. Por
tal razon, es importante que la misma cuente con una estrategia apropiada para
administrarlos adecuadamente; esto es, almacenarlos, recuperarlos y procesarlos de tal
forma que sean importantes para la toma de decisiones.
Visual Basic provee soporte a datos de muchas maneras. Se puede utilizar DAO (Data
Access Objects) para manipular datos provenientes de Microsoft Jet o Access o Excel,
ODBC (Open Database Conectivity), ISAM y tecnologís RDO (Remote Data Object) y
ADO (Actives Data Object).
Los Objetos de acceso a datos (DAO) y el control Data utilizan el motor de base de
datos Microsoft Jet para tener acceso a las bases de datos. El motor de base de datos Jet
puede obtener acceso a tres tipos de bases de datos:
® Bases de datos Microsoft Jet:
Es un motor utilizado por Microsoft Access y Visual Basic. Este motor permite crear
y manipular datos directamente.
® Bases de datos del Método de acceso secuencial indizado (ISAM, Indexed
Sequential Access Method):
Dada la existencia de formatos conocidos y extendidos mundialmente, como por
ejemplo, Btrieve, dBase, Microsoft Visual FoxPro, Paradox, entre otros, DAO
permite la manipulación de datos de éstos formatos.
® Bases de datos compatibles con Open Database Connectivity (ODBC,
Conectividad abierta de bases de datos):
No solo las bases de datos de escritorio o denominadas “desktop” tienen soporte por
parte de DAO. Las bases de datos del estilo cliente/servidor que se ajustan al estándar
ODBC, tal como SQL Server, Oracle u otros a los cuales se diseñó su apropiado
“provider” se le da soporte.
Visual Basic también provee otros métodos de acceso a datos, tal y como:
® Control de origen de datos remotos
En la edición empresarial de Visual Basic se dispone de un control de origen de datos
remotos, que soportan ODBC, tal como Microsoft SQL Server y Oracle.
4
Manual del Alumno
® Bibliotecas ODBC
Le permiten llamar directamente a la interfaz de programación de aplicaciones (API)
de ODBC. Está disponible como un producto independiente.
® Bibliotecas SQL para Visual Basic (VBSQL)
Proporcionan un vínculo directo a Microsoft SQL Server. Están disponibles como
productos independientes.
Descripción de una Base de Datos:
Hoy día la mayoría de sistemas informáticos que administran la información de una
empresa utilizan un esquema relacional. Esto es, relacionan logicamente la información
utilizando tablas y entidades que conforman llaves de enlace.
Un esquema relacional presenta los datos como un grupo de tablas que se relaciona
logicamente entre si. Por ejemplo, Visual Basic provee la base de datos Biblio.Mdb y en
ella radican varias tablas de datos (tales como author y titles) que se relacionan entre si
gracias a la duplicacion que existe en ellas del ID u otros campos.
Se puede observar el diagrama de base de datos denominada Students And Classes
elaborada mediante asistente en Visual Foxpro, versión 6.0. Observense los campos,
índices y relaciones contenidos:
Elementos de una tabla:
La base de datos Students and classes contiene varias tablas que agrupan
información, como studentes, assigments, results, entre otras.
En una base de datos, las filas de tablas se denominan registros o tuplas y las
columnas campos. Los registros pertenecen a la descripción de una entidad, por ejemplo,
un cliente, un proveedor, un empleado, un producto, entre otros. Por otro lado, el campo
5
Manual del Alumno
es parte del registro como puede ser la cedula, el nombre, la existencia, la direccion, el
telefono, entre otros. Asimismo, existen índices (que pueden ser de diferentes tipos) y
reglas (reglas de validación y de integridad referencial).
Clave principal:
Cada tabla contenida en una base de datos posee una clave principal. Esta es un
campo, o la combinación de varios campos, que es exclusiva en cada fila de la tabla. Esto
permite identificar de forma uniequivocamente un registro particular, permitiendo que no
existan registros duplicados o accesar más rapidamente a un registro o a una colección de
ellos. En el caso de la base de la base de datos Students and classes, se tiene una tabla
denominada students, donde la clave primaria es studentid, dado que la identificación de
todos y cada uno de los estudiantes es única en la colección de registros. Asimismo, en la
tabla results se tiene una clave por el mismo campo, siendo una llave que se puede repetir
tantas veces como sea necesaria, dado que un estudiante puede registrar todos los resultados
obtenidas por materia. En este caso se determina una clave externa, dado que la misma no es
propia de la tabla sino de otra que es students.
Registros:
Un registro contiene información referente a una entrada a una tabla. Es preferible
que una tabla no contenga duplicados. En este caso es una tabla maestra. En el caso de la
tabla Students, es necesario que cada estudiante tenga su propio Id y no exista otro similar en
la tabla. Sin embargo, en registros históricos se repetirá tantas veces una llave como sea
necesario.
Campos:
Un campo de una tabla identifica una parte de un registro. Por ejemplo, la tabla
students contiene los campos studentid, lastname y firstname, entre otros.
Indices:
Los indices de una tabla son listas ordenadas en las que se puede localizar más
rápidamente un determinado registro o conjunto de ellos. Se crean en base a un campo de
la tabla, tomando como requerimiento cuál de ellos representa una entidad fuerte dentro de
la colección o cuál es el adecuado para ordenar un conjunto de registros.
Conectividad a datos en Visual Basic:
Existen dos métodos para abrir una base de datos en Visual Basic: utilizando el Data
Control (el cual es un objeto provisto por Visual Basic) o mediante el método OpenDatabase.
El control data es un objeto provisto por Visual Basic para la manipulación de datos en una o
varias tablas.
El control data implementa este acceso mediante la utilización de un motor de bases de
datos tal como Microsoft Jet. Existen métodos apropiados para la manipulación de datos y
se provee de una herramienta de generación automática de formularios que se encarguen de
dicha manipulacion, sin necesidad de escribir mucho código.
6
Manual del Alumno
Para construir una aplicación de bases de datos, utilizando el control data, se
recomienda seguir los siguientes pasos:
1. Agregar un control data al formulario y configurar las propiedades Databasename,
connect y recordsource, principalmente. Lo anterior para establecer la base de
datos a la que se va a conectar el formulario, el tipo de conexión (por defecto es
Microsoft Access) y el origen de datos (la tabla de la base de datos abierta).
2. Agregar controles tales como text box o label para que contengan los datos del
recordsource. Aqui hay que configurar datasource y datafield como las
propiedades que se conectarán a los datos del data control.
Usar controles enlazados a datos:
Cuando se utiliza un data control en un formulario generalmente es necesario diseñar
controles que permitan contener los datos cargados en dicho data control. Estos controles
pueden ser CheckBox, Image, Label, PictureBox, TextBox, ListBox, ComboBox y los
contenedores OLE.
Propiedades y Métodos del Control Data:
Además de las propiedades Databasename y RecordSource el data control posee otras
propiedades y métodos, tales como:
® Propiedad Connect
® Propiedad Exclusive
® Propiedad ReadOnly
® Propiedad Recordset
® Propiedades BOFAction y EOFAction
® Método Refresh
La Propiedad connect especifica el tipo de base de datos que se abrirá. Puede incluir
argumentos tales como Id de usuario y la contraseña.
La propiedad Exclusive determina si la base de datos será abierta en forma exclusiva o
no. Si el valor de esta propiedad es True la base de datos se abrirá exclusivamente y
ninguna aplicación podrá utilizarla hasta que se cierre.
La propiedad ReadOnly determina si la base de datos se abrirá para solo lectura o no.
Si la propiedad es True no se podrán modificar los datos obtenidos.
El objeto Recordset contiene los registros devueltos para el data control, basado en
una apertura de una tabla o producto de la ejecución de una instruccion SQL. Un recordset
tiene propieades y metodos que se pueden utilizar para trabajar con los registros obtenidos.
Las propiedaes BOFAction y EOFAction determinan que acción realizar cuando las
propiedades BOF o EOF del Recordset son True. Por ejemplo, si la propiead EOFAction
del control data es vbAddNew y utiliza el control data para situarse trás el último registro del
recordset, el control data ejecutará automáticamente el metodo AddNew para incluir un
nuevo registro.
7
Manual del Alumno
El método Refresh actualiza un objeto recordset. Si se cambia la propiedad
RecordSource del data control, se deberá llamar a este método para actual el conjunto de
registros en el data. Lo siguiente es un ejemplo de lo anterior:
Data1.RecordSource = "SELECT * FROM Empleados " & _
"WHERE [IdEmpleado] = " & txtEmpID.text
Data1.Refresh
Recordset:
Es el conjunto de registros contenidos en el control data. Un Recordset se almacena
en la memoria, transfiriendo los datos contenidos al disco, si fuera necesario.
Para manipular un Recordset debe utilizarse la propiedad del mismo nombre del data
control.
Para determinar los límites de un Recordset se utilizan las propiedades EOF y BOF.
Cuando se desplace al último registro (EOF) o al inicio del archivo (BOF), se ejecutará la
acción indicada por el valor de la propiedad BOFAction o EOFAction. Seguidamente se
muestra como Microsoft grafica estas propiedades:
Asimismo, se muestra como utilizar un objeto Recordset de un determinado control
Data:
Data1.Recordset.MoveNext 'Va al registro siguiente.
If Data1.Recordset.EOF Then
Data1.Recordset.MoveLast
End If
El ejemplo anterior muestra como desplazarse al registro siguiente en el Recordset y
posteriormente validar si se ha llegado al final del mismo.
Se pueden utilizar otros métodos y propiedades del objeto Recordset para recuperar,
agregar, modificar o eliminar registros de este objeto.
Las propiedades BOF y EOF del objeto Recordset indican si el registro actual está
al inicio o al final del Recordset, respectivamente. Si lo anterior es verdadero, se asignará
True a la propiedad respectiva. Si ambas son verdaderas es que no existen datos en el
Recordset.
8
Manual del Alumno
Método AddNew:
El metodo AddNew de un objeto Recordset se utiliza para agregar un nuevo registro.
Cuando se ejecuta este metodo, Visual Basic elimina los controles enlazados y asigna a la
propiedad EditMode del control data el valor dbEditAdd.
Para actualizar un registro en el Recordset luego de utilizar AddNew debe utilizarse el
método UpdateRecord o Update. Lo siguiente muestra este aspecto:
Sub cmdAdd_Click ()
Data1.Recordset.AddNew
End Sub
Metodo UpdateRecord:
El método UpdateRecord se utiliza para guardar el registro actual en una base de
datos. Es decir, posterior al metodo AddNew, tal y como se muestra a continuación:
Sub cmdUpdate_Click ()
Data1.UpdateRecord
End Sub
Metodo CancelUpdate:
Se utiliza para cancelar un método AddNew o Edit y actualizar los controles enlazados
al control data. Es decir, restaura los datos originales del Recordset. A continuación se
muestra el uso de este método:
Sub cmdCancel_Click ()
Data1.CancelUpdate
End Sub
Metodo CancelUpdate:
Este método se utiliza para eliminar un registro de una tabla. El registro eliminado
continuará siendo el actual hasta tanto el usuario no realice ninguna otra acción, tal y como
se muestra en el codigo siguiente:
Sub cmdDelete_Click ()
Data1.Recordset.Delete
Data1.Recordset.MoveNext
If Data1.Recordset.EOF Then
Data1.Recordset.MoveLast
End If
End Sub
9
Manual del Alumno
Usando Objetos Recordset
Un recordset es un objeto que presenta los registros de una tabla o el resultado de una
consulta. Se pueden usar objetos para manipular los datos de una base de datos a un nivel
de registro.
Se puede usar objetos “Field” para manipular los datos de un archivo a un nivel de
registro.
Los cuatro tipos de recordset son:
Table,
Dynaset,
Snapshot y
Forward-Only
Table: Este Recordset puede ser creado para una tabla Microsoft Access, pero no
soporta ODBC (Open Database Connectivity) o tablas ligadas. Es decir,
funciona para una sola tabla de Microsoft Access. Cuando se crea una tabla,
el motor de bases de datos JET abre dicha tabla y subsecuentemente se
produce la manipulacion de los datos, operando directamente sobre la misma.
Una ventaja importante de este tipo de recordset es que se puede utilizar un
indice. Esto hace más eficiente la ordenacion de registros mediante un índice,
mejorando significativamente las opciones de búsqueda. Para localizar un
determinado registro se puede utilizar el metodo SEEK y no FIND, dado que
este ultimo es mucho más lento.
Dynaset:Este recordset puede crearse sobre una o varias tablas, mediante consultas.
Esto es mediante un conjunto de referencias a registros en una o más tablas.
Con un dynaset, se pueden extraer y actualizar datos desde una o varias tablas,
incluyendo las ligas que puedan existir con otras bases de datos. La
actualizacion heterogenea es una caracteristica importante de los dynaset.
Uno de los más importantes beneficios es que los cambios realizados sobre
una tabla se realizan tambien en otra que se encuentra ligada a ésta.
Asimismo, los cambios hechos por otros usuarios se reflejan en el dynaset.
El dynaset es el más flexible y poderoso de los demás tipos de recordset, sin
embargo es mucho más lento que, por ejemplo, el tipo table.
Snapshot:Este recordset es una copia estática de un conjunto de registros, capturados
en el momento de crearse el recordset. Este puede contener campos de una o
más tablas de una base de datos. Este recordset es actualizable.
La principal ventaja de un snapshot es que éste crea menos “overhead” en el
procesamiento que los demás recordset. Además se pueden ejecutar consultas
y retornar los valores más rápidamente, especialmente cuando se trabaja con
ODBC.
Se debe considerar que para archivos MDB, OLE y archivos memo son
direccionados por punteros en la consulta.
10
Manual del Alumno
Forward-Only: Este tipo de recordset (muchas veces denominado forward-scrolling
snapshot o forward-only snapshot) provee un subconjunto de capacidades de
un snapshot. Con este objeto se puede mover solo en dirección adelante a
traves de registros. Solo se soporta los metodos Move o Move Next. La
ventaja de este tipo de recordset es que usualmente provee la mayor velocidad
entre los demas recordset. Sin embargo, ofrece la menor funcionalidad que
los demas.
Un Snapshot almacena una copia de todos los registros (excepto objetos OLE
y campos memo).
Un Dynaset almacena justamente la llave primaria para cada registro,
copiando la totalidad de registros solo cuando es necesario para edicion o
propositos de despliegue. Desde un snapshot almacena una copia completa de
todos los registros en una tabla, éste puede rendir más lentamente que un
dynaset si el numero de registros es muy cuantioso. Para determinar cuando
usar un dynaset o un snapshot se pueden abrir ámbos, ejecutar la misma
consulta y comparar los tiempos de respuesta.
El tipo de recordset que se use depende de lo que el usuario debe hacer con los
datos. Puede que solo desee realizar una vista o que además deba
actualizarlos. Por ejemplo, si el usuario desea ordenar los datos o trabajar
con indices, usese un tipo table, dado que estos estan indexados y son más
rapidos de localizar. Otro caso es que si el usuario desea actualizar un
conjunto de registros seleccionados por una consulta, debe usarse el dynaset.
Finalmente, si la tabla que se desea consultar pocas veces está disponible y se
desea realizar una busqueda en un conjunto de registros determinados puede
usarse un tipo forward-only.
Creando una variable objeto:
Para crear una variable objeto, debe usarse el método OpenRecordset. Primero debe
declararse el tipo de variable y el grupo de variables para el objeto, retornado por el metodo
OpenRecordset. Puede usarse el método OpenRecordset con Database, TableDef,
QueryDef, y objetos existentes.
La sintaxis del método OpenRecordset es:
Set variable = database.OpenRecordset (source [, type [, options [, lockedits ]]])
La sintaxis del método OpenRecordset para todos los tipos de objetos es:
Set variable = object.OpenRecordset ([type [, options [, lockedits ]]])
11
Manual del Alumno
La argumento variable es el nombre del nuevo objeto. El argumento database es el nombre
del objeto de base de datos abierto. El argumento Object es el TableDef, QueryDef, o el
objeto existente.
El argumento source especifica el origen de los registros para el nuevo objeto. El valor
del origen es el valor del objeto DAO. Cuando se crea un nuevo objeto desde un objeto
Database el argumento source es un TableDef o QueryDef en la base de datos o un retorno
válido de una consulta SQL o sentencia. Cuando se crea un objeto de este tipo el mismo
provee el origen de datos para el nuevo objeto.
El argumento type es una constante intrinseca que especifica la clase de objeto que se
desea crear. Se pueden utilizar las siguientes constantes:
dbOpenTable
dbOpenDynaset
dbOpenSnapshot
dbOpenForwardOnly
El tipo de constante dbOpenForwardOnly reemplaza al tipo de constante
dbForwardOnly que estaba contenida en versión posterior de DAO. Es decir, esta
constante se mantiene por compatibilidad con la version anterior de DAO.
La siguiente sección muestra los tipos, opciones y argumentos de “lockedits” en detalle:
Tipos por defecto:
Porque DAO automaticamente escoge el tipo por defecto, dependienteo del origen de
datos y como se realiza la apertura, no se necesita especificar el tipo. Sin embargo, se puede
especificar un tipo diferente usando el argumento type en el metodo OpenRecordset.
La lista siguiente describe los tipos disponibles y los tipos por defecto, dependiendo de
cómo se abra el objeto:
Usando el método OpenRecordset con un objeto Database
Set rstNew = dbs.OpenRecordset("Data Source")
Si el recurso de datos es una tabla local en la base de datos, los cuatro tipos estan
disponibles y el tipo table está por defecto.
Si el origen de datos es diferente, solo
Dynaset y Snapshot estan disponibles. Dynaset es el default.
Usando el método OpenRecordset con un objeto TableDef:
12
Manual del Alumno
Set rstNew = tdfTableData.OpenRecordset
Si tdfTableData se refiere a una tabla de Microsoft Access (*.mdb) o ISAM abierta
directamente, entonces los cuatro tipos estan disponibles. Si tdfTableData es un base
de datos abierta mediante ODBC o es una tabla ligada en una tabla externa, solo los
tipos dynaset y snapshot estan disponibles.
Usando el método OpenRecordset con un objeto QueryDef:
Set rstNew = qdfQueryData.OpenRecordset
Solo los objetos dynaset y snapshot estan disponibles.
Usando el método OpenRecordset con un objeto existente:
Set rstNew = rstExisting.OpenRecordset
Solo los objetos dynaset y snapshot estan disponibles.
Opciones OpenRecordset:
Con el argumento Options del metodo OpenRecordset se puede especificar un numero
de otras caracteristicas para un objeto. Se pueden utilizar las siguientes constantes:
DbAppendOnly: Los usuarios pueden agregar nuevos registros, pero no se puede
editar o eliminar. Esto es util en aplicaciones que capturan y archivan muchos datos
(solo dynaset).
DbReadOnly: No se pueden hacer cambios. Este argumento es proveido solo por
compatibilidad de versiones anteriores. Se debe utilizar la constante dbReadOnly en
el argumento lockedits para ello.
DbSeeChanges: Si otro usuario realiza cambios en los datos en los registros (con edit
y update) se produce un error en tiempo de ejecucion. Esto es util en aplicaciones
donde multiples usuarios realizan, simultaneamente, operaciones de lectura/escritura
sobre los mismos datos (solo dynaset y table).
DbDenyWrite: Cuando se usa con dynaset o snapshot, esta opcion previene a otros
usuarios de inserciones o modificaciones de registros. Cuando una tabla está en uso
otros usuarios no pueden abrir ningun otro tipo de recordset.
DbDenyRead: Otros usuarios no pueden leer datos en la tabla (solo tabla)
DbForwardOnly: Esta opcion crea un recordset forward-only o snapshot. Este es
proveido solo por compatibilidad con versiones anteriores. Use la constante
dbOpenForwardOnly en el argumento type.
DbSQLPassThrough: Cuando el argumento source es una sentencia SQL, use esta
constante para pasar las sentencias SQL a una base conectada via ODBC, para
procesamiento. Si la usa con un dynaset, los datos no son actualizables (dynaset y
snapshot solamente).
dbConsistent (Default): Solo consiste en actualizaciones (solo dynaset). Se puede
usar esta constante con la constante dbInconsistent.
13
Manual del Alumno
DbInconsistent: Son permitidas actualizaciones inconsistentes. Esta es la contraria
de dbConsistent (solo dynaset). No se puede usar esta constante con la constante
dbConsistent.
Con el argumento lockedits del metodo OpenRecordset se puede controlar como
bloquear un objeto. Se pueden utilizar las siguientes constantes:
DbReadOnly: No se pueden realizar cambios. Esta constante reemplaza a la
constante dbReadOnly que fue usada en el argumento options en versiones
anteriores de DAO.
dbPessimistic (Default): Microsoft Jet usa bloqueo pesimista para determinar como
los cambios pueden ser hechos en un ambiente multiusuario.
DbOptimistic: Microsoft Jet usa bloqueo optimista para determinar como los
cambios pueden ser hechos en un ambiente multiusuario.
El valor por defecto es dbPessimistic. El solo efecto de usar dbPessimistic o
dbOptimistic es para configurar el valor de los objetos con la propiedad LockEdits.
14
Manual del Alumno
Creando un Objeto desde un Formulario:
Usted puede crear un objeto Recordset basado en un formulario de Microsoft Access.
Para hacer esto, use la propiedad RecordsetClone del formulario. Este crea un Recordset
tipo dynaset que se refiere a la misma consulta o “data” del fomulario. Si el formulario está
basado en una consulta, hay que referirse a la propiedad RecordsetClone. Esto es el
equivalente a crear un dynaset con la misma consulta. Se puede utilizar la propiedad
RecordsetClone cuando necesite aplicar un metodo provisto de acceso a todos los metodos y
propiedades que se pueden utilizar con un dynaset. La sintaxis para la propiedad
RecordsetClone es:
Set variable = form.RecordsetClone
El argumento variable es el nombre de un objeto Recordset existente. El argumento
form es el nombre de un formulario Microsoft Access. El siguiente ejemplo muestra como
asignar a un objeto Recordset los registros en el formulario “Orders”:
Dim rstOrders As Recordset
Set rstOrders = Forms!Orders.RecordsetClone
Este codigo siempre crea el tipo de Recordset siendeo cloneado (el tipo de Recordset
basado en el formulario); otros tipos no están disponibles.
Creando un Objeto desde una Tabla:
El metodo que se use para crear un objeto Recordset desde una tabla depende de si la
tabla es local a la base de datos actual o si está ligada a otra tabla en otra base de datos. La
siguiente discusion explica las diferencias y provee de ejemplos para cada tipo de tabla:
Creando un Recordset desde una tabla en una base de datos local Microsoft Access:
El siguiente ejemplo usa el metodo OpenRecordset para crear una tabla tipo
Recordset:
Objeto para una tabla en la base de datos actual:
Dim dbs As Database, rstCustomers As Recordset
Set dbs = CurrentDb
Set rstCustomers = dbs.OpenRecordset("Customers")
Notese que no es necesario usar la constante dbOpenTable para crear una tabla de tipo
Recordset. Si se omite la constante type DAO selecciona el tipo de Recordset de mayor
funcionalidad, dependiendo del objeto, en cuanto el Recordset creado y el origen de datos.
Para cuando el tipo de tabla está disponible y se abre un Recordset, DAO usa
automáticamente este tipo.
15
Manual del Alumno
Creando un Recordset desde una tabla ligada a una tabla de un formato diferente:
El siguiente ejemplo crea un Recordset de tipo dynaset, para una tabla Paradox® 3.X.
Pero el tipo tabla no está disponible cuando se abre un Recordset desde una tabla ligada a
una base de datos que no sea Microsoft Access. DAO selecciona el tipo próximo de
Recordset que sea más eficiente:
Dim dbs As Database
Dim tdf As TableDef
Dim rstTableData As Recordset
' Obtiene la base de datos activa.
Set dbs = CurrentDb
Set tdf = dbs.CreateTableDef("PDXAuthor")
' Conecta a la base de datos la tabla Paradox Author.
' C:\PDX\Publish.tdf.Connect = "Paradox 3.X;DATABASE=C:\PDX\Publish"
tdf.SourceTableName = "Author"
' Liga la tabla.
dbs.TableDefs.Append tdf
' Crea un Recordset tipo dynaset-type para la tabla.
Set rstTableData = tdf.OpenRecordset()
Se puede abrir directamente una tabla Paradox, abriendo inicialmente la base de datos
Paradox.
Usando un Indice en un Recordset de tipo tabla:
Se pueden ordenar registros en un Recordset de tipo “table” configurando la propiedad
Index. Cualquier objeto Index en la colección Indexes para los objetos tipo Recordset
pueden ser especificados en la propiedad Index.
El ejemplo siguiente crea un Recordset de tipo “table” basado en la tabla Customer y
utilizando un indice existente llamado City:
Dim dbs As Database, rstTableData As Recordset
Set dbs = CurrentDb
Set rstTableData = dbs.OpenRecordset("Customers", dbOpenTable)
' Mueve el puntero al primer registro de la tabla
rstTableData.MoveFirst
' El primer registro sin indice
MsgBox rstTableData!CompanyName
rstTableData.Index = "City"
' Selecciona el indice City.
rstTableData.MoveFirst
„ Mueve el puntero al primer registro.
MsgBox rstTableData!CompanyName
rstTableData.Close
Si se utiliza la propiedad Index para un indice que no existe, un error de ejecucion
capturable ocurre. Si se desea ordenar registros de acuerdo a un índice que no existe, se
16
Manual del Alumno
puede crear el indice primero o crear un Recordset de tipo dynaset o snapshot, usando una
consulta que retorne registros en un orden establecido.
Es importante utilizar la propiedad Index antes de usar el metodo Seek.
Creando un objeto Recordset desde una consulta:
Se puede crear un objeto Recordset basado en una consulta almacenada o no. En el
siguiente ejemplo se muestra la lista de productos en una consulta almacenada en la base de
datos activa:
Dim dbs As Database, rstProducts As Recordset
Set dbs = CurrentDb
Set rstProducts = dbs.OpenRecordset("Current Product List")
Si la consulta almacenada no existe, el metodo OpenRecordset tambien acepta una
cadena SQL. El ejemplo anterior puede ser reescrito tal como sigue:
Dim dbs As Database, rstProducts As Recordset
Dim strQuerySQL As String
Set dbs = CurrentDb
strQuerySQL = "SELECT * FROM Products WHERE Discontinued = No " _
& "ORDER BY ProductName;"
Set rstProducts = dbs.OpenRecordset(strQuerySQL)
La desventaja de esta última es que la consulta basada en una cadena debe ser
compilada cada vez que se ejecuta, mientras que una almacenada solo lo hace la primera
vez. Esto mejora el rendimiento.
Cuando se crea un objeto Recordset, usando una cadena SQL o una consulta
almacenada, el codigo de la misma no necesita seguir ejecutandose hasta que la consulta
retorna la primera fila en el Recordset.
Ordenando y Filtrando Registros:
Si se utiliza un Recordset de tipo “Table” y se utiliza la propiedad Index se puede estar
seguro que los registros apareceran en un orden especifico. Sin embargo, usualmente se
necesita extraer registros en un especifico orden en otro tipo de Recordset. Por ejemplo, se
puede necesitar visualizar las facturas ordenadas por número, o clientes en orden alfabetico.
Para ordenar los registros en un objeto Recordset que no es una tabla, utilice la claúsula
Order By en la consulta. Tambien se puede filtrar los datos mediante la utilización de la
claúsula Where
El siguiente ejemplo muestra el uso de ámbos:
Dim dbs As Database, rstManagers As Recordset
Set dbs = CurrentDb
17
Manual del Alumno
Set rstManagers = dbs.OpenRecordset("SELECT FirstName, LastName FROM " _
& "Employees WHERE Title = 'Sales Manager' ORDER BY LastName")
Una desventaja de ejecutar una consulta SQL en un metodo OpenRecordset es que
éste deventaja de ser precompilado cada vez que se ejecuta. Por ende, si la consulta es usada
frecuentemente, se obtiene mayor rendimiento creando un procedimiento almacenado usando
la misma consulta y abriendo un objeto Recordset cada vez que se requiera la consulta, tal y
como se muestra a continuacion:
Dim dbs As Database
Dim rstSalesReps As Recordset
Dim qdf As QueryDef
Set dbs = CurrentDb
Set qdf = dbs.CreateQueryDef("SalesRepQuery")
qdf.SQL = "SELECT * FROM Employees WHERE Title = 'Sales Representative';"
Set rstSalesReps = dbs.OpenRecordset("SalesRepQuery")
Para una mayor flexibilidad y control en tiempo de ejecucion se pueden usar
parametros para determinar criterio de orden y filtro.
Recreando a consulta desde un objeto Recordset:
Se puede utilizar un objeto Recordset abierto desde un QueryDef creando un objeto
de este tipo.
Para hacer esto, se debe utilizar el metodo CopyQueryDef. Este es
usualmente utilizado en situaciones donde un la variable de un objeto Recordset creado
desde un objeto QueryDef es pasado a una funcion y ésta debe recrear el SQL equivalente a
la consulta y posiblemente modificarla.
Modificando una Consulta desde un objeto Recordset:
Se puede utilizar el metodo Requery en un Recordset de tipo dynaset o snapshot
cuando se requiere ejecutar una consulta cada vez que se cambian los parametros. Esto es
más conveniente que abrir un nuevo Recordset y se ejecuta más velozmente.
El siguiente ejemplo crea un objeto Recordset y llama a una funcion que usa el
metodo CopyQueryDef para extraer la cadena correspondiente a la consulta SQL. Este
entonces dispone al usuario para que pueda utilizar los parametros de la consulta. El codigo
usa el metodo Requery para ejecutar la consulta modificada por el usuario:
Sub AddQuery()
Dim dbs As Database
Dim qdf As QueryDef
Dim rstSalesReps As Recordset
Set dbs = CurrentDb
Set qdf = dbs.CreateQueryDef("SalesRepQuery")
qdf.SQL = "SELECT * FROM Employees WHERE Title = 'Sales Representative'"
set rstSalesReps = qdf.OpenRecordset()
18
Manual del Alumno
'Llama a la funcion para agrear el constraint.
AddQueryFilter rstSalesReps
' Retorna a la base de datos original.
dbs.QueryDefs.Delete "SalesRepQuery"
rstSalesReps.Close
End Sub
Function AddQueryFilter(rst As Recordset)
Dim qdf As QueryDef
Dim strNewFilter As String, strRightSQL As String
Set qdf = rst.CopyQueryDef
'"LastName LIKE 'D*'".
strNewFilter = InputBox("Digite nuevo criterio : ")
strRightSQL = Right(qdf.SQL, 1)
' Quita caracteres desde el final de la consulta, como sea necesario.
Do While strRightSQL = " " Or strRightSQL = ";" Or strRightSQL = vbCR Or _
strRightSQL = vbLF
qdf.SQL = Left(qdf.SQL, Len(qdf.SQL) - 1)
trRightSQL = Right(qdf.SQL, 1)
Loop
qdf.SQL = qdf.SQL & " AND " & strNewFilter & ";"
rst.Requery qdf
' Reconsulta el Recordset.
rst.MoveLast
' llena el Recordset.
' "Lastname LIKE 'D*'" podría retornar 2 registros.
MsgBox "Número de Registros encontrados = " & rst.RecordCount
End Function
Para usar el metodo Requery, la propiedad Restartable del objeto Recordset debe
estar en True. La propiedad Restartable es simpre True cuando el Recordset es creado
desde una consulta. No se puede reiniciar consultas pass-through. Se puede o no tener
disponibilidad para reiniciar consultas con tablas ligadas en otro formato de base de datos,
sino es la nativa Microsoft Access. Para determinar cuando un objeto puede reejecutar una
consulta chequee la propiedad Restartable.
Ordenamiento DAO y propiedades de filtro.
Otra manera de ordenar y filtrar objetos Recordset está para configurar las
propiedades de ordenamiento y filtrado del objeto DAO. Sin embargo, este es usualmente
más lento que incluir un criterio de ordenamiento o filtrado en la consulta original o
cambiando los parametros y ejecutando este de nuevo con el metodo Requery. Las
propiedades de ordenamiento y filtrado de DAO son usualmente utilizadas cuando se
necesitar utilizar un conjunto de resultados para un usuario, pero el origen de los datos no
está disponible para una nueva consulta. Por ejemplo, cuando un objeto Recordset tiene mas
de 100 registros. Usar el metodo CopyQueryDef es preferible para cuando esa cantidad es
inferior.
19
Manual del Alumno
Moviendose a traves de objetos Recordset:
Un objeto Recordset usualmente tiene una posición actual, a menudo es un registro.
Cuando se refiere a un campo en el RecordSet, se obtienen valores desde el registro en la
posicion actual. Sin embargo la posicion actual puede tambien ser la siguiente o la anterior.
En ciertos casos, la posicion actual es indefinida.
Se puede utilizar los siguientes métodos Move para movilizarse a traves de los
registros de un Recordset:
MoveFirst mueve al primer registro.
MoveLast mueve al último registro.
MoveNext mueve al registro próximo.
MovePrevious mueve al registro anterior.
Move [n] mueve n registros hacia delante.
Se pueden usar cualquiera de estos métodos en un Recordset de tipo Table, Dynaset o
Snapshot. En el tipo Forward-Only solo se puede usar el MoveNext y se pueden especificar
la cantidad de registros que se desea desplazar.
El siguiente ejemplo abre un objeto Recordset utilizando la tabla “Employees”, la cual
contiene todos los registros que tienen un valor Null en el campo ReportsToField. La
función entonces actualiza los registros para indicar que éstos son empleados temporales.
Para cada registro en el Recordset, el ejemplo cambia los campos Title y Notes y salva los
cambios con el método Update. Este usa el método MoveNext para moverse al registro
próximo:
Function UpdateEmployees()
Dim dbs As Database, rstEmployees As Recordset, strQuery As String
Dim intI As Integer
On Error GoTo ErrorHandler
Set dbs = CurrentDb
' Abre un Recordset con todos los registros desde la tabla Employees
' que tienen un valor Null en el camop ReportsTo.
strQuery = "SELECT * FROM Employees WHERE ReportsTo IS NULL;"
Set rstEmployees = dbs.OpenRecordset(strQuery, dbOpenDynaset)
' Si el Recordset está vacio, sale.
If rstEmployees.EOF Then Exit Function
intI = 1
With rstEmployees
Do Until .EOF
.Edit
![ReportsTo] = 5
![Title] = "Temporary"
![Notes] = rstEmployees![Notes] & "Temp #" & intI
20
Manual del Alumno
.Update
.MoveNext
intI = intI + 1
Loop
.Close
End With
ErrorHandler:
Select Case Err
Case 0
Exit Function
Case Else
MsgBox "Error " & Err & ": " & Error, vbOKOnly, "ERROR"
Exit Function
End Select
End Function
Cabe aclarar que este ejemplo es meramente ilustrativo, dado que el método Update
funcionaría mejor utilizando una sentencia SQL.
Detectando los límites de un objeto Recordset:
En un objeto Recordset si se intenta mover a una posición que no es la correcta, se
produce un error de ejecución. Por ejemplo, si se intenta mover al próximo registro estando
en el último o al anterior estando en el primero.
La propiedad EOF indica el fin del archivo y BOF el inicio del mismo. Se puede
acceder a estas propiedades para determinar el inicio o final del archivo y asi controlar el
movimiento del puntero. En ambos casos se pretende obtener si la propiedad es False o
True.
El siguiente ejemplo muestra como utilizar las propiedades BOF y EOF para detectar
el inicio y final de un Recordset. Este fragmento de codigo crea un Recordset de tipo table
basada en la tabla Orders. Se realizan movimientos a traves de los registros, primero desde
el inicio del Recordset hasta el final de la tabla y luego hacia el final.
Dim dbs As Database, rstOrders As Recordset
Set dbs = CurrentDb
Set rstOrders = dbs.OpenRecordset("Orders", dbOpenTable)
Do Until rstOrders.EOF
.
. ' Manipulación de los datos.
.
rstOrders.MoveNext
' Mueve el puntero al próximo registro.
Loop
rstOrders.MoveLast
' Mientras sea inicio del archivo
' Mueve el puntero al último registro.
21
Manual del Alumno
Do Until rstOrders.BOF
.
. ' Manipulación de los datos.
' Mueve el puntero al registro anterior.
rstOrders.MovePrevious
Loop
rstOrders.Close
' Cierra el Recordset.
Las propiedades BOF y EOF tienen las siguientes características:
Si el Recordset no contiene registros cuando se abre éste, tanto BOF como EOF son
True.
Cuando BOF o EOF son True, la propiedad True se mantienen hasta que exista un
movimiento hasta un registro existente, asignando False a BOF o EOF.
Cuando BOF o EOF es False, y solo el registro en el Recordset es eliminado, la
propiedad devuelve False hasta que se desplaze el puntero hasta otro registro.
En el momento en que se crea o abre un Recordset que contiene más de un registro,
el primero de ellos es el registro actual y tanto BOF y EOF son False.
Si el primer registro en el registro actual cuando use el método MovePrevious, BOF
es puesto en True. Si se utiliza MovePrevious mientras BOF es True, un error en
tiempo de ejecución ocurre.
Similarmente, moviéndose al último registro en el Recordset se cambia el valor de la
propiedad EOF a True. Si se utiliza el método MoveNext mientras EOF es True, un
error en tiempo de ejecución se presenta.
Contando el número de registros en un objeto Recordset:
Si puede conocer el número de registros en un objeto Recordset. Por ejemplo, si se
necesita crear un formulario que muestre cuantos registros existen en cada tabla de la base de
datos o los cambios presentados cuando se realizan inclusiones.
La propiedad RecordCount contiene el número de registros en un Recordset tipo tabla o
el total de registros accesados en un dynaset o snapshot. Un objeto Recordset sin registros
posee la propiedad RecordCount en 0.
El siguiente ejemplo crea un Recordset tipo snapshot y determina el número de registros
en el mismo:
Function RecCount(strSQL As String) As Long
Dim rstCount As Recordset, dbs As Database
On Error GoTo ErrorHandler
Set dbs = CurrentDb
Set rstCount = dbs.OpenRecordset(strSQL)
If rstCount.EOF Then
22
Manual del Alumno
rstCount.Close
RecCount = 0
Exit Function
Else
rstCount.MoveLast
RecCount = rstCount.RecordCount
rstCount.Close
Exit Function
End If
ErrorHandler:
Select Case Err
Case 0
Exit Function
Case Else
MsgBox "Error:" & Err & ":" & Error, vbOKOnly, "ERROR"
Exit Function
End Select
End Function
Cuando se eliminan registros en un Recordset tipo dynaset, el valor de RecordCount
decrece. Sin embargo, en un ambiente multiusuario, los registros eliminados por otros
usuarios no son reflejados en el valor del RecordCount hasta que el registro acutal es
posicionado sobre un registro eliminado. Al mismo tiempo la configuración de la propiedad
RecordCount es actualizada. Utilizando el método Requery sobre un Recordset, seguido
por un metodo MoveLast, configuran la propiedad RecordCount al número total de
registros en el Recordset.
Un Recordset tipo snapshot es estático y el valor RecordCount de este no se cambia
cuando se agregan o eliminan registros.
Buscando la posición actual en un objeto Recordset:
Dos propiedades están disponibles para indicar la posición actual de un registro:
AbsolutePosition y PercentPosition.
El valor de la propiedad AbsolutePosition es la posición del registro actual relativo a
0. Sin embargo, no se piense como un número de registro; si el registro actual está
indefinido, la propiedad AbsolutePosition retorna –1. En consecuencia, esto no significa
que el registro tendrá la misma posicion absoluta si el objeto Recordset es recreado, porque
el orden de los registros individuales en un Recordset utilizando SQL que incluyan la
claúsula Order By cambia dichas posiciones.
La propiedad PercentPosition muestra la posicion actual expresada como un
porcentaje del total de registros indicados en la propiedad RecordCount. Para asegurarse la
correcta asignación del porcentaje referido utilicese los metodos MoveLast y MoveFirst
antes de abrir el Recordset.
23
Manual del Alumno
La propiedad PercentPosition es solamente una aproximacion y no debería ser
utilizada como un parametro critico. Esta propiedad es mejor para manejar un indicador que
marque el progreso en una operación de registros procesados.
El siguiente ejemplo abre un objeto Recordset en una tabla llamada Employees. El
procedimiento entonces permite moverse a traves de la tabla y usa la funcion SysCmd para
desplegar una barra de progreso, mostrando el porcentaje que ha sido procesada en la tabla.
Si la fecha de nacimiento del empleado es posterior al 1 Enero de 1993, el texto “Senior
Staff” es agregado al campo Notes.
Function PercentPos()
Dim dbs As Database, strMsg As String, rstEmployees As Recordset, intRet%
Dim intCount As Integer, strQuery As String, sngPercent As Single
Dim varReturn As Variant
Dim lngEmpID() As Long
On Error GoTo ErrorHandler
strQuery = "SELECT * FROM Employees;"
Set dbs = CurrentDb
Set rstEmployees = dbs.OpenRecordset(strQuery, dbOpenDynaset)
With rstEmployees
If .EOF Then
' Si no hay registros, sale.
Exit Function
Else
strMsg = "Procesando tabla Employees ..."
intRet = SysCmd(acSysCmdInitMeter, strMsg, 100)
End If
Do Until .EOF
If !HireDate < #1/1/93# Then
.Edit
!Notes = !Notes & ";" & "Senior Staff"
.Update
End If
If .PercentPosition <> 0 Then
intRet = SysCmd(acSysCmdUpdateMeter, .PercentPosition)
End If
.MoveNext
Loop
.Close
End With
intRet = SysCmd(acSysCmdRemoveMeter)
ErrorHandler:
Select Case Err
Case 0
24
Manual del Alumno
Exit Function
Case Else
MsgBox "Error " & Err & ": " & Error, vbOKOnly, "ERROR"
' limpia la métrica de progreso.
varReturn = SysCmd(acSysCmdSetStatus, " ")
Exit Function
End Select
End Function
Buscando un registro específico:
Muchas veces se desea encontrar un registro particular de un empleado, por ejemplo,
basados en el código del mismo, o todos los detalles que se especifican para una determinada
orden. En estos casos, se puede realizar una búsqueda utilizando los métodos Find y Seek.
Se puede utilizar el metodo Seek con tipos de Recordset Tables y el método Find con
dynaset y snapshot. Forward-Only no soporta el uso de Seek o cualquiera de los Find.
Buscando un registro en un Recordset tipo Table:
Se puede utilizar el metodo Seek para localizar registros en un Recordset tipo Table.
Cuando se usa el método Seek para localizar un registro, el motor de base de datos
Microsoft Jet usa los indices de la tabla, definidas por la propiedad Index.
Si se utiliza el método Seek en un Recordset tipo table, sin haber habilitado el indice
primero, ocurre un error en tiempo de ejecucion.
La sintaxis para el método Seek es el siguiente:
table.Seek comparison, key1, key2 ...
El argumento table es el Recordset tipo table utilizado.
El argumento comparación es una cadena que determina la clase de comparacion que
esta siendo aplicada. La siguiente tabla lista las cadenas de comparacion que se pueden
utilizar con el método Seek.
Cadena de comparación
Description
"="
">="
">"
"<="
"<"
Igual al valor especificado
Mayor o igual al valor especificado
Mayor al valor especificado
Menor o igual al valor especificado
Menor al valor especificado
25
Manual del Alumno
El argumento Keyn son una serie de uno o más valores que corresponden al campo o
campos que inicializan el indice actual del Recordset. Microsoft Jet compara estos valores
con los valores en los campos correspondientes del Recordset.
El ejemplo siguiente abre un Recordset tipo table llamada Employees y utiliza el
metodo Seek para localizar el record conteniendo el valor de IngEmpID en el campo
EmployeeID. Este returna la fecha de nacimiento del empleado referido:
Function intGetHireDate(lngEmpID As Long, varHireDate As Variant) As Integer
Dim rstEmployees As Recordset, dbs As Database
Const conFilePath$ = "C:\Program Files\Microsoft _ Office\Office\Samples\"
On Error GoTo ErrorHandler
Set dbs = OpenDatabase(conFilePath & "Northwind")
Set rstEmployees = dbs.OpenRecordset("Employees", dbOpenTable)
rstEmployees.Index = "PrimaryKey" 'Nombre del indice para Employee ID.
rstEmployees.Seek "=", lngEmpID
If rstEmployees.NoMatch Then
varHireDate = Null
'La constante conErrNoMatch, conSuccess, y conFailed están definidos
'en el nivel de módulo como una constante publica con valores enteros
'entre -32,761, 0, y -32,737 respectivamente.
intGetHireDate = conErrNoMatch
Exit Function
Else
varHireDate = rstEmployees!HireDate
intGetHireDate = conSuccess
Exit Function
End If
ErrorHandler:
Select Case Err
Case 0
Exit Function
Case Else
varHireDate = Null
intGetHireDate = conFailed
MsgBox "Error " & Err & ": " & Error, vbOKOnly, "ERROR"
Exit Function
End Select
End Function
El metodo Seek siempre inicia buscando registros desde el principio del Recordset.
Se puede utilizar la propiedad NoMatch en el Recordset para comprobar si el registro
buscado no fue encontrado, mediante examinación del valor True o False.
El siguiente ejemplo ilustra como se puede crear una función que utilice el método
Seek para localizar un registro utilizando un indice de multiples campos:
Function GetFirstPrice(lngOrderID As Long, lngProductID As Long) As Variant
Dim dbs As Database, rstOrderDetail As Recordset
26
Manual del Alumno
On Error GoTo ErrorHandler
Set dbs = CurrentDb
Set rstOrderDetail = dbs.OpenRecordset("Order Details", dbOpenTable)
rstOrderDetail.Index = "PrimaryKey"
rstOrderDetail.Seek "=", lngOrderID, lngProductID
If rstOrderDetail.NoMatch Then
GetFirstPrice = Null
MsgBox "No se encontró el registro..."
Else
GetFirstPrice = rstOrderDetail!UnitPrice
End If
rstOrderDetail.Close
ErrorHandler:
Select Case Err
Case 0
Exit Function
Case Else
MsgBox "Error " & Err & ": " & Error, vbOKOnly, "ERROR"
Exit Function
End Select
End Function
En este ejemplo, la llave primaria de la tabla consiste en dos campos: OrderID y
ProductID. Cuando llama a la función GetFirstPrice con una combinación válida de campos
para OrderID y ProductID, la función retorna el precio unitario desde el registro encontrado.
Si no se encuentra la combinación de valores de campo la función retorna un valor Null.
Buscando un registro en tipos de Recordset Dynaset- o Snapshot:
Se puede utilizar el método Find para localizar un registro en un Recordset tipo
Dynaset o Snapshot. DAO provee cuatro metodos para lo anterior:
FindFirst : busca el primer registro que satisface el criterio de búsqueda.
FindLast : busca el último registro que satisface el criterio de búsqueda.
FindNext: busca el próximo que satisface el criterio de búsqueda.
FindPrevious : busca el registro que satisface el criterio de búsqueda.
Nota: Para localizar un registro en un Recordset tipo table, debe utilizarse el método
Seek, descrito anteriormente.
Cuando se utilice el metodo Find desde especificarse el criterio de busqueda;
tipicamente es una expresion con un nombre de campo con un valor especifico.
Se pueden localizar los registros buscados en orden último, anterior o siguiente,
utilizando los métodos FindLast, FindPrevious y FindNext.
27
Manual del Alumno
DAO almacena True en la propiedad NoMatch cuando un metodo Find falla y el
registro actual yace indefinido. Para retornar al registro actual debe utilizarse el metodo
bookmark.
El ejemplo siguiente muestra como se puede utilizar el método FindNext para
encontrar todas las ordenes en la tabla Orders que tiene los registros que no corresponden en
la tabla Order Details y agrega los valores en el campo OrderID al arreglo IngOrderID():
Function FindEx(lngOrderID() As Long)
Dim dbs As Database, rstOrders As Recordset
Dim strQuery As String, rstOrderDetails As Recordset
Dim intIndex As Integer
On Error GoTo ErrorHandler
Set dbs = CurrentDb
'Abre un recordset con los registros de las tablas
'orders y Order Details.
'Si éstas no contienen registros sale.
strQuery = "SELECT * FROM Orders ORDER BY OrderID;"
Set rstOrders = dbs.OpenRecordset(strQuery, dbOpenSnapshot)
If rstOrders.EOF Then Exit Function
strQuery = "SELECT * FROM [Order Details] ORDER BY OrderID;"
Set rstOrderDetails = dbs.OpenRecordset(strQuery, dbOpenSnapshot)
' Para el primer registro en Orders, encuentra el primer registro.
' en OrderDetails. Si no no encuentra, redimensiona el arreglo de order IDs
' y agrega el order ID al arreglo.
rstOrderDetails.FindFirst "OrderID = " & rstOrders![OrderID]
If rstOrderDetails.NoMatch Then
ReDim Preserve lngOrderID(1 To intIndex)
lngOrderID(intIndex) = rstOrders![OrderID]
End If
' La primera búsqueda ha sido exitosa, entonces usa FindNext para encontrar
' próximo registro que satisfaga el criterio de búsqueda.
intIndex = 0
Do Until rstOrders.EOF
rstOrderDetails.FindNext "OrderID = " & rstOrders![OrderID]
If rstOrderDetails.NoMatch Then
intIndex = intIndex + 1
ReDim Preserve lngOrderID(1 To intIndex)
lngOrderID(intIndex) = rstOrders![OrderID]
End If
rstOrders.MoveNext
Loop
ErrorHandler:
Select Case Err
Case 0
Exit Function
28
Manual del Alumno
Case Else
MsgBox "Error " & Err & ": " & Error, vbOKOnly, "ERROR"
Exit Function
End Select
End Function
Si la búsqueda de registros es frecuente en un Recordset tipo dynaset, se pueden
encontrar éstos facilmente creando una tabla indexada temporal y usar el método Seek.
Marcando la posición de un registro con Bookmarks:
Un bookmark es un sistema generado que identifica univocamente a cada registro. La
propiedad bookmark de DAO en un Recordset cambia cada vez que se mueve a un nuevo
registro. Para identificar un registro, se marca el registro actual con Bookmark en una
variable tipo Variant. Para retornar a éste registro, asignele al bookmark el valor de esta
variable.
El siguiente ejemplo ilustra como se puede usar bookmark para salvar la posicion
actual del registro. Se pueden realizar otras operaciones sobre el recordset y posteriormente
regresar al registro marcado.
Function BookMarkEx() As Integer
Dim dbs As Database, rstProducts As Recordset
Dim vBookMark As Variant, sngRevenue As Single
Dim strQuery As String, rstCategories As Recordset, strCriteria As String
On Error GoTo ErrHandler
BookMarkEx = 0
strQuery = "SELECT * FROM Products WHERE UnitsOnOrder >= 40 _
ORDER BY " _
& "CategoryID, UnitsOnOrder DESC;"
Set dbs = CurrentDb
Set rstProducts = dbs.OpenRecordset(strQuery, dbOpenSnapshot)
Set rstCategories = dbs.OpenRecordset("SELECT CategoryID FROM " _
& "Categories ORDER BY CategoryID;", dbOpenSnapshot)
If rstProducts.NoMatch Then Exit Function
' Por cada categoría encontrada el producto genera el ingreso mínimo
' y el producto de mayor precio.
Do Until rstCategories.EOF
strCriteria = "CategoryID = " & rstCategories![CategoryID]
rstProducts.FindFirst strCriteria
sngRevenue = rstProducts![UnitPrice] * rstProducts![UnitsOnOrder]
If Not rstProducts.NoMatch Then
'Inicializa el bookmark en el primer registro conteniendo la
29
Manual del Alumno
'CategoryID.
vBookMark = rstProducts.Bookmark
'Buscar el producto generando el de mayor precio
Do While rstProducts![CategoryID] = rstCategories![CategoryID]
If rstProducts![UnitPrice] * rstProducts![UnitsOnOrder] > sngRevenue Then
sngRevenue = rstProducts![UnitPrice] * _
rstProducts![UnitsOnOrder]
End If
rstProducts.MoveNext
Loop
' Se mueve al primer registro que contiene el campo CategoryID.
rstProducts.Bookmark = vBookMark
sngRevenue = rstProducts![UnitPrice] * _
rstProducts![UnitsOnOrder]
' Encuentra el producto, generando el de menor precio/ingreso.
Do While rstProducts![CategoryID] = rstCategories![CategoryID]
If rstProducts![UnitPrice] * rstProducts![UnitsOnOrder] < sngRevenue Then
sngRevenue = rstProducts![UnitPrice] * rstProducts![UnitsOnOrder]
End If
rstProducts.MoveNext
Loop
End If
rstCategories.MoveNext
Loop
' Error Handler.
ErrHandler:
Select Case Err
Case 0
Exit Function
Case Else
MsgBox "Error " & Err & ": " & Error, vbOKOnly, "ERROR"
Exit Function
End Select
End Function
Un “bookmark” es particularmente útil si un método falla, por que la posición actual
del registro se pierde.
La propiedad LastModified del objeto Recordset clarifica el como usar el bookmark.
Esta propiedad retorna el bookmark del ultimo registro en el Recordset a ser añadido o
modificado. Para usar este, se debe configurar la propiedad Bookmark de DAO para que
sea igual a la propiedad LastModified, tal y como se muestra a continuación:
RstClientes.BookMark = rstClientes.LastModified
30
Manual del Alumno
Lo anterior mueve el registro actual hasta el último que fue agregado o modificado.
Esto es particularmente útil cuando se agregan nuevos registros, por que después de hacerlo
el registro actual es el que se añadió. Con la propiedad LastModified, se puede mover al
registro recién agregado.
Cuando usted cierra un objeto Recordset, cualquier bookmarks que haya salvado se
invalida. No se puede usar un bookmark de un Recordset en otro. Sin embargo, puede
usarse como en el siguiente ejemplo:
Dim dbs As Database
Dim rstOriginal As Recordset, rstDuplicate As Recordset
Dim strPlaceholder As String
Set dbs = CurrentDb
' Crea el primer Recordset.
Set rstOriginal = dbs.OpenRecordset("Orders", dbOpenDynaset)
' Guarda la posición del registro actual
strPlaceholder = rstOriginal.Bookmark
' Crea un Recordset duplicado.
Set rstDuplicate = rstOriginal.Clone()
' Se desplaza al mismo registro.
rstDuplicate.Bookmark = strPlaceholder
rstOriginal.Close
Que objetos Recordset no soportan Bookmarks?
Los Recordset de tipo Dynaset basados en ciertas tablas ligadas, como tablas Paradox
que no tienen llaves primarias no soportan bookmarks. Se puede determinar si un objeto
Recordset soporta o no bookmarks chequeando el valor de la propiedad bookmarkable, tal y
como se muestra en el ejemplo siguiente:
If rstLinkedTable.Bookmarkable Then
MsgBox "Esta tabla soporta bookmarks."
Else
MsgBox "Esta tabla no soporta bookmarks."
End If
Cambiando datos:
Despues que se crea un Recordset tipo table o dynaset, se pueden modificar, eliminar o
agregar nuevos registros. Lo anterior no se puede realizar si el Recordset es snapshot, o
forward-only.
Usando consultas parametrizadas:
31
Manual del Alumno
Una consulta parametrizada es aquella que cuando se ejecuta despliega una caja de
diálogo que le indica al usuario que debe digitar información, como por ejemplo un criterio
para la búsqueda de registros o valores para insertar en un campo. Se pueden utilizar
procedimientos almacenados para brindar procesos de mantenimiento de bases de datos o
para recuperar consultas de uso intensivo.
Para crear un procedimiento o consulta almacenada deben realizarse los siguientes
pasos:
 Cree una consulta almacenada, especificando los parámetros que el usuario
necesita proveer.
 Cuando se abre un Recordset utilizando esta consulta, la aplicación abre una caja
de diálogo que solicita al usuario los valores para los parámetros establecidos en la
consulta.
El siguiente ejemplo toma dos cadenas que representan fechas y crea una consulta
parametrizada que retorna todos aquellos registros en la tabla Orders cuyo campo Order Date
esté entre las dos fechas. Este añade todos los valores del campo OrderId retraido en la
consulta y los almacena en un arreglo:
Function OrdersFromTo(strDateFrom As Variant, strDateTo As Variant, _
lngOrderIDs() As Long)
Dim dbs As Database, rstOrders As Recordset
Dim qdf As QueryDef, strSQL As String, intI As Integer
On Error GoTo ErrorHandler
Set dbs = CurrentDb
strSQL = "PARAMETERS [DateFrom] DateTime, [DateTo] DateTime; "
strSQL = strSQL & "SELECT * FROM Orders WHERE OrderDate BETWEEN
"
strSQL = strSQL & "[DateFrom] AND [DateTo];"
' Crea una consulta almacenada parametrizada.
Set qdf = dbs.CreateQueryDef("", strSQL)
' configura los parámetros de la consulta.
qdf.Parameters("DateFrom") = strDateFrom
qdf.Parameters("DateTo") = strDateTo
'Abre un Recordset forward-only snapshot.
Set rstOrders = qdf.OpenRecordset(dbOpenSnapshot, dbForwardOnly)
' Carga en un arreglo todos los registros del campo OrderIDs
' obtenidos en la consulta
intI = 1
While rstOrders.EOF = False
ReDim lngOrderIDs(1 To intI)
32
Manual del Alumno
lngOrderIDs(intI) = rstOrders!OrderID
intI = intI + 1
rstOrders.MoveNext
Wend
ErrorHandler:
Select Case Err
Case 0
Exit Function
Case Else
MsgBox "Error " & Err & ": " & Error, vbOKOnly, "ERROR"
Exit Function
End Select
End Function
Para más información veáse en la ayuda: “parameter queries.”
Marcando cambios Bulk:
Muchos de los cambios que se pueden realizar en una base de datos pueden
eficientizarse utilizando consultas definidas. El ejemplo siguiente crea un objeto QueryDef
para actualizar la tabla Employees:
Dim dbs As Database, qdfChangeTitles As QueryDef
Set dbs = CurrentDb
Set qdfChangeTitles = dbs.CreateQueryDef("")
qdfChangeTitles.SQL = "UPDATE Employees SET Title = 'Account Executive' " _
& "WHERE Title = 'Sales Representative';"
qdfChangeTitles.Execute dbFailOnError
' llama a la consulta.
Se puede reemplazar toda la cadena SQL en este ejmplo con una consulta almacenada
parametrizable. El siguiente ejemplo muestra como el código anterior puede ser reescrito
como una consulta almacenada y no como consulta definida:
Dim dbs As Database, qdfChangeTitles As QueryDef
Dim strSQLUpdate As String, strOld As String
Dim strNew As String
Set dbs = CurrentDb
strSQLUpdate = "PARAMETERS [Old Title] Text, [New Title] Text; " _
& "UPDATE Employees SET Title = [New Title] WHERE Title = [Old
Title];"
' Crea el objeto QueryDef.
Set qdfChangeTitles = dbs.CreateQueryDef("", strSQLUpdate)
33
Manual del Alumno
' Solicita al usuario el valor anterior.
strOld = InputBox("Digite el puesto anterior")
' Solicita al usuario el nuevo valor.
strNew = InputBox("Digite el nuevo puesto")
' Configura los parametros.
qdfChangeTitles.Parameters("Old Title") = strOld
qdfChangeTitles.Parameters("New Title") = strNew
' Invoca la consulta.
qdfChangeTitles.Execute
Nota: una consulta de eliminación en más eficiente que el código que busca ciclicamente a
través de un conjunto de registros, especialmente con bases de datos creadas con Access para
Windows 95 o superior.
Modificando un registro existente:
Se pueden modificar registros existentes en un Recordset tipo table o dynaset usando
los métodos Edit y Update.
Para modificar un registro existente en un Recordset table o dynaset:
1.
2.
3.
4.
Localizar el registro que se quiere modificar.
Use el método Edit para preparar el registro actual para edición.
Realice los cambios necesarios al registro.
Use el método Update para salvar los cambios hechos.
El siguiente ejemplo ilustra como cambiar el título del puesto para todos los
representantes de ventas contenidos en una tabla llamada Employees.
Dim dbs As Database, rstEmployees As Recordset
Set dbs = CurrentDb
Set rstEmployees = dbs.OpenRecordset("Employees")
rstEmployees.MoveFirst
Do Until rstEmployees.EOF
If rstEmployees!Title = "Sales Representative" Then
rstEmployees.Edit
rstEmployees!Title = "Account Executive"
rstEmployees.Update
End If
rstEmployees.MoveNext
34
Manual del Alumno
Loop
rstEmployees.Close
Es importante notar que si se utiliza el método Edit antes que se realicen los cambios
ocurre un error de ejecución. Si se edita el registro actual y entonces se mueve el puntero de
registros o se cierra el Recordset sin haber primeramente ejecutado un metodo Update, los
cambios realizados no surtirán efecto.
Se puede finalizar el método Edit y cualquier otra transacción pendiente sin salvar los
cambios utilizando el método CancelUpdate.
Actualizaciones inconsistentes:
Un Recordset tipo dynaset puede estar basados en una consulta de múltiples tablas con
una relación tipo uno a muchos. Por ejemp.lo, supongase que se desea crear una consulta de
múltiples tablas que combinan campos de las tablas Orders y Order Details. Generalmente
hablando no se pueden cambiar valores en la tabla Orders por que están subordinados a los
valores Order Details. Los cambios en estos casos tienen que realizarse en ámbas tablas para
que no se presenten inconsistencias. En tal caso puede utilizarse la constante dbInconsistent
del método OpenRecordset para crear un dynaset inconsistente.
Por ejemplo:
Set rstTotalSales = dbs.OpenRecordset("Sales Totals" ,, _ dbInconsistent)
Cuando se actualiza un dynaset inconsiste, se puede facilmente destruir la integridad
referencial de los datos en el dynaset. Por ende, debe procurarse realizar los cambios en
todas aquellas tablas que mantengan esta integridad referencial.
La constante dbInconsisten está disponible solo para los objetos Recordset de tipo
dynaset. Este es ignorado para los de tipo table, snapshot y forward-only.
Elimimando un registro existente:
Se puede eliminar un registro existente en un Recordset de tipo table o dynaset
utilizando el método Delete. No se pueden eliminar registros de un snapshot.
El ejemplo siguiente muestra como eliminar todos los registros duplicados en la tabla
Shippers:
Function DeleteDuplicateShippers() As Integer
Dim rstShippers As Recordset, strQuery$, dbs As Database, strName$
On Error GoTo ErrorHandler
strQuery = "SELECT * FROM Shippers ORDER BY CompanyName;"
Set dbs = CurrentDb
Set rstShippers = dbs.OpenRecordset(strQuery, dbOpenDynaset)
' Si no hay registros en la tabla Shippers table, sale.
35
Manual del Alumno
If rstShippers.EOF Then Exit Function
strName = rstShippers![CompanyName]
rstShippers.MoveNext
Do Until rstShippers.EOF
If rstShippers![CompanyName] = strName Then
rstShippers.Delete
Else
strName = rstShippers![CompanyName]
End If
rstShippers.MoveNext
Loop
ErrorHandler:
Select Case Err
Case 0
' La constante conSuccess y conFailed son definidas en
' el nivel de módulo como constante pública con valor entero.
' con valores de 0 y -32,737 respectivamente.
DeleteDuplicateShippers = conSuccess
Exit Function
Case Else
MsgBox "Error " & Err & ": " & Error, vbOKOnly, "ERROR"
DeleteDuplicateShippers = conFailed
Exit Function
End Select
End Function
Cuando se usa el metodo Delete, Microsoft Jet inmediatamente elimina el registro
actual (después de ejecutar la búsqueda, por supuesto), sin presentar ninguna advertencia al
respecto. Si se elimina el registro actual el motor no provee un desplazamiento automático al
siguiente o al registro anterior en el Recordset, por tanto debe utilizarse el método MoveNext
para realizarlo.
Si se intenta accesar un registro previamente eliminado en el Recordset tipo table se
puede presentar una condición de error: 3167 “Registro eliminado”. En un dynaset se puede
obtener el error 3021 “No registro actual”.
Agregando un nuevo registro:
Se puede agregar un nuevo registro a un Recordset tipo table o dynaset utilizando el
método AddNew.
Para agregar un nuevo registro a un Recordset table o dynaset:
1. Use el método AddNew para crear un nuevo registro.
2. Asigne valores para cada campo del registro.
3. Use el método Update para guardar los datos o CancelUpdate para cancelar.
36
Manual del Alumno
El siguiente ejemplo agrega un nuevo registro en el Recordset tipo tabla de nombre
rstShippers:
Dim dbs As Database, rstShippers As Recordset
Set dbs = CurrentDb
Set rstShippers = dbs.OpenRecordset("Shippers")
rstShippers.AddNew
rstShippers!CompanyName = "Global Parcel Service"
.
. ' Set remaining fields.
.
rstShippers.Update
rstShippers.Close
Cuando se utiliza el método AddNew, Microsoft Jet prepara un nuevo registro en
blanco y lo habilita como el registro actual. Si se añade un registro a un Recordset de tipo
dynaset este nuevo registro aparece al final del Recordset. Para forzarlo a que aparezca en la
posición adecuada de un ordenamiento se debe recrear el Recordset mediante el método
Requery. Si se agrega un registro a un Recordset tipo table, el registro aparece posicionado
de acuerdo al indice actual, o al final de la tabla si no hay un índice abierto.
Transacciones Microsoft Jet:
Una transacción es un conjunto de operaciones realizadas que son tratadas como una
sola unidad de trabajo. El trabajo en una transacción pueden ser completadas como un todo;
si una parte de la misma falla toda la transacción falla. Las transacciones ofrecen al
desarrollador de software la posibilidad para reforzar la integridad de la informacion.
Cuando múltiples tablas son involucradas en procesos de actualización, por ejemplo, pueden
surgir fallas que pueden crear inconsistencias peligrosas de información.
El ejemplo más común de transacciones son las que atañen a la automatización
bancaria de algunos procesos, Automated Teller Machine (ATM). El proceso de dispensar
dinero y el débito y crédito de cuentas son consideradas una unidad lógica de trabajo y que
son tratadas como una transacción: el dinero no puede ser dispensado sin que el sistema esté
disponible para realizar el débito de la cuenta.
Utilizando Transacciones en aplicaciones:
Microsoft Jet soporta transacciones a traves de DAO: BeginTrans, CommitTrans y
Rollback. La sintaxis básica de estos métodos se muestran en la siguiente tabla:
Método
Workspace.BeginTrans
Workspace.CommitTrans
Operación
Inicia la transacción
Después de la transacción escribe y actualiza permanente a los
objetos relacionados en la base de datos.
37
Manual del Alumno
Workspace.Rollback
Cancela las transacciones, por error o por decisión del usuario.
El siguiente ejemplo cambia los títulos de oficio de todos los representantes de ventas
en la tabla Employees de la base de datos Northwind. Después del método BeginTrans
inicia una transacción que realiza los cambios respectivos en la tabla Employees, el método
CommitTrans guarda los cambios realizados. Se puede utilizar el método Rollback para
cancelar los cambios realizados con el Update y restaurar la tabla a su estado original.
Sub ChangeTitle()
Dim dbsSales As Database
Dim rstEmp As Recordset
Dim wrkCurrent As Workspace
Set wrkCurrent = DBEngine.Workspaces(0)
Set dbsSales = OpenDatabase("Northwind.mdb")
Set rstEmp = dbsSales.OpenRecordset("Employees", dbOpenTable)
wrkCurrent.BeginTrans
Do Until rstEmp.EOF
If rstEmp!Title = "Sales Representative" Then
rstEmp.Edit
rstEmp!Title = "Sales Associate"
rstEmp.Update
End If
rstEmp.MoveNext
Loop
If MsgBox("Guarda todos los cambios?", vbQuestion + vbYesNo) = vbYes Then
wrkCurrent.CommitTrans
Else
wrkCurrent.Rollback
End If
rstEmp.Close
dbsSales.Close
End Sub
Cuando se utilizan transacciones, todas las bases de datos y los objetos Recordset en el
Workspace especificado son afectados. Esto es, las transacciones afectan al workspace
global no a una base de datos o recordset específico. Asimismo, los métodos asociados
tienen su incidencia a traves de todo el workspace: CommitTrans y Rollback.
Nota: Se pueden utilizar los métodos BeginTrans, CommitTrans y Rollback con el objeto
DBEngine. En este caso, la transacción es aplicada al workspace por defecto, como:
DBEngine.Workspaces(0).
Manejando Transacciones:
Microsoft Jet utiliza algoritmos sofisticados para proveer transacciones seguras y
confiables.
38
Manual del Alumno
Tamaño de transacciones:
El tamaño de las transacciones que se pueden realizar con Visual Basic, utilizando el
motor Jet, está limitada únicamente por la cantidad física de espacio en el disco duro en que
se realicen. Esto es, Microsoft Jet puede almacenar una cantidad de transacciones tan grande
como lo sea el medio de almacenamiento donde se realice. Si por ejemplo, en la ejecución
de una transacción se agota el espacio físico en disco se produce un error interceptable
(2004) y reacciona de acuerdo al código de la aplicación. El caso es que si la aplicación
realiza el commit después de ocurrido este error, es factible que Microsoft Jet no pueda
regenerar la base de datos por tanto no se puede determinar el número de registros afectados
(olvidemos el dbrecordaffected), poniendo a esta en un estado inconsistente. Si por el
contrario, se realiza un rollback de los cambios la base de datos queda en su estado original.
Niveles Transaccionales:
Se pueden tener hasta 5 niveles de transacciones activas al mismo tiempo. Esto es, se
pueden crear hasta 5 niveles transaccionales en una misma ejecución. Si se desea tener
transacciones con “overlapping”, “nonnested scopes” se pueden abrir objetos workspace
adicionales y manejar dichas transaccionales en dichos workspaces.
Cuando una transacción es “Rolled Back” por el motor de bases de datos Jet:
Si se cierra un objeto Workspace, cualquier transacción es automaticamente “Rolled
Back”. Microsoft Jet nunca realiza un Commit automáticamente sin haberla iniciado.
Asimismo, cuando se cierra una base de datos, variables atinentes o transacciones sin
committ sufren el mismo efecto.
Transacciones en origenes de datos externos:
Las transacciones no son soportadas en origenes de datos externas ( a excepción de
datos mediante ODBC). Por ejemplo, si la base de datos tiene ligas a tablas de FoxPro® o
dBASE®, las transacciones a estos objetos son ignorados. Para determinar o no si una base
de datos u objeto Recordset soporta transacciones se puede chequear el valor de la propiedad
Transactions. Un valor True indica que si lo soporta.
Transacciones y rendimiento:
En versiones anteriores de Microsoft Access era generalmente recomendado que se
usaran transacciones como un factor de mejoramiento en el rendimiento de aplicaciones.
Hoy día todas las transacciones para DAO (agregación, modificación y eliminación) son
mejoradas en su rendimiento internamente y automaticamente. En muchas situaciones, este
soporte automático provee a las aplicaciones el mejor rendimiento posible. Sin embargo,
muchas veces se debe recurrir a configurar algunas variables en el registro de Windows para
mejorar en este aspecto.
39
Manual del Alumno
Extrayendo Datos desde un registro:
Después de que se ha localizado un registro particular, se puede necesitar extraer datos
para usar en un aplicación, sea quiza para modificarlo.
Copiando un campo simple:
Se puede copiar un campo simple de un registro a una variable del tipo apropiado. El
ejemplo siguiente extrae tres campos del primer registro del objeto Recordset:
Dim dbs As Database, rstEmployees As Recordset
Dim strFirstName As String, strLastName As String
Dim strTitle As String
Set dbs = CurrentDb
Set rstEmployees = dbs.OpenRecordset("Employees")
rstEmployees.MoveFirst
strFirstName = rstEmployees!FirstName
strLastName = rstEmployees!LastName
strTitle = rstEmployees!Title
rstEmployees.Close
Copiando registros completos en un arreglo:
Para copiar uno o más registros completos, se puede crear un array bidimensional y
copiar los registros en él. Se incrementa el primer parámetro por cada campo y el segundo
por cada registro.
Una via rápida para hacer esto es con el método GetRows. El método GetRows retorna
un arreglo bidimensional. El primer parámetro identifica el campo y el segundo el número
de fila, tal y como se muestra en el ejemplo siguiente:
varRecords(intField, intRecord)
El siguiente ejemplo utiliza una sentencia SQL para extraer tres campos desde una tabla
llamada Employees y los coloca en un objeto Recordset. Este entonces utiliza el método
GetRows para extraer los primeros tres registros del Recordset y almacena los registros
seleccionados en un arreglo bidimensional. Imprime entonces cada registro, campo por
campo, usando los dos arreglos indexados para seleccionar campos especificos y registros.
Para aclarar este punto se ilustra como los arreglos indexados son utilizados. El
ejemplo utiliza una sentencia separada para identificar e imprimir cada campo de cada
registro. En la práctica, es mejor utilizar dos ciclos, uno en el otro, y proveer variables
enteras para los indices que navegan en ambas dimensiones del arreglo:
40
Manual del Alumno
Sub GetRowsTest()
Dim dbs As Database
Dim rstEmployees As Recordset
Dim varRecords As Variant
Dim intNumReturned As Integer
Dim intNumColumns As Integer
Dim intColumn As Integer, intRow As Integer
Set dbs = CurrentDb
Set rstEmployees = dbs.OpenRecordset("SELECT FirstName, LastName, Title" _
& " FROM Employees", dbOpenSnapshot)
varRecords = rstEmployees.GetRows(3)
intNumReturned = UBound(varRecords, 2) + 1
intNumColumns = UBound(varRecords, 1) + 1
For intRow = 0 To intNumReturned - 1
For intColumn = 0 To intNumColumns - 1
Debug.Print varRecords(intColumn, intRow)
Next intColumn
Next intRow
rstEmployees.Close
End Sub
Se puede utilizar subsecuentes llamadas al método GetRows si hay más registros
disponibles.
Usando Consultas sumarizadas:
Se puede sumarizar datos utilizando las claúsulas GROUP BY y HAVING. GROUP
BY organiza los datos dentro de grupos; HAVING configura condiciones de grupos para
incluir en el resultado. Estas claúsulas pueden estar juntas, HAVING usada sin GROUP BY
puede producir resultados confusos.
Usando Eventos del Control Data:
Tres eventos se pueden utilizar en aplicaciones de bases de datos, utilizando Visual
Basic: Validate, Error y Reposition.
Evento Validate:
Este evento se utiliza para comprobar los datos del RecordSet antes de guardar el
registro en la base de datos. Este evento se produce antes de que Visual Basic escriba los
datos en la base de datos. Es útil para solicitar al usuario confirme los cambios a realizar en
la base de datos.
La sintaxis de este evento es la siguiente:
Private Sub Data1_Validate (index As Integer, action As Integer, save As Integer)
El argumento action indica la operación que provocó el evento Validate. Este evento
es producto de realizar las siguientes operaciones: MoveFirst, MovePrevious, MoveNext,
41
Manual del Alumno
MoveLast, AddNew, Update, Delete, Find, establecer la propiedad BookMark, cerrar la
base de datos, descargar el formulario. Para cancelar cualquiera de estas acciones, se debe
asignarsele vbDataActionCancel al argumento action.
El argumento save indica si va a guardarse o no el registro. Si save es True, los datos
enlazados han cambiado. Para cancelar la acción de guardar puede asignarse False al
argumento save.
El código siguiente solicita al usuario que confirme los cambios realizados en la base
de datos. Si el usuario indica que no, los cambios se cancelarán automáticamente:
Private Sub Data1_Validate (Action As Integer, Save As Integer)
Dim Respuesta As Integer
If Save = True Then
Respuesta = MsgBox ("¿Guardar cambios?", vbYesNo)
If Respuesta = vbNo Then
Save = False
Data1.UpdateControls ' Actualiza campos.
End If
End If
End Sub
Evento Reposition:
El evento Reposition se utiliza para modificar la apariencia de un formulario o realizar
una acción necesaria cuando se desplace a un nuevo registro. Esto se realiza cuando Visual
Basic se desplaza a otro registro o cuando se abre la base de datos.
Se puede variar la apariencia de un formulario mostrando información sobre cuántos
registros se encuentran en el Recordset activo. Para ello, debe utilizarse la propiedad
AbsolutePosition. El código siguiente ilustra como demostrar lo anterior:
Private Sub Data1_Reposition()
Data1.Caption = "Número de registro " & _
Data1.Recordset.AbsolutePosition + 1
End Sub
Evento Error:
Este evento se ejecuta cuanto el usuario interactua con el control data y se produce un
error de acceso a datos. En este caso debe agregarse una rutina personalizada de tratamiento
de errores.
Los valores contenidos en objetos enlazados a datos no cambian cuando se produce un
error. Puede asignar el valor 0 al argumento Response para que no se muestre el mensaje de
error. El siguiente ejemplo muestra lo anterior:
42
Manual del Alumno
Private Sub Data1_Error(DataErr As Integer, Response As Integer)
If DataErr = 3022 Then 'Error de clave duplicada
MsgBox "Escriba un número de Id. de empleado único"
txtEmpID.SetFocus
Response = 0
Else
Response = 1 'muestra el mensaje de error estándar
End If
End Sub
Enlazando datos a Controles ActiveX:
Un control ActiveX es un objeto que permite interactuar a un usuario con una
aplicación. Estos objetos tienen eventos que se pueden utilizar. La extensión de estos
objetos ActiveX es .OCX.
Un ejemplo clásico, y muy utilizado, es el objeto Cuadrícula (DbGrid). Este objeto
permite mostrar datos de una base de datos en forma de cuadrícula, con lo que le permite al
usuario manipular varios registros a la vez. Este control posee la propiedad DataSource que
permite guardar el nombre del origen de datos desde el cual se cargarán sus celdas. El
DBGrid permite ver y modificar registros simultáneamente.
Se puede utilizar la colección columns para recuperar el texto de la celda seleccionada
actualmente en tiempo de ejecución mediante el siguiente código:
MsgBox DBGrid1.Columns(DBGrid1.Col).Text
Para modificar la información cargada en el DBGrid del objeto Recordset asociado
puede utilizarse el siguiente código:
Data1.Recordset.Edit
Data1.Recordset.Fields("Campeon") = "L.D.A."
Data1.Recordset.Update
El evento BeforeUpdate se ejecuta antes de que se muevan los datos desde un DBGrid
al control data. En este caso puede validarse la información respectiva y cancelarse si es
necesario.
Un objeto que tiene similares particularidades al DBGrid es el MSFlexGrid. Sin
embargo, este control solo permite datos de captura. Puede almacenar en el datos
provenientes de un data control o de una consulta mediante SQL. Muchas veces es necesario
combinar celdas, tal y como se realiza en Excel. MSFlexGrid permite realizar esto
asignando a la propiedad MergeCells un valor distinto de cero y posteriormente asignar las
propiedades de matrices MergeRow() y MergeCol() el valor True para las filas que se
deseen combinar. El siguiente ejemplo muestra como combinar las celdas de la tabla
Facturas:
43
Manual del Alumno
Private Sub Form_Load()
MSFlexGrid1.MergeCells = flexMergeFree
MSFlexGrid1.MergeCol(1) = True
End Sub
Un control de cuadro de lista enlazado a datos (DBList) o el cuador combinado
enlazado a datos (DBCombo) se utilizan para presentar un listado de datos de un Recordset.
Esto es útil cuando se desea validar datos por parte del usuario. Por ejemplo, si se desea
enviar una notificación a un cliente, se puede utilizar un control DBList para presentar la
lista de todos los nombres de clientes válidos y obtener asi su ID.
En un DBCombo para asignar datos debe configurarse la propiedad RowSource con el
nombre de un control data existente y en la propiedad ListField el nombre de un campo.
Si se debe actualizar un campo de la base de datos debe establecerse las propiedades
DataSource y DataField con lo requerido. Para establecer la relación entre la tabla que
contiene los valores de búsqueda y la tabla que se está modificando realmente, debe
establecerse la propiedad BoundColumn.
DBEngine y WorkSpace:
DAO es una metodología de acceso a datos que permite la manipulación de estos
mediante la utilización de objetos que pertenecen a su jerarquía. Sin embargo, para poder
accesar datos de una base, debe utilizarse primero el objeto DBEngine para abrir un
espacio de trabajo, o sesión, denominado Workspace.
El objeto DBEngine es el objeto de nivel superior dentro del modelo DAO. En este
contexto se pueden utilizar los métodos RepairDatabase y CompactDataBase de este
objeto para darle mantenimiento a una base de datos, tal y como se muestra en el código
siguiente:
MsgBox "El número de versión de DAO es " & _
DBEngine.Version
DBEngine.RepairDatabase "C:\MIBD.MDB"
El objeto WorkSpace define una sesión de usuario, determinando cuanto interactua su
aplicación con los datos de la base de datos. Si se abre una base de datos sin definir el
Workspace se utiliza DBEngine.Workspaces(0) como valor predeterminado para la sesión.
En el ejemplo siguiente se crea una sesión de usuario que utiliza el motor de bases de datos
de Microsoft Jet:
Dim wspNew as Workspace
Set wspNew = DBEngine.CreateWorkspace _
("NewJetWorkspace", "Admin", "", dbUseJet)
Usando el método OpenDatabase:
44
Manual del Alumno
Con este método se puede crear un objeto tipo Database. Este método se caracteriza
por manejar las siguientes tareas: el nombre de la base de datos origen, el metodo de acceso
a datos, indicación del ambiente de uso (monousuario, multiusuario), indicacion de
lectura/escritura en la base de datos o alguno de ellos, conexión ISAM, DAO u ODBC.
La sintaxis de esta función es la siguiente:
Set basededatos = espaciodetrabajo.OpenDatabase
opciones, sólolectura, conexión)
(nombrebasededatos,
La sintaxis del método OpenDatabase consta de las siguientes partes:
Argumento
basededatos
espaciodetrabajo
nombrebasededatos
Argumento
opciones
sólolectura
conexión
Descripción
Una variable de objeto que representa el objeto
Database que va a abrir.
Opcional. Una variable de objeto que representa el
objeto Workspace existente que contendrá la base de
datos. Si no incluye un valor para espaciodetrabajo,
OpenDatabase utiliza el espacio de trabajo
predeterminado.
Un tipo de datos String que es el nombre de un archivo
de base de datos Microsoft Jet existente o el nombre del
origen de datos (DSN) de un origen de datos ODBC
existente. Consulte la propiedad Name para obtener más
información acerca de este valor.
Descripción
Opcional. Un tipo de datos Variant que establece varias
opciones para la base de datos, como se especifica en
Valores.
Opcional. Un valor de tipo de datos Variant (subtipo
Boolean) que es True si desea abrir la base de datos con
acceso de sólo lectura o False (predeterminado) si desea
abrir la base de datos con acceso de lectura/escritura.
Opcional. Un tipo de datos Variant (subtipo String) que
especifica información variada sobre la conexión,
incluyendo las contraseñas.
Valores:
Para los espacios de trabajo Microsoft Jet, puede utilizar los siguientes valores para el
argumento opciones:
Valor
True
False
Descripción
Abre la base de datos en modo exclusivo.
(Predeterminado) Abre la base de datos en modo compartido.
Para los espacios de trabajo ODBCDirect, el argumento opciones determina si y
cuando informar al usuario acerca de establecer la conexión. Puede utilizar una de
las siguientes constantes:
45
Manual del Alumno
Constante
dbDriverNoPrompt
DbDriverPrompt
DbDriverComplete
Constante
DbDriverCompleteRequired
Descripción
El Administrador del controlador de ODBC utiliza la
cadena de conexión suministrada en nombrebasededatos
y conexión. Si no proporciona información suficiente,
se produce un error en tiempo de ejecución.
El Administrador del controlador de ODBC muestra
el cuadro de diálogo ODBC Data Sources, que
muestra cualquier información interesante
suministrada en nombrebasededatos o conexión. La
cadena de conexión se realiza con el DSN que el
usuario selecciona en los cuadros de diálogo o, si el
usuario no especifica un DSN, se utiliza el valor
predeterminado de DSN.
(Predeterminado) Si los argumentos conexión y
nombrebasededatos incluyen toda la información
necesaria para completar la conexión, el Administrador
del controlador de ODBC utiliza la cadena en conexión.
De otro modo funciona como cuando especifica
dbDriverPrompt.
Descripción
Esta opción funciona como dbDriverComplete
excepto que el controlador ODBC desactiva el
envío de información que no necesita para
completar la conexión.
Abriendo tablas ISAM:
Muchas veces se necesita abrir tablas o archivos que son tipo DBF, por ejemplo. En
este caso se puede utilizar la siguiente sintaxis:
Set dbMydb = OpenDatabase("\\PlaniVB", False, False, "Dbase III;")
Set rs = dbMydb.OpenRecordset("Planilla", dbOpenTable)
rs.Index = "Cedula"
Se puede observar que PlaniVB es el directorio donde está contenida la tabla que se
desea abrir. Puede observarse que el comando OpenDatabase no abre ninguna base de datos
sino que solo referencia el directorio en donde yace la tabla DBF. Seguidamente, se crea un
Recordset, de tipo Table, para abrir la tabla DBF respectiva. En la tercera línea puede
observarse como habilitar un índice “Cedula” que utiliza la tabla y que podemos utilizar en el
Recordset para realizar una búsqueda indizada en la misma, tal y como se muestra en el
código siguiente:
rs.Seek "=", Busc
Busc en este caso, es una variable que contiene el valor obtenido, por ejemplo, de un
TextBox o de cualquier objeto.
El índice puede crearse utilizando Visual Data de Visual Basic. Se abre la tabla y se
modifica mediante la apertura en modo diseño y se crea el índice que se necesita. Visual
Basic mantiene posteriormente el índice actualizado automaticamente. También puede
46
Manual del Alumno
crearse inicialmente un archivo tipo texto, con cualquier editor y salvarlo con extensión INF.
El
archivo
debe
ser
tal
y
como
se
muestra:
Archivo INDCED.INF:
NDX = CEDULA.NDX
Comentarios:
Cuando abre una base de datos, automáticamente se agrega a la colección Databases.
Además, en un espacio de trabajo ODBCDirect, el objeto Connection correspondiente al
objeto Database nuevo también se crea y anexa a la colección Connections del mismo objeto
Workspace.
Estas son algunas
nombrebasededatos:
consideraciones
que
se
debe
aplicar
cuando
utilice
Si se hace referencia a una base de datos que ya está abierta para acceso en
modo exclusivo por otro usuario, se produce un error.
Si no se hace referencia a una base de datos existente o a un nombre de un origen de
datos ODBC válido, se produce un error.
Si es una cadena de longitud cero ("") y conexión es "ODBC;", se muestra un cuadro
de diálogo que enumera todos los nombres de orígenes de datos ODBC registrados
para que el usuario pueda seleccionar una base de datos.
Si se está abriendo una base de datos mediante un espacio de trabajo ODBCDirect y
proporciona el DSN en conexión, puede establecer nombrebasededatos a una
cadena de su elección que puede utilizar para hacer referencia a esta base de datos
en el código más adelante.
El argumento conexión se expresa en dos partes: el tipo de base de datos, seguido por
punto y coma (;) y los argumentos opcionales. Primero debe proporcionar el tipo de base de
datos, como "ODBC;" o "FoxPro 2.5;". A continuación, os argumentos opcionales sin un
orden concreto, separados por punto y coma. Uno de los parámetros puede ser la contraseña
(si hay alguna asignada). Por ejemplo:
"FoxPro 2.5; pwd=micontraseña"
Utilizar el método NewPassword en un objeto Database distinto de una base de datos
de ODBCDirect cambia el parámetro de contraseña que aparece en la parte ";pwd=..." de este
argumento. Debe aplicar los argumentos opciones y sólolectura
para proporcionar una cadena de origen. Consulte la propiedad Connect para su
sintaxis.
Para cerrar una base de datos y, de este modo, quitar el objeto Database de la colección
Databases, utilice el método Close en el objeto .
47
Manual del Alumno
Nota Cuando tiene acceso a un origen de datos ODBC conectado a Microsoft Jet,
puede mejorar el rendimiento de sus aplicaciones abriendo un objeto Database conectado al
origen de datos ODBC, en vez de vincular objetos TableDef individuales a tablas específicas
en el origen de datos ODBC.
El cuadro siguiente muestra el soporta que brinda Visual Basic a varios formatos de
bases de datos:
Database Format
DataBaseName
Microsoft Access
drive:\path\file.mdb
Betrieve
drive:\path\file.ddf
DBase III
drive:\path
Dbase IV
drive:\path
FoxPro Versión 2.5
drive:\path
ODBC (SQL Server,Oracle) Registered data source
Name(usualmente el
Nombre del servidor)
Connect
(none)
“Betrieve”
“dBASE III”
“DBASE IV”
“FoxPro 2.5”
“Odbc;Dsn=Server;
Uid=User; Pd = Password”
Desarrollo Avanzado de Bases de Datos:
Exigencia de la Integridad de los datos:
Este apartado no se refiere al manejo de datos con sistemas administradores de bases
de datos de alto nivel como Oracle, Sybase o Microsoft SQL Server, dado que sobrepasan el
alcance de este manual. Se refiere, por el contrario, a la manipulación de datos con
Microsoft Access o Microsoft Jet como motor de administración de datos para aplicaciones
de mediana complejidad y de uso de escritorio y no cliente servidor.
Cuando se crea una aplicación de base de datos, debe asegurarse de que sólo se
escriban datos válidos en la misma.
Microsoft Access permite crear reglas y establecer restricciones de integridad
referencial en una base de datos para ayudar a garantizar la integridad de los datos.
Se recibirá un error de tiempo de ejecución si se trata de infringir estas restricciones.
En este caso debe proporcionarse el código que intercepta estos errores y que permite al
usuario corregir los datos y continuar.
Cuando se utiliza una base de datos de Microsoft Jet, pueden definirse reglas que
especifiquen los datos que son válidos en un campo o en una tabla. Estas reglas se almacenan
con la base de datos y se exigirán independientemente de cómo modifique los datos. Por
ejemplo puede establecerse que en una tabla determinada de una base de datos no permita
valores nulos o ceros.
Propiedades de las reglas:
Con Visual Basic se pueden crear o ver las reglas de una base de datos. Los objetos
Field y Recordset cuentan con dos propiedades relacionadas con las reglas.
Estas propiedades sólo se admiten en las bases de datos de Microsoft Jet:
48
Manual del Alumno
ValidationRule: Define los criterios de la regla.
ValidationText : Proporciona un mensaje de error que puede mostrarse si se infringe
la regla.
Por ejemplo, si una tabla denominada Pedidos posee un campo Cantidad, puede
escribirse código que permita mostrar las propiedades Validadtion Rule y Validation Text,
tal como sigue:
Sub DisplayRule()
Set rstOrders = dbMydb.OpenRecordset("Detalles de pedidos")
' Imprime ">0"
Debug.Print rstOrders("Cantidad").ValidationRule
' Imprime "La cantidad debe ser mayor que 0"
Debug.Print rstOrders("Cantidad").ValidationText
End Sub
Nota Se puede establecer la propiedad ValidationRule de un objeto Field para definir
una regla de forma programada. Sin embargo, aquí no se explica cómo definir una regla por
programa. Para obtener más información acerca de la creación de reglas, vea la propiedad
ValidationRule en la Ayuda de Visual Basic.
Errores de infracción de las reglas:
Si cuando se actualiza un registro se infringe una regla, se producirá un error de tiempo
de ejecución. La propiedad ValidateOnSet determina cuándo se produce el error de
infracción de la regla.
Si ValidateOnSet es True, el error de tiempo de ejecución se produce cuando se
establece el campo, como se muestra en el ejemplo siguiente:
recOrders.Edit
recOrders("Cantidad").ValidateOnSet = True
recOrders("Cantidad").Value = 0 'SE GENERA UN ERROR
recOrders.Update
Si ValidateOnSet es False, el error de tiempo de ejecución se producirá cuando se
ejecute el método Update, como en el siguiente ejemplo:
recOrders.Edit
recOrders("Cantidad").ValidateOnSet = False
recOrders("Cantidad").Value = 0
recOrders.Update 'Se genera un error.
Tratamiento de las infracciones de las reglas:
Normalmente, si se produce una infracción de las reglas se deseará permitir al usuario
corregir los datos y tratar de volver a actualizarlos.
49
Manual del Alumno
Si se utiliza el control Data e infringe una regla, se producirá el evento Error. El
control Data muestra la propiedad ValidationText y cancela la actualización. El usuario
puede modificar los datos y hacer clic en el control Data para tratar de volver a realizar la
actualización.
Se puede agregar código al evento Error para realizar una acción distinta cuando se
produzca el error. Por ejemplo, puede mostrarse texto en un título en lugar de un cuadro de
mensajes o bien cambiar la propiedad ForeColor de los campos para indicar que se ha
producido un error.
Si se utilizan Objetos de acceso a datos (DAO) e infringe una regla, se recibirá un error
de tiempo de ejecución. Debe interceptarse este error e informar al usuario.
El siguiente fragmento de código muestra como desplegar un mensaje cuando se
produzca una infracción a la regla (cantidad igual a cero, por ejemplo):
Private Sub cmdUpdate_Click()
On Error GoTo update_err
rs.Edit
rs("Cantidad") = txtQuantity.Text
rs.Update
Exit Sub
update_err:
If Err.Number = 3316 Then 'determina la infracción
lblError.Caption= Err.Description
rs.CancelUpdate
Exit Sub
End If
End Sub
La propiedad Description del objeto Err contiene el valor de la propiedad
ValidationText.
Puede utilizarse el método CancelUpdate para cancelar el método Edit o AddNew
actual y para actualizar el registro actual. Pueden dejarse los datos del usuario en el
formulario o bien actualizar el formulario con los datos actuales de la base de datos.
Reglas de Integridad Referencial:
La integridad referencial hace referencia a las restricciones que se deben poner en una
base de datos para mantener las relaciones definidas entre dos tablas cuando se agregan,
cambian o eliminan registros.
Definir las restricciones de integridad referencial:
El motor de base de datos Microsoft Jet puede exigir automáticamente la integridad
referencial en las bases de datos de Microsoft Jet.
50
Manual del Alumno
Por ejemplo, puede existir una tabla denominada Clientes y otra Pedidos. No se
pueden incluir pedidos si no existe un cliente; eso es mantener la integridad referencial de los
datos.
Nota Pueden definirse las relaciones entre tablas mediante programación con Visual
Basic o bien manualmente, mediante Microsoft Access. Para obtener información acerca de
la creación de relaciones mediante programación, vea el método CreateRelation en la
Ayuda de Visual Basic.
Actualizaciones y eliminaciones en cascada:
Una vez establecida la integridad referencial, pueden especificarse las opciones
Actualizar en cascada los campos relacionados o Eliminar en cascada los registros
relacionados. Si establece estas opciones, el motor de base de datos Microsoft Jet permitirá
al usuario cambiar y eliminar registros, y cambiará o eliminará automáticamente los registros
relacionados de las tablas relacionadas.
Por ejemplo, suponga que se ha creado una relación entre las tablas Clientes y Pedidos.
Si se selecciona la opción Eliminar en cascada los registros relacionados para la tabla
Clientes, cada vez que el usuario elimine el registro de un cliente, el motor de base de datos
Microsoft Jet eliminará automáticamente todos los registros relacionados con ese cliente de
la tabla Pedidos.
Nota El motor de base de datos Microsoft Jet no muestra ninguna advertencia cuando
elimina los registros relacionados. Es muy importante que se entienda cómo están definidas
las relaciones en la base de datos para evitar pérdidas de datos al eliminar registros.
Tratamiento de las infracciones de integridad referencial:
Si se trata de infringir las restricciones de integridad referencial, se recibirá un error de
tiempo de ejecución. Debe interceptarse este error e informar al usuario.
El siguiente fragmento de código muestra un mensaje de error si el usuario intenta
eliminar el registro de un cliente que tiene un pedido:
Private Sub cmdDelete_Click()
On Error GoTo del_err
rs.Delete
rs.MoveNext
FillFields 'procedimiento escrito por el usuario
Exit Sub
del_err:
'Referencia a la violación de integridad
If Err.Number = 3200 Then
lblError.Caption = "No se puede borrar un cliente con órdenes."
Exit Sub
Else
'controlar otros errores...
51
Manual del Alumno
End If
End Sub
El registro no se elimina y no se vuelve a posicionar en el registro actual. Los cuadros
de texto del formulario siguen conteniendo los datos actuales.
Prevenir infracciones de integridad referencial:
Para evitar infracciones de integridad referencial puede proporcionarse un cuadro de
lista con entradas válidas. Si las entradas válidas son demasiado largas y no caben en un
cuadro de lista, permitase que el usuario especifique criterios de búsqueda y muestrese
después una lista de entradas válidas.
Por ejemplo, cuando agrega un nuevo producto, se puede permitir al usuario escribir un
valor para un nombre de producto y a continuación mostrar una cuadrícula con todos los
productos que coincidan con ese nombre. Seguidamente, el usuario puede seleccionar un
producto válido de la cuadrícula.
Transacciones:
Una transacción es una serie de cambios realizados en una base de datos. Pueden
agruparse varios cambios de bases de datos en una transacción.
Si todas las operaciones se realizan correctamente, confirmará la transacción. Si alguna
de las operaciones falla, deshará la transacción.
Métodos de transacciones:
Los métodos de transacciones se aplican al objeto Workspace. Se Utiliza el método
BeginTrans antes de actualizar el primer registro. Si alguna de las siguientes actualizaciones
falla, se utiliza el método Rollback para deshacerlas todas. Se usa el método CommitTrans
tras haber actualizado correctamente el último registro.
El siguiente fragmento de código muestra la utilización de transacciones con el fin de
agregar un entrada a la tabla Pedidos y a la tabla Detalles de pedidos:
Sub cmdAddOrder_Click ()
On Error GoTo AddOrder_Err
DBEngine.WorkSpaces(0).BeginTrans
db.Execute "Insert Into Pedidos...", dbFailOnError
db.Execute "Insert Into [Detalles de pedidos]..." , dbFailOnError
DBEngine.WorkSpaces(0).CommitTrans
lblStatus.Caption = "Actualizaciones terminadas"
Exit Sub
AddOrder_Err:
lblStatus.Caption = Err.Description
Msgbox "No se han efectuado todas las actualizaciones con éxito"
DBEngine.Workspaces(0).Rollback
52
Manual del Alumno
Exit Sub
End Sub
Utilizar transacciones para mejorar el rendimiento:
Además de utilizar las transacciones para mantener la integridad de los datos, puede
mejorarse el rendimiento si se incluye una instrucción Update de SQL en una transacción.
Las operaciones de la transacción se almacenan en búfer y no se escriben a disco hasta que
se confirma la transacción.
El siguiente ejemplo incluye una instrucción Update de SQL en una transacción:
DBEngine.Workspaces(0).BeginTrans
stSQL = "UPDATE Productos SET [PrecioUnidad] = [PrecioUnidad] * 0.1"
db.Execute strSQL, dbFailOnError
DBEngine.Workspaces(0).CommitTrans
Nota Es posible anidar transacciones en un objeto Workspace. Se deberá confirmar o
deshacer la transacción actual antes de poder confirmar o deshacer una transacción de un
nivel superior.
Si se desea disponer de transacciones que se solapen pero que sean independientes
entre sí, puede crearse un segundo objeto Workspace.
Para obtener más información acerca de la creación de objetos Workspace, veáse el
método CreateWorkspace en la Ayuda de Visual Basic.
Consideraciones multiusuario:
Las aplicaciones de bases de datos se vuelven más complejas si se desea permitir que
varios usuarios tengan acceso a una base de datos simultáneamente. Deberán tratar diversos
errores que pueden producirse cuando varios usuarios tengan acceso a los mismos datos. Por
ejemplo:
Dos usuarios tratan de modificar los mismos datos: El motor de base de datos
Microsoft Jet bloquea automáticamente los datos para evitar que dos usuarios
modifiquen un registro simultáneamente. Si los usuarios tratan de actualizar un
registro bloqueado, recibirán un error de tiempo de ejecución.
Alguien cambia un registro que otro usuario está viendo: Si un usuario intenta
actualizar un registro y los datos se han modificado desde que se ejecutó el método
Edit, el motor de base de datos Microsoft Jet devolverá un error.
Alguien elimina un registro que otro usuario está viendo: Si un usuario intenta
tener acceso a un registro que se ha eliminado, el motor de base de datos Microsoft
Jet devolverá un error.
Abrir una tabla para acceso exclusivo:
53
Manual del Alumno
Para evitar los problemas que ocurren cuando se permite que varios usuarios tengan
acceso a una base de datos, puede abrirse un tabla para uso exclusivo. Al abrirse una tabla de
este modo se evita que otro usuario la utilice. Aunque se trata de una solución bastante
restrictiva, puede resultar adecuada para tal situación.
El argumento opciones para el método OpenRecordset determina cómo se abre el
Recordset.
En el código siguiente se abre una tabla para uso exclusivo. Puede modificarse la tabla
Pedidos, pero ningún otro usuario podrá ver ni modificar los datos de dicha tabla:
Set recOrders = dbMydb.OpenRecordset _
("Pedidos", dbOpenTable, dbDenyRead + dbDenyWrite)
En el código siguiente se abre una tabla para escritura en modo exclusivo. Puede verse
y modificarse la tabla Pedidos; los otros usuarios pueden ver los datos de la tabla, pero no
pueden modificarlos:
Set recOrders = dbMydb.OpenRecordset _
("Pedidos", dbOpenTable, dbDenyWrite)
En el código siguiente se abre un Recordset en modo de sólo lectura. Pueden leerse los
datos de la tabla, pero no modificarlos. Esto no afecta a los demás usuarios:
Set recOrders = dbMydb.OpenRecordset _
("Pedidos", dbOpenTable, dbReadOnly)
En el código siguiente se abre un Recordset en modo anexar. Sólo pueden agregarse
registros; no pueden verse ni modificar los registros existentes. Esto no afecta a los demás
usuarios:
Set recOrders = dbMydb.OpenRecordset _
("Pedidos", dbOpenDynaset, dbAppendOnly)
Si sólo se desea agregar registros a una tabla, debe utilizarse la opción dbAppendOnly
para obtener mayor rendimiento. La opción dbDenyRead sólo está disponible en el
Recordset de tipo table.
Para obtener una lista de valores válidos para el argumento opciones, veáse el método
OpenRecordset en la Ayuda de Visual Basic.
El bloque de Microsoft Jet:
Todos los sistemas de administración de bases de datos proporcionan algún tipo de
mecanismo de bloqueo para evitar que dos usuarios actualicen datos simultáneamente.
El motor de base de datos Microsoft Jet proporciona un bloqueo de página. Las páginas
son bloques de registros de 2048 bytes (2 KB) de tamaño. Visual Basic almacena tantos
registros como quepan en cada página. Cuando Visual Basic bloquea la página que contiene
un registro que está modificando, todos los demás registros de esa página también quedarán
bloqueados.
54
Manual del Alumno
Nota La información aquí consignada es específica de las bases de datos que utilizan
el motor Microsoft Jet. Cuando se tiene acceso a bases de datos a través de ODBC, el
mecanismo de bloqueo está controlado por el sistema de administración de bases de datos.
Bloqueo pesimista frente a bloqueo optimista:
El valor de la propiedad LockEdits del objeto Recordset determina cuándo se activa
un bloqueo para los registros del Recordset.
Bloqueo pesimista:
Si la propiedad LockEdits tiene el valor True, el bloqueo pesimista está activo. La
página que contiene el registro actual se bloquea en el momento en que utiliza el método
Edit. La página se desbloquea cuando utiliza el método Update.
La estrategia de bloqueo predeterminada bloquea los registros durante un período de
tiempo mayor, pero garantiza que después de ejecutarse el método Edit ningún otro usuario
pueda modificar los datos.
Bloqueo optimista:
Si la propiedad LockEdits tiene el valor False, el bloqueo optimista está activo. La
página que contiene el registro sólo se bloquea mientras se actualiza el registro.
Los bloqueos permanecen activos durante un período de tiempo más corto y varios
usuarios pueden utilizar el método Edit sin bloquear la página. No obstante, en este caso
deberán tratarse los posibles errores cuando el usuario ejecute el método Update.
Si se piensa que los usuarios no intentarán modificar el mismo registro a menudo,
considerese la posibilidad de asignar a la propiedad LockEdits el valor False.
Desbloqueo:
El motor de base de datos Microsoft Jet marca una página para que se desbloquee en
cuanto se complete la actualización. Sin embargo, los bloqueos se quitan realmente cuando
no se produce ninguna otra acción.
Puede utilizarSE el método Idle del objeto DBEngine con la opción dbFreeLocks
para suspender el proceso y permitir al motor de base de datos Microsoft Jet liberar los
bloqueos y seguir con las otras tareas en segundo plano, como muestra el siguiente código:
DBEngine.Idle dbFreeLocks
Controlar los errores de bloqueo:
Cuando varios usuarios están actualizando una base de datos, deberá agregarse código
que intercepte los errores siguientes:
3260: No se pudo actualizar; actualmente está bloqueado: Se produce en el
método Edit cuando el registro está bloqueado. El código debe esperar unos
instantes y tratar de aplicar Edit otra vez o informar al usuario del error.
55
Manual del Alumno
3186: No se pudo guardar el registro; bloqueado actualmente: Se produce en el
método Update cuando el registro está bloqueado. El código debe esperar unos
instantes y volver a intentarlo o informar al usuario del error.
3197: Los datos han cambiado; la operación se ha detenido: Se produce en el
método Update o Edit si otro usuario cambió los datos desde la última vez que
tuvo acceso a ellos. El código debe actualizar el formulario con los datos más
recientes o informar al usuario del error.
Para obtener una lista completa de los errores disponibles en Visual Basic, veáse
Errores interceptables en la Ayuda de Visual Basic.
Tratar errores en Edit
El siguiente código de ejemplo muestra como tratar los errores de bloqueo que pueden
ocurrir en el método Edit:
Private Sub cmdEdit_Click()
On Error GoTo HandleError
rs.Edit
ButtonEditAddMode
Exit Sub
HandleError:
Select Case Err.Number
Case 3260 'la página está bloqueada actualmente
MsgBox "El registro está bloqueado actualmente. Inténtelo más
tarde."
Case 3197 'los datos han cambiado
MsgBox "Los datos han cambiado y se van a actualizar."
rs.Bookmark = rs.Bookmark 'obtener datos actualizados
FillFields
rs.Edit
Case Else
MsgBox Err.Number & ": " & Err.Description
End Select
End Sub
Si el registro está bloqueado cuando el usuario ejecuta el método Edit, se mostrará un
mensaje; si los datos han cambiado, el código actualizará el formulario con los datos más
recientes.
La instrucción recOrders.Bookmark = recOrders.Bookmark actualiza el registro
actual del Recordset con los datos más recientes de la base de datos.
Tratar errores en Update:
56
Manual del Alumno
El siguiente código de ejemplo muestra como tratar los errores de bloqueo que pueden
ocurrir en el método Update:
Private Sub cmdSave_Click()
Dim answer As Integer
On Error GoTo HandleError
rs.Fields("NombreCategoría") = txtCategoryName.Text
rs.Fields("Descripción") = txtDescription.Text
rs.Update
rs.Bookmark = rs.LastModified
FillFields
ButtonNonEditAddMode
Exit Sub
HandleError:
Select Case Err.Number
Case 3260
MsgBox "El registro está bloqueado actualmente. " & _
"Intente guardarlo más tarde o cancele los cambios."
Case 3197
answer = MsgBox("Los datos han sido modificados por otro
usuario. " & _ "¿Sobrescribir cambios?", vbYesNo)
If answer = vbYes Then
Resume
Else
rs.Bookmark = rs.Bookmark 'actualizar con cambios de
'otros usuarios
cmdCancel_Click
End If
Case Else
MsgBox Err.Number & ": " & Err.Description
End Select
End Sub
Tratar errores de los registros eliminados por otros usuarios:
Cuando el Recordset de tipo dynaset se ha llenado por completo, si otro usuario
elimina un registro de la base de datos, el puntero a esa entrada no se eliminará del dynaset.
Sin embargo, dicho registro ya no aparecerá en la base de datos. No podrá verse y se
producirá un error si se intenta tener acceso a los datos de ese registro.
La forma más sencilla de tratar este error es eliminar la entrada que hace referencia al
registro eliminado del dynaset.
El siguiente fragmento de código muestra cómo tratar un error generado por un intento
de tener acceso a un registro eliminado:
57
Manual del Alumno
Private Sub cmdMoveNext_Click()
On Error GoTo err_movefirst:
retry_MoveNext:
rs.MoveNext
txtLName.Text = rs("Apellidos") 'Puede ocurrir un error
Exit Sub
err_Movefirst:
If Err.Number = 3167 Then
'se borra el registro,
'eliminar la entrada del dynaset
rs.Delete
Resume retry_MoveNext
Else
MsgBox Err.Description
Exit Sub
End If
End Sub
Anteriormente, se dedicó gran parte a explicar el objeto Recordset y se comentó acerca
de la creación y manipulación de estos. Seguidamente, se brinda un reforzamiento de
conceptos al respecto.
Acceso a bases de datos externas:
Desde Visual Basic puede tenerse acceso a diversas bases de datos ISAM (Método de
acceso secuencial indizado) como dBASE, Paradox, Btrieve y Microsoft Visual FoxPro.
Visual Basic proporciona controladores para diversas bases de datos ISAM. Estos
controladores se muestran en el registro.
También puede utilizar Visual Basic para tener acceso a archivos de datos estándar,
como archivos de texto delimitados por comas.
Nota Las bases de datos que utilizan el motor de base de datos Microsoft Jet se
consideran nativas de Visual Basic.
Opciones de acceso:
Hay dos formas de tener acceso a los datos de una base de datos externa: adjuntar la
base de datos externa y abrirla directamente.
Adjuntar una tabla a una base de datos de Microsoft Jet:
Cuando se adjunta una base de datos externa a una base de datos de Microsoft Jet, los
datos de la tabla permanecen en la base de datos externa. Sin embargo, la información de
conexión y la definición de la tabla se almacenan en la base de datos de Microsoft Jet. Puede
58
Manual del Alumno
utilizarse la tabla como cualquier otra tabla de bases de datos de Microsoft Jet, a excepción
de que no puede crear un Recordset de tipo table en una tabla adjunta.
Puede utilizarse Microsoft Access para adjuntar fácilmente una tabla a una base de
datos de Microsoft Jet. También puede adjuntarse una tabla por programa mediante Visual
Basic.
Para obtener información acerca de cómo adjuntar tablas por programa, veáse el
método CreateTableDef en la Ayuda de Visual Basic.
Nota Normalmente puede tenerse acceso más rápidamente a los datos de las tablas
ODBC que se adjuntan a una base de datos de Microsoft Jet que a los datos de una base de
datos ODBC que haya abierto directamente. Si es posible, considerese la posibilidad de
adjuntar tablas externas en lugar de abrirlas directamente.
Abrir una tabla directamente:
Cuando se abre directamente una tabla externa, debe especificarse la información de
conexión mediante los argumentos del método OpenDatabase o la propiedad Connect del
control Data.
El siguiente fragmento de código muestra como utilizar los métodos OpenDatabase y
OpenRecordset para abrir una tabla directamente:
59
Manual del Alumno
Sub OpenTable()
Dim wspCurrent As Workspace
Dim dbCurrent As Database
Dim rstProducts As Recordset
Set wspCurrent = CreateWorkspace("New Workspace", "Admin", "", _
dbUseJet)
Set dbCurrent = wspCurrent.OpenDatabase("C:\Archivos de _
programa\DevStudio\VB\Inventario.mdb")
Set rstProducts = dbCurrent.OpenRecordset("Productos", dbOpenTable)
rstProducts.MoveLast
MsgBox "Hay " & rstProducts.RecordCount & " registros."
rstProducts.Close
dbCurrent.Close
wspCurrent.Close
End Sub
Acceso a bases de datos ISAM:
Para abrir una base de datos ISAM directamente, debe asignarse al argumento
Database del método OpenDatabase (o a la propiedad DatabaseName del control Data) el
nombre de la carpeta que contiene las tablas ISAM. Si la red lo admite, puede utilizar una
ruta de acceso de red para el nombre de la base de datos, por ejemplo,
“\\mi_servidor\mi_recurso”.
Asignese a la propiedad Connect del objeto Database o del control Data el tipo de
base de datos que se va a abrir. Para obtener una lista de valores posibles, veáse la propiedad
Connect en la Ayuda de Visual Basic.
En el ejemplo siguiente se abre directamente una base de datos de Microsoft Visual
FoxPro:
Public Sub OpenFoxProTable()
Dim dbFox As Database
Dim recAccounts As Recordset
Set dbFox = OpenDatabase _
("\\FoxPro\Data\AP", False, False, “FoxPro 5.0;") „direccionado a la red.
Set recAccounts = dbFox.OpenRecordset("Cuentas")
End Sub
Para adjuntar una base de datos ISAM a una base de datos .mdb existente, debe crearse
un nuevo objeto TableDef y establecer las propiedades Connect y SourceTableName del
objeto TableDef.
En el ejemplo de código siguiente se adjunta una tabla de Microsoft FoxPro a una base
de datos de Microsoft Jet. Puede crearse una nueva base de datos de Microsoft Jet o bien
utilizar una existente:
Dim tbdAttach as TableDef
Set dbMydb=OpenDatabase ("Mibd.mdb")
Set tbdAttach = dbMydb.CreateTableDef ("Tabla FoxPro adjunta")
60
Manual del Alumno
tbdAttach.Connect = "FoxPro 5.0;DATABASE=\\FoxPro\AP"
tbdAttach.SourceTableName = "Cuentas"
dbMydb.TableDefs.Append tbdAttach
Cuando se haya adjuntado la tabla a una base de datos de Microsoft Jet, podrá abrirse
de la forma siguiente:
Set dbMydb = OpenDatabase ("Mibd.mdb")
Set recAttach = dbMydb.OpenRecordset ("Tabla FoxPro adjunta")
Distribuir el ejecutable:
Cuando se distribuya una aplicación que tenga acceso a una base de datos ISAM,
deberá proporcionarse al usuario los controladores ISAM adecuados.
Trabajando con archivos de datos:
Pueden crearse aplicaciones de Visual Basic que trabajen con información de archivos
de datos estándar en lugar de hacerlo con una base de datos.
Cuando se trabaje con información procedente de archivos de datos estándar, pueden
utilizarse alguna de las siguientes opciones:
Instrucciones de E/S de archivos: Puede utilizar las instrucciones estándar de
acceso a archivos de Visual Basic para leer y escribir archivos. Para obtener más
información acerca de la instrucción Open, vea Open en la Ayuda de Visual Basic.
Importación manual a una base de datos: Puede utilizar Microsoft Access para
importar manualmente un archivo de texto a una base de datos. Microsoft Access
admite diversos formatos de archivo, como de longitud fija, delimitados por comas,
etc.
Importación mediante programación a una base de datos: Puede crear un
programa que lea un archivo de texto y agregue los datos a una base de datos.
61
Manual del Alumno
El siguiente código de ejemplo muestra los pasos básicos para utilizar la E/S de
archivo con el fin de leer un archivo e insertar los datos en una base de datos:
Private Sub Command4_Click()
Dim fhandle As Integer
fhandle = FreeFile()
Open "newCust.txt" For Input As fhandle
Do
'Leer archivo utilizando alguna instrucción Input
'Crear cadena sql
'Ejecutar instrucción Insert de SQL
Loop Until EOF(fhandle)
End Sub
Acceso a datos mediante ODBC:
La versión 3.5 de DAO presenta una nueva tecnología de conexión cliente-servidor
llamada Open Database Connectivity Direct (ODBCDirect). Esta tecnología establece una
conexión directamente con un origen de datos ODBC, sin cargar el motor de base de datos
Microsoft Jet.
Definir un espacio de trabajo ODBC:
La tecnología de ODBCDirect permite tener acceso directamente a orígenes de datos
ODBC mediante Objetos de acceso a datos (DAO). Pueden utilizarse las características de
esta conexión cliente-servidor para tener acceso rápidamente a una base de datos, sin cargar
el motor de base de datos Microsoft Jet.
El primer paso para utilizar ODBCDirect consiste en definir el tipo de espacio de
trabajo que utilizará. Puede hacerse de dos formas: establecer el tipo de espacio de trabajo
predeterminado o definir un tipo de espacio de trabajo específico como ODBCDirect.
Establecer el espacio de trabajo predeterminado:
Para establecer el tipo de espacio de trabajo predeterminado como ODBCDirect, debe
utilizarse la propiedad DefaultType del objeto DBEngine. Este tipo se utilizará siempre que
se cree un objeto Workspace.
Para definir un tipo de espacio de trabajo específico como ODBCDirect, debe
asignarse a la propiedad DefaultType el valor dbUseODBC. Este valor impide que el motor
de base de datos Microsoft Jet se cargue en memoria, si no ha creado ya un espacio de
trabajo de base de datos de Microsoft Jet.
62
Manual del Alumno
El código siguiente establece el tipo de espacio de trabajo predeterminado como
ODBCDirect:
DBEngine.DefaultType = dbUseODBC
Crear un espacio de trabajo específico:
Puede invalidarse el valor de espacio de trabajo predeterminado para un determinado
espacio de trabajo si utiliza el argumento tipo del método CreateWorkspace.
Conectarse a un origen de datos remoto:
Conectarse con un origen de datos remoto es similar a abrir una base de datos de
Microsoft Jet. Primero debe crearse un espacio de trabajo (como un espacio de trabajo de
ODBCDirect) y después abrir el origen de datos.
Para abrir un origen de datos remoto, debe crease un objeto Connection de la misma
forma en que se crearía un objeto Database para abrir una base de datos de Microsoft Jet.
Una vez creada una conexión, pueden recuperarse datos del origen de datos remoto.
Declaración de la conexión:
La variable de objeto Connection se refiere a su conexión. Utilicese la instrucción Set
para abrir la conexión.
El código siguiente abre una conexión en el espacio de trabajo predeterminado:
Dim conNewConnection As Connection
Set conNewConnection = OpenConnection _
("New", dbDriverPrompt, False, _
"ODBC;DATABASE=pubs;UID=sa;PWD;DSN=SQLServer")
Método OpenConnection:
En la tabla siguiente se enumeran los argumentos del método OpenConnection y se
describe su uso.
63
Manual del Alumno
Argumento
Nombre
Opciones
Solo_lectura
Conexión
Descripción
Cualquier cadena, si especifica un nombre de origen de datos (DSN) ODBC
registrado. Si en la declaración de OpenConnection no se incluye un DSN
válido, el nombre deberá hacer referencia a un DSN válido de ODBC, que
también será la propiedad Name.
Establece diversas opciones para la conexión.
True o False. Si es True, no se permiten modificaciones. El valor
predeterminado es False.
Una cadena de conexión ODBC.
El argumento opciones determina si y cuándo se pedirá al usuario que establezca la
conexión y si la conexión se abrirá o no de manera asíncrona.
En la tabla siguiente se muestran los valores del argumento opciones y se describe su
uso.
Constante
dbDriverNoPrompt
Resultado
El Administrador de controladores ODBC utiliza la cadena de conexión
proporcionada en el argumento conexión
El Administrador de controladores ODBC muestra el cuadro de diálogo
dbDriverPrompt
Seleccionar origen de datos, que presenta toda la información relevante
proporcionada en el argumento conexión. La cadena de conexión está
formada por el DSN seleccionado por el usuario mediante los cuadros de
diálogo o bien, si el usuario no especifica ningún DSN, se usará el DSN
predeterminado.
Si el argumento conexión incluye toda la información necesaria para
dbDriverComplete
completar una conexión, el Administrador de controladores ODBC usará
la cadena del argumento conexión. De lo contrario, se comportará como
cuando especifica la constante predeterminada dbDriverPrompt. Es el
valor predeterminado.
dbDriverCompleteR Esta opción se comporta como la constante dbDriverComplete, excepto
en que el controlador ODBC desactiva las peticiones de información
equired
innecesarias para completar la conexión
Ejecuta el método de forma asíncrona. Esta constante se puede utilizar
dbRunAsync
con cualquiera de las demás constantes de opciones
64
Manual del Alumno
Recuperar datos remotos:
La recuperación de datos remotos mediante ODBCDirect es muy similar a la
recuperación de datos desde una base de datos de Microsoft Jet. Después de establecer una
conexión con una base de datos compatible con ODBC mediante ODBCDirect, cree un
objeto Recordset; para ello, utilicese el método OpenRecordset de la misma forma que si se
recuperaran registros de una base de datos de Microsoft Jet.
Declaración del Recordset:
El siguiente ejemplo de código crea un espacio de trabajo de ODBCDirect con una
conexión al espacio de trabajo y después recupera un Recordset:
Sub OpenODBCDirectRecordset()
Dim wspODBC As Workspace
Dim conODBC As Connection
Dim rstODBC As Recordset
Set wspODBC = CreateWorkspace _
("", "sa", "", dbUseODBC)
Set conODBC = wspODBC.OpenConnection _
("NewConnection", dbDriverNoPrompt, False, _
"ODBC;DATABASE=pubs;UID=sa;PWD=;DSN=SQLServer")
Set rstODBC = conODBC.OpenRecordset _
("SELECT * FROM Autores WHERE au_lname = 'Enrique Gómez'",
_
dbOpenSnapshot, 0, dbReadOnly)
rstODBC.MoveLast
MsgBox "Este conjunto de datos remoto contiene " & _
rstODBC.RecordCount & " registros."
rstODBC.Close
conODBC.Close
wspODBC.Close
End Sub
El Recordset recuperado de un origen de datos remoto se puede controlar de la misma
forma que un Recordset recuperado mediante DAO con el motor de base de datos Microsoft
Jet.
Método OpenRecordset con ODBCDirect:
Además de las opciones disponibles en el método OpenRecordset de un objeto
Database, el método OpenRecordset de un objeto Connection permite recuperar registros
de una base de datos.
Cuando se utiliza un espacio de trabajo de ODBCDirect están disponibles los
siguientes argumentos del método OpenRecordset:
Tipo:
—
dbOpenDynamic
65
Manual del Alumno
Abre un Recordset de tipo dynamic que puede utilizar para agregar, cambiar o eliminar
registros de una o varias tablas de una base de datos subyacente. Un cursor dinámico puede
contener campos de una o más tablas de una base de datos.
Opciones:
—
dbRunAsync
Ejecuta una consulta asíncrona. Una consulta asíncrona es aquella en que las
consultas SQL devuelven el control inmediatamente, incluso aunque los resultados no estén
preparados. Esto permite que una aplicación efectúe otros procesos mientras la consulta se
termina.
—
dbExecDirect
Ejecuta una consulta; para ello elude SQLPrepare y llama directamente a
SQLExecDirect. Utilice esta opción sólo cuando no vaya a abrir un Recordset basado en
una consulta de parámetros.
®
ediciones_bloqueo
—
dbOptimisticValue
Utiliza la concurrencia optimista basada en valores de fila.
—
dbOptimisticBatch
Activa la actualización optimista por lotes.
Nota El espacio de trabajo predeterminado de ODBCDirect es de sólo lectura. Para
que los usuarios puedan leer y escribir registros en un espacio de trabajo de ODBCDirect,
debe especificarse el tipo de bloqueo dbOptimistic. Para obtener información acerca de los
tipos de bloqueo, veáse el método OpenRecordset en la Ayuda de Visual Basic.
66
Manual del Alumno
Problemas de rendimiento:
El diseño de una aplicación afecta en gran medida a su rendimiento. Cuando se
desarrollen aplicaciones de base de datos, tengase en cuenta las siguientes sugerencias:
Analícese lo que se requiere: Algunas consultas SQL son más eficientes que otras.
Por ejemplo, ordenar por un campo que contiene un índice es más rápido que
ordenar por un campo que no contiene un índice.
Para obtener más información acerca de la optimización de consultas, busque
Sugerencias de rendimiento y solución de problemas en los Libros en pantalla de
Visual Basic y haga clic en Sugerencias de rendimiento y solución de problemas.
Solicitar únicamente los datos que necesite: No crear Recordsets de tipo dynaset y
snapshot en toda una tabla. El diseño de una aplicación debe permitir al usuario
especificar criterios y devolver un Recordset limitado.
En lugar de seleccionar todas las columnas de un registro, deben seleccionarse
únicamente las columnas necesarias.
El código siguiente crea un Recordset basándose en un valor especificado por el
usuario. Por ejemplo, si el usuario escribe “A”, el código creará un Recordset de
todos los empleados cuyo apellido empiece por “A”:
strSQL = "Select [Apellidos], [Nombre] From Empleados " & _
"Where [Apellidos] Like " & _
"'" & txtName.Text & "'" & "*"
Set recEmployees = dbMydb.OpenRecordset (strSQL,dbOpenDynaset)
Utilizar únicamente la funcionalidad necesaria: Si no se necesita actualizar la base
de datos, ábrase de sólo lectura. Esto mejora el rendimiento, ya que el motor de
base de datos Microsoft Jet no necesita hacer un seguimiento de los bloqueos.
Si no se necesita actualizar un Recordset, ábrase como de sólo lectura.
Si sólo se va a agregar a un Recordset, ábrase con la opción dbAppendOnly.
Utilizar transacciones: Utilizar una transacción reduce el acceso a disco a una vez
por transacción, no una vez por registro, lo cual puede mejorar considerablemente
el rendimiento de una aplicación. Puede que se desee agrupar transacciones. Si una
transacción es demasiado grande, utilizará memoria que pueden necesitar otras
operaciones.
Utilizar consultas almacenadas: En general, es mejor crear consultas almacenadas
que utilizar SQL en el texto. Una consulta almacenada está precompilada, por lo
que ahorrará tiempo de compilación durante la ejecución. Además, normalmente es
más sencillo mantener consultas que están almacenadas en un archivo .mdb que las
creadas mediante aplicación.
Si es posible, utilice consultas almacenadas o SQL para las operaciones por lotes,
como la actualización de un grupo de registros, en lugar de actualizar los registros
de uno en uno con el método Update.
67
Manual del Alumno
Adjuntar tablas SQL: Adjuntar tablas de bases de datos ODBC a una base de datos
.mdb local. La información de conexión y de definición de datos se mantiene con el
archivo .mdb y, por tanto, mejora el rendimiento.
Escribir sus propias pruebas de rendimiento: Hay muchos factores que afectan al
rendimiento de una aplicación. La mejor manera de averiguar si una característica
mejorará el rendimiento es probarla con los datos de su entorno.
Práctica No. 17
Diseñe el siguiente formulario:
Escriba el siguiente codigo en las declaraciones generales del formulario:
Public CodDML As Integer
Public Mensaje, TituloBarra As String
Function Ver_Forma2()
Form2.Show
End Function
Function Habilita_Textos()
If CodDML = 0 Then
68
Manual del Alumno
Text1.Enabled = True
End If
Text2.Enabled = True
Text3.Enabled = True
Text4.Enabled = True
End Function
Function DesHabilita_Textos()
If CodDML = 0 Then
Text1.Enabled = False
End If
Text2.Enabled = False
Text3.Enabled = False
Text4.Enabled = False
End Function
Function Descubre_Botones()
Incluir.Visible = True
Anterior.Visible = True
Siguiente.Visible = True
Final.Visible = True
Inicio.Visible = True
Modificar.Visible = True
Excluir.Visible = True
Cerrar.Visible = True
Aceptar.Visible = False
Cancelar.Visible = False
End Function
Function Esconde_Botones()
Incluir.Visible = False
Modificar.Visible = False
Excluir.Visible = False
Anterior.Visible = False
Siguiente.Visible = False
Final.Visible = False
Inicio.Visible = False
Cerrar.Visible = False
Aceptar.Visible = True
Cancelar.Visible = True
End Function
Private Sub Aceptar_Click()
Select Case CodDML
Case CodDML = 0
Data1.Recordset.Update
Case CodDML = 1
Data1.Recordset.Update
Case CodDML = 2
Data1.Recordset.Delete
End Select
69
Manual del Alumno
MsgBox Mensaje, 0 + 16 + 56, TituloBarra
Call Descubre_Botones
Call DesHabilita_Textos
End Sub
Private Sub Anterior_Click()
If Not Data1.Recordset.BOF Then
Data1.Recordset.MovePrevious
Siguiente.Enabled = True
Final.Enabled = True
If Data1.Recordset.BOF Then
Anterior.Enabled = False
Inicio.Enabled = False
Data1.Recordset.MoveFirst
MsgBox "Inicio de la tabla"
End If
End If
End Sub
Private Sub Buscar_Click()
Call Ver_Forma2
Data1.Recordset.FindFirst "[CEDULA] = 'ABuscar'"
If Data1.Recordset.NoMatch Then
MsgBox "NO se encuentra"
End If
End Sub
Private Sub Cancelar_Click()
Data1.Refresh
Call DesHabilita_Textos
Call Descubre_Botones
End Sub
Private Sub Cerrar_Click()
End
End Sub
Private Sub Excluir_Click()
CodDML = 2
Call Esconde_Botones
Call Habilita_Textos
Mensaje = "El Registro fue Exitosamente Eliminado"
TituloBarra = "Eliminado.."
End Sub
Private Sub Final_Click()
Data1.Recordset.MoveLast
Siguiente.Enabled = False
Final.Enabled = False
70
Manual del Alumno
Anterior.Enabled = True
Inicio.Enabled = True
End Sub
Private Sub Incluir_Click()
CodDML = 0
Call Esconde_Botones
Call Habilita_Textos
Mensaje = "El Registro fue Exitosamente almacenado"
TituloBarra = "Inclusión de Registros.."
Data1.Recordset.AddNew
End Sub
Private Sub Inicio_Click()
Data1.Recordset.MoveFirst
Anterior.Enabled = False
Inicio.Enabled = False
Siguiente.Enabled = True
Final.Enabled = True
End Sub
Private Sub Modificar_Click()
CodDML = 1
Call Esconde_Botones
Call Habilita_Textos
Data1.Recordset.Edit
Mensaje = "El Registro fue Exitosamente Modificado"
TituloBarra = "Modificación de Registros.."
Text2.SetFocus
End Sub
Private Sub Siguiente_Click()
If Not Data1.Recordset.EOF Then
Data1.Recordset.MoveNext
Anterior.Enabled = True
Inicio.Enabled = True
If Data1.Recordset.EOF Then
Data1.Recordset.MoveLast
Siguiente.Enabled = False
Final.Enabled = False
MsgBox "Final de la tabla"
End If
End If
End Sub
Private Sub Text1_KeyPress(Tecla As Integer)
71
Manual del Alumno
If Tecla = 13 Then
Text2.SetFocus
End If
End Sub
Private Sub Text2_GotFocus()
Text2.SelStart = 0
Text2.SelLength = Len(Text2.Text)
End Sub
Private Sub Text2_KeyPress(Tecla As Integer)
If Tecla = 13 Then
Text3.SetFocus
End If
End Sub
Private Sub Text3_GotFocus()
Text3.SelStart = 0
Text3.SelLength = Len(Text3.Text)
End Sub
Private Sub Text3_KeyPress(Tecla As Integer)
If Tecla = 13 Then
Text4.SetFocus
End If
End Sub
Private Sub Text4_GotFocus()
Text4.SelStart = 0
Text4.SelLength = Len(Text4.Text)
End Sub
Nota: La codificación anterior contiene algunos errores de forma y de fondo.
Corresponderá al estudiante corregirlas y culminar este ejemplo con o sin ayuda del profesor.
Asimismo, el profesor le explicará, paso a paso, la codificación del mismo.
Práctica No. 18
1. Cree una tabla en Microsoft Access o utilizando el VisData de Visual Basic, denominada
Nomina.Mdb y en ella una tabla del mismo nombre con los campos Id_depart(Text 15) y
dep_desc(text 30).
2. Diseñe el siguiente formulario y establezca en el Data Control la base de datos Nomina y
la tabla del mismo nombre.
72
Manual del Alumno
3. Escriba en cualquiera de los botones (los cuales son una matríz de controles) el siguiente
código:
Private Sub Barra_Click(Index As Integer)
Select Case Index
Case 0
If Barra(0).Caption = "Nuevo" Then
Data1.Recordset.AddNew
Barra(0).Caption = "Incluir"
Barra(1).Caption = "Cancelar"
CODIGO.Enabled = True
DESCRIPCION.Enabled = True
For a = 2 To 4
Barra(a).Visible = False
Barra(a).Enabled = False
Next
CODIGO.SetFocus
Else
Data1.Recordset!id_depart = CODIGO.Text
Data1.Recordset!dep_desc = DESCRIPCION.Text
On Error GoTo RUTINA_ERROR
Data1.Recordset.Update
On Error GoTo 0
Barra(0).Caption = "Nuevo"
Barra(1).Caption = "Editar"
CODIGO.Enabled = False
DESCRIPCION.Enabled = False
For a = 2 To 4
Barra(a).Visible = True
Barra(a).Enabled = True
Next
End If
Barra(0).Refresh
73
Manual del Alumno
Barra(1).Refresh
Case 1
If Barra(1).Caption = "Cancelar" Then
CODIGO.Text = ""
DESCRIPCION.Text = ""
Barra(0).Caption = "Nuevo"
Barra(1).Caption = "Editar"
For a = 2 To 4
Barra(a).Visible = True
Barra(a).Enabled = True
Next
Else
Data1.Recordset.Edit
Barra(0).Caption = "Modificar"
Barra(1).Caption = "Cancelar"
DESCRIPCION.Enabled = True
For a = 2 To 4
Barra(a).Visible = False
Barra(a).Enabled = False
Next
DESCRIPCION.SetFocus
End If
Barra(0).Refresh
Barra(1).Refresh
Case 2
Case 3
Case 4
End
End Select
RUTINA_ERROR:
Select Case Err.Number
Case 3022
Msg = "Registro Ya Existe en la Tabla"
MsgBox Msg, , "!!! PANTALLA DE ERRORES !!!"
Err.Clear ' Clear Err object fields
CODIGO.SetFocus
End Select
End Sub
74
Manual del Alumno
SQL
SQL es un lenguaje de programación de base de datos cuyas iniciales vienen de las
palabras inglesas Structured Query Language, que se puede traducir como Lenguaje
Estructurado de Búsqueda o Consulta. Este lenguaje está estrechamente relacionado con la
invención de la base de datos relacional hecha por E.F. Cood a principio de 1970.
El moderno SQL ha evolucionado ampliamente como un estandard en el manejo de
bases de datos relacionales, y está definido por los estandares ANSI. La mayoría de las
implementaciones de SQL tienen pequeñas diferencias de los estandares definidos
incluyendo la version soportada por el motor de bases de datos Microsoft Jet.
SQL Vrs. Navegación:
El motor de base de datos Microsoft Jet provee dos metodos separados para llevar a
cabo la mayoría de tareas relacionadas con las bases de datos:
El modelo navegacional que consiste en ir accesando los registros de la base de datos.
El modelo relacional que está basado en el lenguaje estructurado de búsqueda.
Para los programadores que están familiarizados con sistemas de bases de datos
basados/orientados a archivos como Dbase, Foxpro o Paradox podrían sentirse más comodos
utilizando el metodo navegacional. Sin embargo, en la mayoria de los casos el equivalente
utilizadndo el metodo SQL es mas eficiente y po lo tanto deberian ser tuilizados donde el
rendimiento o tiempo de respuesta es importante. Adicionalmente, SQL tiene la ventaja de
ser la interfase estandar de base de datos de diferentes porveedores, por lo tanto, el
conocimiento de comandos SQL permiten accesar y manipular una gran variedad de
productos de bases de datos.
Codificando consultas con SQL:
El SQL es usado para crear objetos QueryDef utilizandolo como parametro en el
metodo Recordset o en la propiedad de un Data Control. Tambien puede ser utilizado con el
metodo Execute para crear o manipular directamente bases de datos JET y para crear
consultas por Selección o por Acción.
Operaciones DML:
El SQL Data Manipulation Language (Lenguaje de manipulación de datos) son
utilizados para consultar, actualizar, crear o eliminar registros en las tablas de una base de
datos. Esto se realiza mediante el uso de varias instrucciones, pero la mayoría de ellas se
usan con la estructura general del query Select.
COMANDO
CREATE
DESCRIPCIÓN
Se usa para crear nuevas tablas, campos e
75
Manual del Alumno
índices.
Se usa para borrar tablas e índices.
Se usa para modificar tablas, añadiendo campos
o cambiando las definiciones.
Se usa para crear consultas a la base de datos,
según un criterio especificado.
Se usa para cargar datos en la base de datos.
Se usa para actualizar registros de tablas.
Se usa para borrar registros de tablas.
DROP
ALTER
SELECT
INSERT
UPDATE
DELETE
Operadores lógicos:
AND
OR
NOT
Los operadores lógicos se utilizan para conectar expresiones, usualmente en la claúsula
WHERE. Ejemplo:
SELECT *FROM MyTable WHERE condition1 AND condition2
Operadores de comparación:
<
<=
>
>=
<>
BETWEEN
LIKE
IN
Menor que
Menor o igual que
Mayor que
Mayor o igual que
Diferente de
Entre
Parecido a
En
Ejemplo:
SELECT *FROM MyTable WHERE PubId = 5
Sintaxis:
SELECT SELECT [ALL] | DISTINCT | DISTINCTROW | [TOP n [PERCENT]]]
{*| Table.*}| [Table.]field1 [AS alias1] [,[table.]field2[AS alias2][,...]]}
FROM tableexpresion [, ...] [IN externaldatabase]
[WHERE ...]
76
Manual del Alumno
[GROUP BY ...]
[HAVING ...]
[ORDER BY ...]
[WITH OWNERACCESS OPTION]
SELECT field1[, field2[, ...]] INTO newtable [IN externaldatabase] FROM source
INSERT INTO target [IN externaldatabase] [(field1[,field2[, ...]])]
SELECT [source.]field1 [,field2[, ...]
FROM tableexpresion
Single record append query:
INSERT INTO target [(field1[,field2[, ...]]) VALUES (value1[,value2[, ...]])
DELETE [table.*] FROM table WHERE criteria
UPDATE table SET newvalue WHERE criteria;
Ejemplos SQL:
1. SELECT *FROM empleados;
2. SELECT empleados.depto, supervisores.supnombre FROM empleados;
INNER JOIN supervisores ;
WHERE empleados.dpto = supervisores.depto;
3. SELECT BirthDate AS Birth FROM empleados;
4. SELECT COUNT(empleadosID) As HeadCount FROM empleados;
5. SELECT LastName, FirstName, Title, Salary FROM empleados;
AS T1 WHERE Salary >= (SELECT Avg(Salary) FROM empleados;
WHERE T1.Title = empleados.titles) ORDER BY Title;
6. SELECT LastName, Salary FROM empleados WHERE Salary > 100000;
7. SELECT * FROM Orders WHERE ShippedDate = DateValue(„01/02/1999‟);
8. INSERT INTO Customers SELECT * FROM [New Customers];
9. INSERT INTO empleados (FirstName,LastName, Title);
VALUES („Gómez‟,‟Enrique‟Enrique‟,‟Profesor‟);
10. SELECT Title, Count(Title) AS Total FROM empleados;
WHERE Region = „Heredia‟;
GROUP BY Title;
HAVING Count(Title) > 1;
11. SELECT Count(*) AS TotalOrders FROM orders;
12. SELECT IIF(PostalCode BETWEEN 98101 AND 99100, „Local‟, „NonLocal‟);
FROM publishers;
13. SELECT sum(UnitPrice*Quantity) AS [Total UK Sales];
77
Manual del Alumno
FROM orders;
INNER JOIN [Orders Details];
ON orders.OrderId = [Orders Details]. OrderId;
WHERE (ShipCountry = „UK‟);
14. SELECT LastName, FirstName FROM empleados;
WHERE LastName LIKE „[A-D]*‟;
15. SELECT Avg(Freight) AS [Average Freight] FROM orders WHERE Freight > 100;
16. SELECT Count (ShipCountry) AS [UK Orders];
FROM orders WHERE ShipCountry = „UK‟;
17. SELECT DISTINCTROW CompanyName FROM customers;
INNER JOIN orders;
ON customer.customerID = orders.customerID;
ORDER BY CompanyName;
18. UPDATE orders;
SET orderAmount = orderAmount * 1.5;
Freight = Freight + (Freight * .10);
WHERE shipcountry = „UK‟;
78
Manual del Alumno
ACCESOS EXTERNOS:
79
Manual del Alumno
80
Manual del Alumno
81
Manual del Alumno
82
Manual del Alumno