news 2026/1/17 10:10:39

算法题 字母大小写全排列

作者头像

张小明

前端开发工程师

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进行投诉反馈,一经查实,立即删除!
网站建设 2025/12/27 8:00:35

如何用AI自动修复Unsupported Media Type错误

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 开发一个AI辅助调试工具&#xff0c;能够自动检测HTTP请求中的Unsupported Media Type错误。当用户输入一个导致415错误的API请求示例时&#xff0c;系统应分析请求头中的Content-T…

作者头像 李华
网站建设 2025/12/27 1:09:19

Flutter入门实战:手把手教你构建第一个跨平台应用

一、前言&#xff1a;为什么选择Flutter&#xff1f; 在移动开发领域&#xff0c;跨平台框架层出不穷。而 Flutter 凭借其高性能、高一致性、热重载&#xff08;Hot Reload&#xff09;等优势&#xff0c;迅速成为 Google 主推的 UI 框架&#xff0c;并被阿里巴巴、腾讯、字节…

作者头像 李华
网站建设 2026/1/16 23:24:55

25、Unix 文件和目录管理全解析

Unix 文件和目录管理全解析 1. 目录基础概念 在 Unix 系统里,目录是一个简单却重要的概念。它就像一个列表,包含了一系列文件名,每个文件名都对应着一个索引节点(inode)编号。这里,每个文件名被称为一个目录项,而文件名与 inode 编号的映射关系则被叫做链接。当我们使…

作者头像 李华
网站建设 2026/1/16 9:47:35

Redis客户端工具新手指南:从安装到基本操作

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 创建一个交互式Redis学习工具&#xff0c;内置新手教程模块。通过分步引导教用户安装Redis客户端、连接服务器、执行SET/GET等基础命令。包含常见错误模拟和解决方法&#xff0c;比…

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

开发智能体,用Python还是Java?

在开发 AI智能体应用&#xff08;Agent-based AI Applications&#xff09;时&#xff0c;Python 与 Java 技术栈的选择并非简单的“二选一”&#xff0c;而是 高度依赖场景、团队能力、系统边界和长期演进需求。以下是基于 2025年技术生态 的深度对比与决策指南&#xff08;结…

作者头像 李华