news 2026/1/29 4:13:42

算法题 字母大小写全排列

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
算法题 字母大小写全排列

字母大小写全排列

问题描述

给定一个字符串s,通过将字符串中的每个字母改成大写或小写,生成所有可能的字符串。

返回所有可能的字符串组成的列表。数字和特殊字符保持不变。

示例

输入: s = "a1b2" 输出: ["a1b2", "a1B2", "A1b2", "A1B2"] 输入: s = "3z4" 输出: ["3z4", "3Z4"] 输入: s = "12345" 输出: ["12345"]

算法思路

核心

  1. 字符:只有字母需要考虑大小写变化,数字和特殊字符保持不变
  2. 选择:每个字母有2种选择(大写或小写),如果有k个字母,总共有2^k种排列

方法

  • 回溯:递归处理每个位置,遇到字母时尝试两种大小写
  • 迭代:从空结果开始,逐个字符扩展所有可能的结果
  • 位运算:用二进制位表示每个字母的选择(0=小写,1=大写)

代码实现

方法一:回溯

importjava.util.*;classSolution{/** * 使用回溯生成字母大小写全排列 * * @param s 输入字符串 * @return 所有可能的字符串列表 */publicList<String>letterCasePermutation(Strings){List<String>result=newArrayList<>();// 将字符串转换为字符数组char[]chars=s.toCharArray();// 从索引0开始回溯backtrack(chars,0,result);returnresult;}/** * 回溯函数:递归生成所有可能的排列 * * @param chars 当前字符数组 * @param index 当前处理的字符索引 * @param result 结果列表 */privatevoidbacktrack(char[]chars,intindex,List<String>result){// 递归终止条件:处理完所有字符if(index==chars.length){result.add(newString(chars));return;}charcurrent=chars[index];if(Character.isLetter(current)){// 1:保持小写(或原样)chars[index]=Character.toLowerCase(current);backtrack(chars,index+1,result);// 2:转换为大写chars[index]=Character.toUpperCase(current);backtrack(chars,index+1,result);}else{// 非字母字符,直接跳过backtrack(chars,index+1,result);}}}

方法二:迭代

importjava.util.*;classSolution{/** * 使用迭代生成字母大小写全排列 * * @param s 输入字符串 * @return 所有可能的字符串列表 */publicList<String>letterCasePermutation(Strings){List<String>result=newArrayList<>();result.add("");// 初始化为空字符串// 逐个字符处理for(charc:s.toCharArray()){List<String>newResult=newArrayList<>();// 对当前结果中的每个字符串,扩展新的字符for(Stringstr:result){if(Character.isLetter(c)){// 字母:添加小写和大写两种情况newResult.add(str+Character.toLowerCase(c));newResult.add(str+Character.toUpperCase(c));}else{// 非字母:直接添加newResult.add(str+c);}}result=newResult;}returnresult;}}

方法三:位运算

importjava.util.*;classSolution{/** * 使用位运算生成字母大小写全排列 * * @param s 输入字符串 * @return 所有可能的字符串列表 */publicList<String>letterCasePermutation(Strings){// 首先找出所有字母的位置List<Integer>letterIndices=newArrayList<>();char[]chars=s.toCharArray();for(inti=0;i<chars.length;i++){if(Character.isLetter(chars[i])){letterIndices.add(i);}}intletterCount=letterIndices.size();List<String>result=newArrayList<>();// 枚举所有可能的位组合 (0 到 2^letterCount - 1)for(intmask=0;mask<(1<<letterCount);mask++){char[]current=Arrays.copyOf(chars,chars.length);// 根据mask的每一位决定对应字母的大小写for(inti=0;i<letterCount;i++){if((mask&(1<<i))!=0){// 第i位为1:大写current[letterIndices.get(i)]=Character.toUpperCase(current[letterIndices.get(i)]);}else{// 第i位为0:小写current[letterIndices.get(i)]=Character.toLowerCase(current[letterIndices.get(i)]);}}result.add(newString(current));}returnresult;}}

算法分析

  • 时间复杂度:O(2^n × m)

    • n = 字符串中字母的个数
    • m = 字符串总长度
    • 共有2^n种排列,每种排列需要O(m)时间构建
  • 空间复杂度

    • 回溯:O(m) - 递归深度为m,不包括结果存储
    • 迭代:O(2^n × m) - 需要存储所有中间结果
    • 位运算:O(2^n × m) - 需要存储所有结果

算法过程

s = “a1b2”

回溯

开始: index=0, chars=['a','1','b','2'] ├─ index=0, 'a'是字母 │ ├─ 小写: chars=['a','1','b','2'], index=1 │ │ ├─ index=1, '1'不是字母 → index=2 │ │ │ ├─ index=2, 'b'是字母 │ │ │ │ ├─ 小写: chars=['a','1','b','2'], index=3 │ │ │ │ │ └─ index=3, '2'不是字母 → index=4 → 添加"a1b2" │ │ │ │ └─ 大写: chars=['a','1','B','2'], index=3 │ │ │ │ └─ index=3, '2'不是字母 → index=4 → 添加"a1B2" │ │ └─ 大写: chars=['A','1','b','2'], index=1 │ ├─ index=1, '1'不是字母 → index=2 │ │ ├─ index=2, 'b'是字母 │ │ │ ├─ 小写: chars=['A','1','b','2'], index=3 → 添加"A1b2" │ │ │ └─ 大写: chars=['A','1','B','2'], index=3 → 添加"A1B2"

最终结果:[“a1b2”, “a1B2”, “A1b2”, “A1B2”]

迭代:

  • 初始:[“”]
  • 处理’a’:[“a”, “A”]
  • 处理’1’:[“a1”, “A1”]
  • 处理’b’:[“a1b”, “a1B”, “A1b”, “A1B”]
  • 处理’2’:[“a1b2”, “a1B2”, “A1b2”, “A1B2”]

测试用例

publicstaticvoidmain(String[]args){Solutionsolution=newSolution();// 测试用例1:标准示例System.out.println("Test 1: "+solution.letterCasePermutation("a1b2"));// ["a1b2", "a1B2", "A1b2", "A1B2"]// 测试用例2:包含数字System.out.println("Test 2: "+solution.letterCasePermutation("3z4"));// ["3z4", "3Z4"]// 测试用例3:纯数字System.out.println("Test 3: "+solution.letterCasePermutation("12345"));// ["12345"]// 测试用例4:单个字母System.out.println("Test 4: "+solution.letterCasePermutation("a"));// ["a", "A"]// 测试用例5:空字符串System.out.println("Test 5: "+solution.letterCasePermutation(""));// [""]// 测试用例6:多个连续字母System.out.println("Test 6: "+solution.letterCasePermutation("ab"));// ["ab", "aB", "Ab", "AB"]// 测试用例7:包含大写字母的输入System.out.println("Test 7: "+solution.letterCasePermutation("A1B2"));// ["a1b2", "a1B2", "A1b2", "A1B2"]// 测试用例8:混合大小写和特殊字符System.out.println("Test 8: "+solution.letterCasePermutation("C1@d"));// ["c1@d", "c1@D", "C1@d", "C1@D"]// 测试用例9:较长字符串System.out.println("Test 9: "+solution.letterCasePermutation("a1b2c3"));// 8种组合// 测试用例10:只有特殊字符System.out.println("Test 10: "+solution.letterCasePermutation("!@#$%"));// ["!@#$%"]}

关键点

  1. 字符

    • 使用Character.isLetter()准确判断字母
    • 包含所有Unicode字母,不仅仅是a-z和A-Z
  2. 大小写转换

    • 使用Character.toLowerCase()Character.toUpperCase()
  3. 回溯

    • 修改字符数组时要注意状态恢复
  4. 边界情况

    • 空字符串、纯数字、纯字母等特殊情况

常见问题

  1. 为什么回溯空间复杂度更低?
    • 回溯只在递归栈中存储当前路径
    • 迭代需要存储所有中间结果,空间开销大
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/1/27 1:05:08

对比评测:传统vsAI生成Vue脚手架效率

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 生成一份详细的对比报告&#xff0c;展示&#xff1a;1. 手动配置Vue脚手架的标准流程及耗时&#xff1b;2. 使用AI工具生成相同配置的流程及耗时&#xff1b;3. 两者在依赖安装、配…

作者头像 李华
网站建设 2026/1/28 3:12:32

Flutter状态管理实战:Provider与Riverpod深度对比

一、为什么状态管理是Flutter开发的核心痛点&#xff1f; 在Flutter开发中&#xff0c;状态管理是每个开发者必须面对的挑战。当应用复杂度提升时&#xff0c;你会遇到这些问题&#xff1a; ✘ 父子组件通信繁琐✘ 跨层级数据传递困难✘ 状态更新导致不必要的重建✘ 代码可维…

作者头像 李华
网站建设 2025/12/30 16:35:46

小程序毕设选题推荐:基于微信小程序的智能医疗管理系统设计与实现基于springboot+微信小程序的智能医疗管理系统设计与实现【附源码、mysql、文档、调试+代码讲解+全bao等】

博主介绍&#xff1a;✌️码农一枚 &#xff0c;专注于大学生项目实战开发、讲解和毕业&#x1f6a2;文撰写修改等。全栈领域优质创作者&#xff0c;博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java、小程序技术领域和毕业项目实战 ✌️技术范围&#xff1a;&am…

作者头像 李华
网站建设 2026/1/25 4:31:56

如何用AI自动化解决Mac安全策略配置问题

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 开发一个MacOS自动化工具&#xff0c;能够自动执行以下操作&#xff1a;1) 从MacOS恢复模式启动&#xff1b;2) 导航到安全策略设置&#xff1b;3) 将安全策略更改为完整安全。要求…

作者头像 李华
网站建设 2026/1/1 17:34:51

电商后台管理系统:Vue脚手架实战指南

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 开发一个电商后台管理系统的Vue脚手架模板&#xff0c;要求包含&#xff1a;1. 多角色权限控制模块&#xff1b;2. 商品管理CRUD界面&#xff1b;3. 数据可视化仪表盘&#xff1b;4…

作者头像 李华
网站建设 2026/1/1 17:34:48

传统vsAI设计:电压跟随器开发效率提升300%的秘诀

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 请对比传统设计流程和AI辅助流程开发电压跟随器的效率差异。传统流程要求&#xff1a;1. 手动绘制电路图&#xff1b;2. 计算参数&#xff1b;3. 搭建仿真&#xff1b;4. 迭代优化。…

作者头像 李华