Object-C替换自定义占位符"${}"

本文介绍了一种在iOS开发中自定义字符串替换的方法,通过解析带有${}

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

oc字符串替换一般使用系统提供的%@、%d、%u等,但是有的时候根据需求我们需要自定义占位符,如下:

我们上报服务器数据时,需要向Facebook的渠道做一个转发,eventName是转发的事件名称,其中${level}是需要解析本地数据level,如level = 10,则eventName是为trackleveldone_10.

 //此数据是服务器下发给SDK的,告诉SDK转发的渠道、事件名称。
 {
         "channel":"Facebook",
         "eventName":"trackleveldone_${params.level}"
   }

同时,一个字符串中,可能存在多个${},如:${params.name}_trackleveldone_${params.level}${params.name}_tracklevel_${params.coin}_done_${params.level}等等。

实现

  1. 首先获取,要替换的字符串。
//查找要替换的属性
-(NSArray *) findReplaceHold:(NSString *)holdString {
    if (![holdString isKindOfClass: [NSString class]]) {
        return nil;
    }
    //保存要替换的字符串
    NSMutableArray *array = [[NSMutableArray alloc] initWithCapacity:1];
    BOOL flag = TRUE;
    while (flag) {
        //查询"${}" 
        NSRange startRange = [holdString rangeOfString:@"${"];
        NSRange endRange = [holdString rangeOfString:@"}"];
        //只有两个都匹配,才裁剪,否则表示没有匹配的了。
        if (startRange.location == NSNotFound || endRange.location == NSNotFound) {
            flag = FALSE;
            return array;
        }
        //得出${}包围的字符串
        NSRange sliceRange = NSMakeRange(startRange.location + startRange.length, endRange.location - startRange.location - startRange.length);
        
        if ((startRange.location + startRange.length >= holdString.length) || (endRange.location - startRange.location - startRange.length >= holdString.length)) {
            flag = FALSE;
            return nil;
        }
        //获取${}包围的字符串
        NSString* content = [holdString substringWithRange:sliceRange];
        if (![content isKindOfClass:[NSString class]]) {
            flag = FALSE;
             return nil;
        }
        [array addObject:content];
        //去掉${}的内容
        if (startRange.location >= holdString.length) {
            flag = FALSE;
            return nil;
        }
        NSString* startContent = [holdString substringWithRange:NSMakeRange(0,startRange.location)];
        
        if ((endRange.location + endRange.length >= holdString.length) ||
            holdString.length - endRange.location - endRange.length >= holdString.length) {
            //表示”}“后边没有数据了
            holdString = startContent;
            continue;
        }
        
        NSString* endContent = [holdString substringWithRange:NSMakeRange(endRange.location + endRange.length, holdString.length - endRange.location - endRange.length)];
        //组装数据,"${"前  +  "}"后
        holdString = [NSString stringWithFormat:@"%@%@",startContent,endContent];
    } 
    return array;
}
  1. 获取到替换的字符串后,对此字符串做替换
//查找要替换的属性,replaceValue是key即是上边查到的字符串,value是要替换的值。
-(NSString *) replaceHold:(NSString *)holdString replaceValue:(NSDictionary *)replaceValue {
    if (![holdString isKindOfClass: [NSString class]] || ![replaceValue isKindOfClass:[NSDictionary class]]) {
        return nil;
    }
    BOOL flag = TRUE;
    while (flag) {
        //查询"${ }"
        NSRange startRange = [holdString rangeOfString:@"${"];
        NSRange endRange = [holdString rangeOfString:@"}"];
        
        if (startRange.location == NSNotFound || endRange.location == NSNotFound) {
            flag = FALSE;
            return holdString;
        }
        
        //得出${}包围的字符串
        NSRange sliceRange = NSMakeRange(startRange.location + startRange.length, endRange.location - startRange.location - startRange.length);
        //获取${}包围的字符串
        NSString* content = [holdString substringWithRange:sliceRange];
        if (![content isKindOfClass:[NSString class]]) {
            flag = FALSE;
            return nil;
        }
        //去掉${}的内容
        if (startRange.location >= holdString.length) {
            flag = FALSE;
            return nil;
        }
        NSString* startContent = [holdString substringWithRange:NSMakeRange(0,startRange.location)];
        
        if ((endRange.location + endRange.length >= holdString.length) ||
            holdString.length - endRange.location - endRange.length >= holdString.length) {
            //组装数据,"${"前  +  中
            holdString = [NSString stringWithFormat:@"%@%@",startContent,replaceValue[content]];
            continue;
        }
        NSString* endContent = [holdString substringWithRange:NSMakeRange(endRange.location + endRange.length, holdString.length - endRange.location - endRange.length)];
        
        //组装数据,"${"前  +   中   +  "}"后
        holdString = [NSString stringWithFormat:@"%@%@%@",startContent,replaceValue[content],endContent];
    } 
    return holdString;
}
  1. 使用方式
 //1.获取要替换的字符串
  NSArray* keyArray = [self findReplaceHold:eventName];
  //2. 根据keyArray 去获取程序中对应的value
  NSMutableDictionary* keyDict = [self  findKeyValue: keyArray];
 //3.替换字符串
  NSString* finalEventName = [self replaceHold:eventName replaceValue:keyDict];
