java配置文件转义问题

场景: 
    配置文件config.properties 
    配置项cfg.regexp=\d+\t 
    加载配置文件代码 
    InputStream ins = PropManager.class.getResourceAsStream("/config.properties"); 
    prop.load(ins); 

现象: 
    打印cfg.regexp的值输出为"d+    ",而不是期望的"\d+\t" 

原因: 
    看了下代码,发现Properties类对'\'做了转义处理,而且只处理'\uxxxx', '\t','\n','\r','\f'这几种情况,对于其他情况,就简单的把'\'吞掉了 

 1 while (off < end) {  
 2             aChar = in[off++];  
 3             if (aChar == '\\') {  
 4                 aChar = in[off++];     
 5                 if(aChar == 'u') {  
 6                     // Read the xxxx  
 7                     int value=0;  
 8             for (int i=0; i<4; i++) {  
 9                 aChar = in[off++];    
10                 switch (aChar) {  
11                   case '0': case '1': case '2': case '3': case '4':  
12                   case '5': case '6': case '7': case '8': case '9':  
13                      value = (value << 4) + aChar - '0';  
14                  break;  
15               case 'a': case 'b': case 'c':  
16                           case 'd': case 'e': case 'f':  
17                  value = (value << 4) + 10 + aChar - 'a';  
18                  break;  
19               case 'A': case 'B': case 'C':  
20                           case 'D': case 'E': case 'F':  
21                  value = (value << 4) + 10 + aChar - 'A';  
22                  break;  
23               default:  
24                               throw new IllegalArgumentException(  
25                                            "Malformed \\uxxxx encoding.");  
26                         }  
27                      }  
28                     out[outLen++] = (char)value;  
29                 } else {  
30                     if (aChar == 't') aChar = '\t';   
31                     else if (aChar == 'r') aChar = '\r';  
32                     else if (aChar == 'n') aChar = '\n';  
33                     else if (aChar == 'f') aChar = '\f';   
34                     out[outLen++] = aChar;  
35                 }  
36             } else {  
37             out[outLen++] = (char)aChar;  
38             }  
39         }  

解决方法: 
我的解决方法是,自己写个简单的解析 
约定 

    • 每行第一次出现的'='作为分割符(Properties类可以支持key:value格式)
    • key和value都忽略前后空白字符
    • 不对字符串做转义
 1 while((line = buffReader.readLine())!=null){  
 2     line = line.trim();  
 3     if(line.startsWith("#") || line.equals("")){// 忽略#开头的注释  
 4         continue;  
 5     }  
 6       
 7     int index = line.indexOf('=');  
 8     if(index <= 0){  
 9         logger.error("********错误的配置文件格式!********line = " + line);  
10         continue;  
11     }  
12     String key = line.substring(0, index).trim();  
13     String value = (index+1>=line.length()) ? "":line.substring(index+1).trim();// 避免越界  
14     prop.put(key, value);  
15 }  

之所以把它记下来是因为我跟同事说起这个现象的时候,同事发现他之前写的代码用过类似的正则表达式配置方式,正则表达式里面希望匹配绝对的'\.',而经过Properties类转义之后变成了'.',意义就变成了“任意字符”,由于运行并不报错,而且某系情况下结果还是正确的,于是一个潜在的bug就出现了 

 

posted @ 2018-05-03 10:48  光何  阅读(2157)  评论(0编辑  收藏  举报