2009-11-05 15 views

Respuesta

5

Aquí hay un enfoque alternativo que también podría ayudar. Supongamos que usted tiene una NSString* que contiene su dirección IP, llamada ipAddressStr, del formato a.b.c.d:

int ipQuads[4]; 
const char *ipAddress = [ipAddressStr cStringUsingEncoding:NSUTF8StringEncoding]; 

sscanf(ipAddress, "%d.%d.%d.%d", &ipQuads[0], &ipQuads[1], &ipQuads[2], &ipQuads[3]); 

@try { 
    for (int quad = 0; quad < 4; quad++) { 
     if ((ipQuads[quad] < 0) || (ipQuads[quad] > 255)) { 
     NSException *ipException = [NSException 
      exceptionWithName:@"IPNotFormattedCorrectly" 
      reason:@"IP range is invalid" 
      userInfo:nil]; 
     @throw ipException; 
     } 
    } 
} 
@catch (NSException *exc) { 
    NSLog(@"ERROR: %@", [exc reason]); 
} 

podrá modificar el bloque condicional if seguir RFC 1918 directrices, si necesita ese nivel de validación.

+0

Alex, ¿quisiste poner un '&' antes de cada argumento de ipQuad en sscanf? –

+0

Tenía la intención, pero no lo hice. Editaré la respuesta después de agregar este comentario. Además, cambié el tipo de 'ipQuads' de' unsigned char' a 'unsigned int', de modo que la prueba puede fallar. –

+0

Desde iOS 4 RegExps están disponibles en iOS SDK: http://developer.apple.com/library/IOs/#documentation/Foundation/Reference/NSRegularExpression_Class/Reference/Reference.html ... así que, tal vez, si desea marcar esta respuesta como obsoleta "pero útil para fines informativos o educativos" ayudará :) –

1

Un truco que puede hacer es probar el retorno de la llamada BSD inet_aton así:

#include <arpa/inet.h> 

- (BOOL)isIp:(NSString*)string{ 
    struct in_addr pin; 
    int success = inet_aton([string UTF8String],&pin); 
    if (success == 1) return TRUE; 
    return FALSE; 
} 

Sé sin embargo consciente de que este valida la cadena si ésta contiene una dirección IP en cualquier formato, no se limita al formato punteado.

+0

FYI: 'inet_aton' ha quedado en desuso en favor de' inet_pton', que maneja IPv6. – mattt

+1

Por favor, si está devolviendo un valor booleano de un cheque if, simplemente devuelva el cheque if, es decir 'return success == 1' –

21

Aquí hay una categoría que utiliza el moderno inet_pton que devolverá SÍ para una cadena IPv4 o IPv6 válida.

#include <arpa/inet.h> 

    @implementation NSString (IPValidation) 

    - (BOOL)isValidIPAddress 
    { 
     const char *utf8 = [self UTF8String]; 
     int success; 

     struct in_addr dst; 
     success = inet_pton(AF_INET, utf8, &dst); 
     if (success != 1) { 
      struct in6_addr dst6; 
      success = inet_pton(AF_INET6, utf8, &dst6); 
     } 

     return success == 1; 
    } 

    @end 
+3

Para referencia futura, ¿el éxito == 1? VERDADERO: FALSO no es necesario. Un simple éxito == 1 funcionaría. – varikin

0

edición Swift:

func isIPAddressValid(ip: String) -> Bool { 
    guard let utf8Str = (ip as NSString).utf8String else { 
     return false 
    } 

    let utf8:UnsafePointer<Int8> = UnsafePointer(utf8Str) 
    var success: Int32 

    var dst: in_addr = in_addr() 
    success = inet_pton(AF_INET, utf8, &dst) 
    if (success != 1) { 
     var dst6: in6_addr? = in6_addr() 
     success = inet_pton(AF_INET6, utf8, &dst6); 
    } 

    return success == 1 
} 
Cuestiones relacionadas