<think>好的,我现在需要解决用户的问题:如何在Java中动态替换Word模板中的占位符。用户提到了两种方法,一种是使用poi-tl,另一种是FreeMarker。我需要先理解这两种方法的具体实现步骤,并确保回答结构清晰,符合用户的要求。 首先,用户提供的引用[1]是关于poi-tl的,这是一个基于Apache POI的模板引擎,专门用于处理Word文档。引用[2]则是关于FreeMarker,这是一个模板引擎,通常用于生成文本输出,比如HTML、XML等,但也可以用于Word文档,不过可能需要将Word文档转换为XML格式进行处理。 接下来,我需要整理这两种方法的步骤。对于poi-tl,步骤可能包括添加依赖、创建模板、编写替换代码。而对于FreeMarker,可能需要将Word另存为XML,处理变量,然后生成文档。需要注意的是,FreeMarker处理Word可能较为复杂,因为需要直接操作XML内容,而poi-tl更直接,因为它专为Word设计。 另外,用户可能关心这两种方法的优缺点。例如,poi-tl是否支持复杂的格式,FreeMarker是否适合简单的替换。需要指出poi-tl更适合保留格式,而FreeMarker可能需要更多手动处理。 还要考虑用户可能的疑问,比如如何处理循环、条件语句,或者在替换时格式是否会受到影响。在回答中可能需要提到这些点,但根据用户的问题,主要是替换占位符,所以可能不需要深入,但生成的相关问题可以涉及这些方面。 然后,检查是否需要符合系统级指令,比如数学公式的格式,但这个问题不涉及数学内容,所以可能不需要。但需要确保使用中文,结构清晰,分步骤说明,并添加相关问题和引用标识。 最后,生成相关问题时要考虑用户可能进一步想知道的内容,比如性能比较、其他工具推荐,或者如何处理复杂模板。确保问题覆盖用户可能的后续疑问。</think>在Java中实现Word模板占位符替换主要有两种主流方案,以下是具体实现步骤和对比分析: ### 方案一:使用poi-tl模板引擎(推荐) 1. **添加依赖** ```xml <dependency> <groupId>com.deepoove</groupId> <artifactId>poi-tl</artifactId> <version>1.12.1</version> </dependency> ``` 2. **创建模板文件** 新建.docx文件,使用`{{variable}}`格式定义占位符,如: ``` 尊敬的{{name}}: 您的订单号:{{orderNo}} ``` 3. **核心替换代码** ```java XWPFTemplate template = XWPFTemplate.compile("template.docx").render( new HashMap<String, Object>(){{ put("name", "张三"); put("orderNo", "20230815001"); }}); template.writeAndClose(new FileOutputStream("output.docx")); ``` 该方法支持保留原始格式、图片替换、表格循环等复杂操作[^1]。 ### 方案二:FreeMarker+XML方案 1. **准备模板文件** 将.docx另存为XML格式,使用`${variable}`格式定义变量: ```xml <w:t>${title}</w:t> ``` 2. **Java处理代码** ```java Configuration cfg = new Configuration(Configuration.VERSION_2_3_30); cfg.setClassForTemplateLoading(this.getClass(), "/templates"); Template template = cfg.getTemplate("template.xml"); Map<String, String> data = new HashMap<>(); data.put("title", "项目报告"); data.put("author", "李四"); StringWriter writer = new StringWriter(); template.process(data, writer); String modifiedContent = writer.toString(); // 将XML转换回.docx格式 FileUtils.writeStringToFile(new File("output.xml"), modifiedContent); ``` 需手动处理XML到DOCX的转换,适合简单文本替换[^2]。 ### 方案对比 | 特性 | poi-tl | FreeMarker | |-----------|--------------------------|--------------------| | 格式保留 | ✅完整支持 | ❌需手动处理XML结构 | | 图片替换 | ✅直接支持 | ❌无法直接操作 | | 表格循环 | ✅内置语法 | ❌需自定义实现 | | 学习曲线 | 中等 | 较高(需理解Word XML)| | 性能 | 较高 | 中等 | ### 最佳实践建议 1. 优先选择poi-tl方案,其语法更简洁且直接操作.docx文件 2. 需要处理表格循环时,使用poi-tl的区块控制语法: ```java put("tableData", MiniTable.create( new RowRenderData(Arrays.asList("名称", "价格"), "#eebf34"), Arrays.asList( new RowRenderData(Arrays.asList("商品A", "¥99"), null), new RowRenderData(Arrays.asList("商品B", "¥129"), null) ) )); ``` 3. 处理图片替换时: ```java put("logo", Pictures.ofLocal("logo.png").size(100, 100).create()); ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值