警告:格式字符串不是字符串文字

AAV*_*AAV 4 objective-c clang ios

我从以下行获得"格式字符串不是字符串文字"警告

NSString *formattedString = [[NSString alloc] initWithFormat:format arguments:valist];
Run Code Online (Sandbox Code Playgroud)

我在以下功能中使用它

- (void)logMessage:(NSString *)format
         level:(LoggingLevel)level
withParameters:(va_list)valist {
         if (level >= self.loggingLevel) {
              NSString *formattedString = [[NSString alloc] initWithFormat:format arguments:valist];        
         } 
Run Code Online (Sandbox Code Playgroud)

知道如何解决这个问题吗?我正在使用Xcode 4.6.3

tro*_*foe 10

使用以下方法抑制它:

#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wformat-nonliteral"

- (void)logMessage:(NSString *)format
         level:(LoggingLevel)level
withParameters:(va_list)valist {
         if (level >= self.loggingLevel) {
              NSString *formattedString = [[NSString alloc] initWithFormat:format arguments:valist];        
         } 

#pragma clang diagnostic pop
Run Code Online (Sandbox Code Playgroud)


Mar*_*n R 10

If you tell the compiler that your method has a format-like argument, using the NS_FORMAT_FUNCTION macro:

- (void)logMessage:(NSString *)format
         level:(LoggingLevel)level
withParameters:(va_list)valist NS_FORMAT_FUNCTION(1,0) {
         if (level >= self.loggingLevel) {
              NSString *formattedString = [[NSString alloc] initWithFormat:format arguments:valist];        
         } 
}
Run Code Online (Sandbox Code Playgroud)

then

  • the compiler warning in your method goes away, but
  • you get a warning if you call your method with a format string which is not a string literal.

Example:

NSString *abc = @"foo %@ bar";
[self logMessage:abc level:7 withParameters:NULL];

warning: format string is not a string literal [-Wformat-nonliteral]
[self logMessage:abc level:7 withParameters:NULL];
                 ^~~
Run Code Online (Sandbox Code Playgroud)

ADDED: The same applies to the functions mentioned in your comments. They should also be "tagged" with NS_FORMAT_FUNCTION:

+ (void)logVeryFineWithFormat:(NSString *)format, ... NS_FORMAT_FUNCTION(1,2)
{
    va_list ap;
    va_start(ap, format);
    [[self sharedInstance] logMessage:format level:VERY_FINE withParameters:ap];
    va_end(ap);
}

+ (void)say:(NSString *)formatstring, ... NS_FORMAT_FUNCTION(1,2)
{
    va_list arglist;
    va_start(arglist, formatstring);
    // This is not needed: 
    // NSString *litralString = [NSString stringWithFormat:@"%@",formatstring];
    NSString *statement = [[NSString alloc] initWithFormat:formatstring arguments:arglist];
    va_end(arglist);
    [ModalAlert ask:statement withCancel:@"Okay" withButtons:nil];
}
Run Code Online (Sandbox Code Playgroud)