2009-11-05 13 views

Odpowiedz

5

Oto alternatywne podejście, które może również pomóc. Załóżmy, że masz NSString* który zawiera adres IP, o nazwie ipAddressStr, w formacie 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]); 
} 

Można modyfikować warunkowego bloku if podążać RFC 1918 wytyczne, jeśli trzeba, że ​​poziom walidacji.

+0

Alex, czy chciałeś wstawić "&" przed każdym argumentem ipQuad do sscanf? –

+0

Chciałem, ale nie zrobiłem. Będę edytować odpowiedź po dodaniu tego komentarza. Zmieniłem też typ 'ipQuads' z' unsigned char' na 'unsigned int', aby test mógł się zawieść. –

+0

Ponieważ iOS 4 RegExps są dostępne w SDK iOS: http://developer.apple.com/library/IOs/#documentation/Foundation/Reference/NSRegularExpression_Class/Reference/Reference.html ... więc, może jeśli chcesz zaznaczyć ta odpowiedź jako przestarzała "ale przydatna do celów informacyjnych lub edukacyjnych" pomoże :) –

1

Sztuczka można zrobić, to sprawdzić powrót inet_aton rozmowy BSD tak:

#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; 
} 

Należy pamiętać jednak, że ta sprawdza ciąg czy zawiera on adres IP w dowolnym formacie, to nie jest ograniczony do formatu kropkowanego.

+0

FYI: 'inet_aton' jest przestarzałe na rzecz' inet_pton', który obsługuje IPv6. – mattt

+1

Proszę, jeśli yo zwraca wartość boolowską, jeśli sprawdzisz, jeśli zwrócisz samo sprawdzanie, np. 'Return success == 1' –

21

Oto kategoria wykorzystująca nowoczesny inet_pton, który zwróci TAK dla prawidłowego ciągu IPv4 lub IPv6.

#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

Dla odniesienia w przyszłości sukces == 1? TRUE: FALSE nie jest potrzebne. Prosty sukces == 1 zadziałałby. – varikin

0

wydanie 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 
} 
Powiązane problemy