2009-07-01 8 views
22

Mam ciąg URL w następującym formacie.Konwersja & do & w Objective-C

http://myserver.com/_layouts/feed.aspx?xsl=4&web=%2F&page=dda3fd10-c776-4d69-8c55-2f1c74b343e2&wp=476f174a-82df-4611-a3df-e13255d97533

Chcę zastąpić & z & w powyższym adresie URL. Mój wynik powinien być:

http://myserver.com/_layouts/feed.aspx?xsl=4&web=%2F&page=dda3fd10-c776-4d69-8c55-2f1c74b343e2&wp=476f174a-82df-4611-a3df-e13255d97533

Czy ktoś może odpowiedzieć mi kod, aby to zrobić?

Dzięki

Odpowiedz

14
[urlString stringByReplacingOccurrencesOfString:@"&" withString:@"&"]; 
+0

zrobiłem to samo ... ale czy jest jakiś sposób, aby wbudowane zrób to ... – nbojja

+2

@nbojja Ile więcej wbudowałeś, ile chcesz? Jeśli to dotyczy, dodaj metodę, która robi to jako kategorię na NSString, a następnie jest wbudowana. – Abizern

+10

@Abizern: Wiele języków ma wbudowane metody kodowania i dekodowania obiektów HTML, Obj-C brakuje tego i wielu innych rzeczy, które programiści biorą za pewnik od 2002 roku. Wyszukiwanie i zastępowanie jest złym substytutem, ponieważ będziesz musiał spędzić sporo czasu, aby wiedzieć, że dostajesz wszystkie jednostki. –

8

Nie ma wbudowanej funkcji do tego w iPhone SDK. Powinieneś file a bug, że chcesz funkcjonalność. W normalnym pakiecie Mac OS X SDK możesz albo załadować ten fragment do postaci NSAttributedString jako HTML i poprosić go o przekazanie zwykłego ciągu znaków, albo użyć CFXMLCreateStringByUnescapingEntities().

@interface NSString (LGAdditions) 
- (NSString *) stringByUnescapingEntities; 
@end 

@implementation NSString (LGAdditions) 
- (NSString *) stringByUnescapingEntities { 
    CFStringRef retvalCF = CFXMLCreateStringByUnescapingEntities(kCFAllocatorDefault, (CFStringRef)self, NULL); 
    return [NSMakeCollectable(retvalCF) autorelease]; 
} 
@end 
+0

To nie działa z funkcją automatycznego liczenia odwołań (ARC) {sigh} – mpemburn

+0

@mpemburn czy próbowałeś: 'CFStringRef retvalCF = CFXMLCreateStringByUnescapingEntities (kCFAllocatorDefault, (__bridge CFAllocatorRef) self, NULL); return (NSString *) CFBridgingRelease (retvalCF); ' –

+0

Nie należy go łączyć z CFAllocatorRef, ale raczej CFStringRef. To było również błędne w oryginalnym kodzie. – dgatwood

113

Sprawdź mój NSString category for HTML. Oto dostępne metody:

// Strips HTML tags & comments, removes extra whitespace and decodes HTML character entities. 
- (NSString *)stringByConvertingHTMLToPlainText; 

// Decode all HTML entities using GTM. 
- (NSString *)stringByDecodingHTMLEntities; 

// Encode all HTML entities using GTM. 
- (NSString *)stringByEncodingHTMLEntities; 

// Minimal unicode encoding will only cover characters from table 
// A.2.2 of http://www.w3.org/TR/xhtml1/dtds.html#a_dtd_Special_characters 
// which is what you want for a unicode encoded webpage. 
- (NSString *)stringByEncodingHTMLEntities:(BOOL)isUnicode; 

// Replace newlines with <br /> tags. 
- (NSString *)stringWithNewLinesAsBRs; 

// Remove newlines and white space from string. 
- (NSString *)stringByRemovingNewLinesAndWhitespace; 
+0

Dzięki za to, Michael - bardzo przydatny! (Tak przydatne, jak odpowiedź na to pytanie, które zostało zaakceptowane, jest błędne!) –

+0

Nie ma problemu;) Cieszę się, że okazało się przydatne! –

+0

Tak, bardzo przydatne, dzięki Michael – Jack

4

Dla iOS następujący kod powinien działać dla kodów numerycznych. Powinien on być stosunkowo łatwo rozszerzyć do m.in. &amp; ...

-(NSString*)unescapeHtmlCodes:(NSString*)input { 

NSRange rangeOfHTMLEntity = [input rangeOfString:@"&#"]; 
if(NSNotFound == rangeOfHTMLEntity.location) { 
    return input; 
} 


NSMutableString* answer = [[NSMutableString alloc] init]; 
[answer autorelease]; 

NSScanner* scanner = [NSScanner scannerWithString:input]; 
[scanner setCharactersToBeSkipped:nil]; // we want all white-space 

while(![scanner isAtEnd]) { 

    NSString* fragment; 
    [scanner scanUpToString:@"&#" intoString:&fragment]; 
    if(nil != fragment) { // e.g. '&#38; B' 
     [answer appendString:fragment];   
    } 

    if(![scanner isAtEnd]) { // implicitly we scanned to the next '&#' 

     int scanLocation = (int)[scanner scanLocation]; 
     [scanner setScanLocation:scanLocation+2]; // skip over '&#' 

     int htmlCode; 
     if([scanner scanInt:&htmlCode]) { 
      char c = htmlCode; 
      [answer appendFormat:@"%c", c]; 

      scanLocation = (int)[scanner scanLocation]; 
      [scanner setScanLocation:scanLocation+1]; // skip over ';' 

     } else { 
      // err ? 
     } 
    } 

} 

return answer; 

} 

Niektóre kodu Jednostka test ...

-(void)testUnescapeHtmlCodes { 

NSString* expected = @"A & B"; 
NSString* actual = [self unescapeHtmlCodes:@"A &#38; B"]; 
STAssertTrue([expected isEqualToString:actual], @"actual = %@", actual); 

expected = @"& B"; 
actual = [self unescapeHtmlCodes:@"&#38; B"];  
STAssertTrue([expected isEqualToString:actual], @"actual = %@", actual); 

expected = @"A &"; 
actual = [self unescapeHtmlCodes:@"A &#38;"]; 
STAssertTrue([expected isEqualToString:actual], @"actual = %@", actual); 

}