Como eliminar campos de una tabla en access
Tengo una tabla con 2 campos que tiene la siguiente info:
Nombre Id de Pedido (estos son los 2 campos)
condición cliente
juan perez 2
juan diaz 3
condición no cliente
maria perez 4
maria diaz 5
condición cliente
juana lopez 6
La tabla contiene muchos registros y lo que necesito es ELIMINAR todos los registros de los No Clientes de la tabla, ¿sería con un loop? Alguna idea que me puedan ayudar, desde ya muy agradecido !
Una SQL para eliminar los registros que cumplan la condición de que el contenido del campo sea un valor numérico
Opciones:
CurrentDb.Execute "Delete * From [aqui-el-nombre-de-la-tabla] Where Val ([Id de Pedido]) = 0" CurrentDb.Execute "Delete * From [aqui-el-nombre-de-la-tabla] Where IsNumeric ([Id de Pedido])"
CurrentDb.Execute y DoCmd.RunSQL son equivalentes, pero no genera mensajes por lo que no se necesita manipularlos.
Val ([Id de Pedido]) = 0 .... la función VAL extrae del campo el valor numérico (devuelve cero si no hay dígitos en al inicio)
IsNumeric ([Id de Pedido]) ... La función IsNumeric devuelve un Verdadero / Falso si el contenido del campo no se puede evaluar como numero o el campo no existe/localiza.
Enrique antes que nada Muchas gracias. TE cuento que el problema que tengo es que la tabla es bastante rara y necesito eliminar los registros desde una línea que dice Condición No cliente, desde ahí para abajo todos los registros hasta que encuentre una línea que diga Condición Cliente.
Todos esos registros que quedan en el medio son los que quiero eliminar y luego que siga recorriendo la tabla hasta que vuelva a encontrar otra línea que diga Cond No cliente y borrar todos lo registros debajo hasta que nueva mente aparezca un campo que diga Condición Cliente y ahí pare y borre todo eso .
No se si fui claro.
Muchas gracias!
Lo que solicitas solo se puede garantizar si el orden es invariable (y no parece que exista un orden).
Si el origen real fuera un listado en TXT seria más sencillo, pero si es lo único disponible ...
Pregunta:
¿Seria válido el crear una tabla y traspasar los grupas de registros de tipo 'cliente'?.
(Copiándolos, el original quedaría intacto)
Si es válido sugiere un nombre para la nueva tabla (y el nombre de la actual), porque se requiere VBA y no se que dominio tienes sobre ese entorno a momento actual.
Si el proceso se va a repetir lo crearía como una función a la que se la pasarían dos parámetros (datos), el nombre de la tabla que cede los datos y el de tabla destino (se asumen que los nombres de los campos y las condiciones se mantienen constantes).
Anoto algo más.
¿El dato numérico se respeta o se remunera?, si esta relacionado con otra tabla debería respetarse.
Todos las datos serán clientes + ID, el registro que los delata no se traspasa a la nueva tabla.
El listado original es un txt que lo importo en una tabla, luego de sacarle varios caracteres . Como dices, no existe un orden.
El nombre de la tabla actual es APEFALL y la nueva tabla ser[ia CLIENTES.
GRACIAS
Escribí algo así pero me da error, tampoco es que esté seguro de que vaya a estar bien...
Private Sub Elimina_EXCLIENTES()
Dim rst As DAO.Recordset 'Creas la variable recordset
Set rst = CurrentDb.OpenRecordset("APEFALL") '
If rst.RecordCount = 0 Then GoTo Salida
rst.MoveFirst
Do
Do While APEFALL.APELLIDO_NOMBRE = " CLI - CLIENTE"
If APEFALL.APELLIDO_NOMBRE = " EXC - EXCLIENTE" Then
CurrentDb.Execute "Delete * From APEFALL"
rst.MoveNext
End If
Loop
Loop Until APEFALL.APELLIDO_NOMBRE = " CLI - CLIENTE"
Salida:
rst.Close
Set rst = Nothing
End Sub
Trabajando con el original se minimizan los errores.
(La alternativa es añadir un autonumérico en la tabla 'APEFALL', no importa el valor numérico, solo nos interesa la numeración correlativa para mantener el orden original)
Este tipo de importaciones suele ser reiterativo, por lo que aporto la función con dos versiones, una requiere nombres (para crear diferentes tablas, por ejemplo por años), la otra asume los nombres actuales.
La que solicita nombres se utiliza así : Extraer ("APEFALL ", "CLIENTES")
Public Function Extraer(Torigen$, Tdestino$) Dim Xcondicion As Boolean, Tbl_Temp As DAO.Recordset Set Tbl_Temp = CurrentDb.OpenRecordset(Torigen, , dbReadOnly) With Tbl_Temp If .RecordCount = 0 Then Exit Function .MoveLast: .MoveFirst Xcondicion = IIf(!id_de_pedido = "cliente", -1, 0) Do Until .EOF If !nombre = "condición" Then Xcondicion = IIf(!id_de_pedido = "cliente", -1, 0): .MoveNext If Xcondicion Then CurrentDb.Execute "Insert into " & Tdestino & " (nombre, id_de_pedido) Values ('" & !nombre & "', '" & !id_de_pedido & "')" .MoveNext Loop .Close Set Tbl_Temp = Nothing End With End Function
La que no solicita nombres se utiliza asi : Extraer ()
Public Function Extraer() Dim Xcondicion As Boolean, Tbl_Temp As DAO.Recordset Set Tbl_Temp = CurrentDb.OpenRecordset("APEFALL", , dbReadOnly) With Tbl_Temp If .RecordCount = 0 Then Exit Function .MoveLast: .MoveFirst Xcondicion = IIf(!id_de_pedido = "cliente", -1, 0) Do Until .EOF If !nombre = "condición" Then Xcondicion = IIf(!id_de_pedido = "cliente", -1, 0): .MoveNext If Xcondicion Then CurrentDb.Execute "Insert into Clientes (nombre, id_de_pedido) Values ('" & !nombre & "', '" & !id_de_pedido & "')" .MoveNext Loop .Close Set Tbl_Temp = Nothing End With End Function
El resultado obtenido (con los datos publicados)
nombre id_de_pedido
juan perez 2
juan diaz 3
juana lopez 6
Consulta cualquier duda y si ofreces un trozo del TXT original se puede prescindir de la tabla APFALL por innecesaria.
¿No seria interesante vaciar la tabla APFALL tras procesarla para evitar duplicados si se ejecuta más de una vez con los mismos datos? ... Innecesario si la tabla Clientes esta indexada sin duplicados pero ...
- Compartir respuesta
2 respuestas más de otros expertos
Le dejo este ejemplo, no requiere crear más tablas, solo mediante código adiciono un campo auxiliar que llamo "tomar" tipo byte para marcar los registros que debo eliminar.
TABLA ORIGINAL
TABLA ACTUALIZADA
CODIGO DE LA FUNCIÓN (Lo puede cambiar como un procedimiento)
Public Function retira() Dim db As Database Dim rs As Recordset Dim strAux As String Dim flag As Boolean Set db = CurrentDb 'Adiciono un campo auxiliar CurrentDb.Execute "ALTER TABLE tblClientes ADD COLUMN tomar BYTE;" CurrentDb.TableDefs.Refresh Set rs = db.OpenRecordset("tblClientes") Do Until rs.EOF If rs!APELLIDO_NOMBRE = "EXC - CLIENTE" Then flag = True rs.Edit rs!tomar = 1 rs.Update rs.MoveNext End If If rs!APELLIDO_NOMBRE <> "CLI - CLIENTE" And flag = True Then rs.Edit rs!tomar = 1 rs.Update rs.MoveNext Else flag = False rs.MoveNext End If Loop rs.Close Set rs = Nothing 'Retiro los NA: CurrentDb.Execute "DELETE FROM tblClientes WHERE LEFT(DOCUMENTO,2)='" & "NA'" 'Retiro los registro elegidos en el campo auxiliar tomar CurrentDb.Execute "DELETE FROM tblClientes WHERE tomar=1" 'Elimino el campo auxiliar tomar CurrentDb.Execute "ALTER TABLE tblClientes DROP COLUMN tomar;" CurrentDb.TableDefs.Refresh End Function
En caso que no quiere crear por código el campo "tomar" puede crearlo directamente en diseño de tipo byte y suprimir las líneas que lo crea y elimina de la función,
Si quiere el ejemplo puede solicitarlo a [email protected], favor en el asunto anotar la consulta.
- Compartir respuesta
No dices desde donde quieres eliminarlo, si desde un formulario o una consulta de eliminación. Vamos a suponer que es desde un formulario. En cualquier evento, por ejemplo al hacer clic de un botón, puedes poner
Docmd.setwarnings false
docmd.runsql"delete * from Nombredetutabla where [Id de pedido] ="" No cliente """
Lo de set Warnings es para que no te aparezca la dichosa ventana de "Va a eliminar..."
Vamos a suponer que es desde una consulta
Donde yo pongo pais tu tienes que poner el campo Id de Pedido y debajo, donde yo pongo Como "Alemania" tu tienes que poner Como "No cliente"
Es que me expresé mal...la tabla se ve así tal como la mostré, el nombre del campo es Id de Pedido
y en ese campo tenemos 1, 2, 3, cliente, no cliente, 4 ,5 6
Nombre Id de Pedido (estos son los 2 campos)
condición cliente
juan perez 2
juan diaz 3
condición no cliente
maria perez 4
maria diaz 5
condición cliente
juana lopez 6
y quiero que la tabla me quede resultado:
Nombre Id de Pedido (estos son los 2 campos)
juan perez 2
juan diaz 3
juana lopez 6
que serían solamente los clientes reales...es una tabla rara, sí, pero es la que tengo que trabajar...lo que tengo que hacer es eliminar las líneas que dicen condición cliente, condición no cliente, y todos los registros que van debajo de no cliente hasta que vuelva a aparecer el registro cliente
Muchas gracias
Pues es como te he dicho. Lo que hace la instrucción de VB o la consulta de eliminación es, valga la redundancia, eliminar de la tabla todos aquellos registros en que el valor del campo Id de Pedido sea No cliente.
¡Gracias!
Pero es que no son los registros en los que dice No cliente los que tengo que eliminar, sino que son los que tienen una numeración y que están debajo de ese registro
Chico, la culpa es mía, estaba obsesionado. Basta conque pongas
Docmd.runsql"delete * from nombretabla where [id de pedido]<>""No cliente """
Corrijo, por fin leí, lo que tenía que eliminar. Ponlo como
Docmd.runsql"delete * from tabla where [id de pedido]<>""cliente"""
Es decir, elimina todos aquellos registros en que Id de pedido sea distinto de Cliente
Aca te paso una imagen de como se ve mi tabla, porque es dificil explicarlo...Lo que necesito eliminar son todos los registros que est[an entre la fila EXC-EXCLIENTE y la fila CLI-CLIENTE
Escribí esto pero no me funciona, a ver si me puedes ayudar
Private Sub Elimina_EXCLIENTES()
Dim rst As DAO.Recordset 'Creas la variable recordset
Set rst = CurrentDb.OpenRecordset("APEFALL") '
If rst.RecordCount = 0 Then GoTo Salida
rst.MoveFirst
Do
Do While APEFALL.APELLIDO_NOMBRE = " CLI - CLIENTE"
If APEFALL.APELLIDO_NOMBRE = " EXC - EXCLIENTE" Then
CurrentDb.Execute "Delete * From APEFALL"
rst.MoveNext
End If
Loop
Loop Until APEFALL.APELLIDO_NOMBRE = " CLI - CLIENTE"
Salida:
rst.Close
Set rst = Nothing
End Sub
¿Entonces de Id de Pedido nada de nada? Hubiera sido mucho más fácil empezar con la imagen
Si tengo la tabla Tabla1 como
Y un formulario con un botón, ya dije que no sería necesario, por que se podría poner en cualquier otro evento
En el evento Al hacer clic del botón le pongo
Private Sub Comando48_Click() DoCmd.RunSQL "delete * from tabla1 where apellido_nombre <>""cli - cliente"" and apellido_nombre <> ""ex - excliente""" End Sub
De forma que cuando lo pulso me deja la tabla como
- Compartir respuesta