Estoy trabajando en detectar errores en mi aplicación y estoy pensando en usarlos NSError
. Estoy un poco confundido acerca de cómo usarlo y cómo poblarlo.
¿Podría alguien dar un ejemplo de cómo uso y luego uso NSError
?
Bueno, lo que suelo hacer es que mis métodos que pueden generar errores en el tiempo de ejecución hagan referencia a un NSError
puntero. Si algo sale mal en ese método, puedo rellenar la NSError
referencia con datos de error y devolver nil desde el método.
Ejemplo:
- (id) endWorldHunger:(id)largeAmountsOfMonies error:(NSError**)error {
// begin feeding the world's children...
// it's all going well until....
if (ohNoImOutOfMonies) {
// sad, we can't solve world hunger, but we can let people know what went wrong!
// init dictionary to be used to populate error object
NSMutableDictionary* details = [NSMutableDictionary dictionary];
[details setValue:@"ran out of money" forKey:NSLocalizedDescriptionKey];
// populate the error object with the details
*error = [NSError errorWithDomain:@"world" code:200 userInfo:details];
// we couldn't feed the world's children...return nil..sniffle...sniffle
return nil;
}
// wohoo! We fed the world's children. The world is now in lots of debt. But who cares?
return YES;
}
Entonces podemos usar el método como este. Ni siquiera se moleste en inspeccionar el objeto de error a menos que el método devuelva nil:
// initialize NSError object
NSError* error = nil;
// try to feed the world
id yayOrNay = [self endWorldHunger:smallAmountsOfMonies error:&error];
if (!yayOrNay) {
// inspect error
NSLog(@"%@", [error localizedDescription]);
}
// otherwise the world has been fed. Wow, your code must rock.
Pudimos acceder a los errores localizedDescription
porque establecemos un valor para NSLocalizedDescriptionKey
.
El mejor lugar para obtener más información es la documentación de Apple . Realmente es bueno.
También hay un tutorial simple y agradable sobre Cocoa Is My Girlfriend .
id
aBOOL
. Cualquier ligera variación compatible con ARC sería muy apreciada.BOOL
. RegreseNO
en caso de error y en lugar de verificar el valor de retorno, simplemente verifiqueerror
. Sinil
sigue adelante, si lo!= nil
maneja.**error
no es nulo. De lo contrario, el programa arrojará un error que es completamente hostil y no hace evidente lo que está sucediendo.Me gustaría agregar algunas sugerencias más basadas en mi implementación más reciente. He visto algunos códigos de Apple y creo que mi código se comporta de la misma manera.
Las publicaciones anteriores ya explican cómo crear objetos NSError y devolverlos, por lo que no me molestaré con esa parte. Solo intentaré sugerir una buena forma de integrar errores (códigos, mensajes) en su propia aplicación.
Recomiendo crear 1 encabezado que será una descripción general de todos los errores de su dominio (es decir, aplicación, biblioteca, etc.). Mi encabezado actual se ve así:
FSError.h
FSError.m
Ahora, al usar los valores anteriores para errores, Apple creará un mensaje de error estándar básico para su aplicación. Se podría crear un error como el siguiente:
El mensaje de error estándar (
error.localizedDescription
) generado por Apple para el código anterior tendrá el siguiente aspecto:Error Domain=com.felis.myapp Code=1002 "The operation couldn’t be completed. (com.felis.myapp error 1002.)"
Lo anterior ya es bastante útil para un desarrollador, ya que el mensaje muestra el dominio donde ocurrió el error y el código de error correspondiente. Sin
1002
embargo, los usuarios finales no tendrán idea de qué significa el código de error , por lo que ahora necesitamos implementar algunos mensajes agradables para cada código.Para los mensajes de error, debemos tener en cuenta la localización (incluso si no implementamos mensajes localizados de inmediato). He utilizado el siguiente enfoque en mi proyecto actual:
1) crea un
strings
archivo que contendrá los errores. Los archivos de cadenas son fácilmente localizables. El archivo podría tener el siguiente aspecto:FSError.strings
2) Agregue macros para convertir códigos enteros en mensajes de error localizados. He usado 2 macros en mi archivo Constantes + Macros.h. Siempre incluyo este archivo en el encabezado del prefijo (
MyApp-Prefix.pch
) por conveniencia.Constantes + Macros.h
3) Ahora es fácil mostrar un mensaje de error fácil de usar basado en un código de error. Un ejemplo:
fuente
Constants+Macros.h
e importo este archivo en el encabezado (.pch
archivo) del prefijo para que esté disponible en todas partes. Si quiere decir que solo está usando 1 de las 2 macros, eso podría funcionar. Quizás la conversión deint
aNSString
no sea realmente necesaria, aunque no he probado esto..strings
archivo), ya que ahí es donde se verá la macro de Apple. Lea sobre el usoNSLocalizedStringFromTable
aquí: developer.apple.com/library/mac/documentation/cocoa/conceptual/…FS_ERROR_LOCALIZED_DESCRIPTION
verifica la cadena localizable en un archivo llamadoFSError.strings
. Es posible que desee consultar la guía de localización de.strings
archivos de Apple si esto le resulta extraño.Gran respuesta Alex. Un problema potencial es la desreferencia NULL. Referencia de Apple sobre creación y devolución de objetos NSError
fuente
C objetivo
Swift 3
fuente
Consulte el siguiente tutorial
Espero que sea útil para usted, pero antes de que tenga que leer la documentación de NSError
Este es un enlace muy interesante que encontré recientemente ErrorHandling
fuente
Trataré de resumir la gran respuesta de Alex y el punto de jlmendezbonini, agregando una modificación que hará que todo sea compatible con ARC (hasta ahora no es así, ya que ARC se quejará, ya que debe regresar
id
, lo que significa "cualquier objeto", peroBOOL
no es un objeto tipo).Ahora, en lugar de verificar el valor de retorno de nuestra llamada al método, verificamos si
error
todavía estánil
. Si no es así, tenemos un problema.fuente
Otro patrón de diseño que he visto implica el uso de bloques, que es especialmente útil cuando un método se ejecuta de forma asincrónica.
Digamos que tenemos definidos los siguientes códigos de error:
Definiría su método que puede generar un error así:
Y luego, cuando lo llame, no necesita preocuparse por declarar el objeto NSError (la finalización del código lo hará por usted) o verificar el valor devuelto. Solo puede suministrar dos bloques: uno que se llamará cuando haya una excepción y otro que se llamará cuando tenga éxito:
fuente
Bueno, está un poco fuera de alcance, pero en caso de que no tenga una opción para NSError, siempre puede mostrar el error de nivel bajo:
fuente
que puedo usar
NSError.defaultError()
siempre que no tenga un objeto de error válido.fuente