热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

深入解析暴力递归:从左至右的尝试模型,真实Facebook面试题目

题目要求将数字字符串转换为对应的字母组合,例如“111”可以转化为“AAA”、“KA”或“AK”。本文通过深入解析暴力递归方法,详细探讨了这一问题的解法,并结合真实的Facebook面试题目,提供了从左至右尝试模型的具体实现和优化策略。

题目:规定1和A对应、2和B对应、3和C对应…那么一个数字字符串比如"111”就可以转化为:'AAA"、 “KA"和"AK”。给定一个只有数字字符组成的字符串str,返回有多少种转化结果

我们先来分析下题目:

数字1对应字母A,2对应B,…一直到26对应Z。数字0是没有字符与之对应的,也就是说,如果碰到0是没有转换结果的

仔细分析这道题,会发现在暴力递归中,这是一个从左往右的尝试模型

什么是从左往右的尝试模型呢,之前写过的打印一个字符串的全部排列和子序列都是这种模型,简单来说,就是从左往右一个一个暴力尝试。

下面我们具体分析:

1)首先考虑base case,如果i位置来到字符串的终止位置,有两种可能,字符串没有字符了,所以返回空字符,算是一种结果;或者之前已经有转换好的答案,所以返回一个结果。

2)如果当前位置的字符是在3~9之间,i+1位置无论是什么字符,i和i+1连起来都会超过26,所以这种情况下,i位置单独转换,然后去i+1位置递归。

3)当前位置是1字符,i+1位置无论任何字符,i和i+1连起来都不会超过26,所以又有两种可能。第一种:i位置单独转换,去i+1位置递归;第二种:i和i+1一起转换,去i+2位置递归。

4)当前位置是2字符,如果i+1位置的字符是在[0~6],i和i+1可以一起转换,去i+2位置递归;否则,只能i单独转换,去i+1位置递归。

给大家画个图:

在这里插入图片描述
当然,博主一开始写这种题也不会有这么清晰的思路,也只能是写一步看一步,发现写不下去了,就检查是不是缺少变量啥的导致无法写下去。总之就是要硬写+多写,哈哈哈。

package com.harrison.class12;public class Code05_ConvertToLetterString {// 0~i-1位置已经转换好了&#xff0c;做好决定了&#xff0c;不用再考虑// i及其往后的位置有多少种转换的结果public static int process1(char[] str,int i) {// i来到终止位置&#xff0c;没有字符了&#xff0c;转换为空字符// 要么认为0~i-1已经转换好了&#xff0c;返回一个点数if(i&#61;&#61;str.length) {return 1;}// i没有到终止位置if(str[i]&#61;&#61;&#39;0&#39;) {// 0没有对应的字符return 0;}// i没有到终止位置 && i位置不是0字符// i位置是1字符&#xff0c;i&#43;1位置无论是什么字符都不会超过26&#xff0c;都可以转换if(str[i]&#61;&#61;&#39;1&#39;) {int res&#61;process1(str, i&#43;1);// i位置自己单独转换&#xff0c;后续有多少种方法if(i&#43;1<str.length) {res&#43;&#61;process1(str, i&#43;2);// i和i&#43;1位置一起转换&#xff0c;后续有多少种方法}return res;}// i位置是2字符&#xff0c;如果i&#43;1位置的字符是0~6&#xff0c;i和i&#43;1就可以一起转换&#xff0c;然后i&#43;2位置接着递归// 否则只能i位置自己转换&#xff0c;然后i&#43;1位置接着递归if(str[i]&#61;&#61;&#39;2&#39;) {int res&#61;process1(str, i&#43;1);if(i&#43;1<str.length && (str[i&#43;1]>&#61;&#39;0&#39; && str[i&#43;1]<&#61;&#39;6&#39;)) {res&#43;&#61;process1(str, i&#43;2);}return res;}return process1(str, i&#43;1);}public static int convert1(String s) {if(s&#61;&#61;null || s.length()&#61;&#61;0) {return 0;}char[] str&#61;s.toCharArray();return process1(str, 0);}public static void main(String[] args) {String test&#61;"111";System.out.println(convert1(test));}
}

另外据说这道题是曾经Facebook的面试真题哦&#xff0c;你搞懂了吗

上面的暴力尝试并不是最优的&#xff0c;所以在这基础上改动态规划也会比较麻烦&#xff0c;下面的写法的暴力尝试比上面的好。

package com.inspect.inspect100;/*** &#64;author Harrison* &#64;create 2022-05-09-14:55* &#64;motto 众里寻他千百度&#xff0c;蓦然回首&#xff0c;那人却在灯火阑珊处。*/
public class T068_ConvertToLetterString {public static int convert1(String s){if(s&#61;&#61;null || s.length()&#61;&#61;0){return 0;}char[] str&#61;s.toCharArray();return process1(str,0);}public static int process1(char[] str,int i){if(i&#61;&#61; str.length){return 1;}if(str[i]&#61;&#61;&#39;0&#39;){return 0;}int ans&#61;process1(str,i&#43;1);if(i&#43;1<str.length && ((str[i]-&#39;0&#39;)*10&#43;str[i&#43;1]-&#39;0&#39;<27)){ans&#43;&#61;process1(str,i&#43;2);}return ans;}public static int dp(String s){if(s&#61;&#61;null || s.length()&#61;&#61;0){return 0;}char[] str&#61;s.toCharArray();int N&#61;str.length;int[] dp&#61;new int[N&#43;1];dp[N]&#61;1;for(int i&#61;N-1; i>&#61;0; i--){if(str[i]!&#61;&#39;0&#39;){int ans&#61;dp[i&#43;1];if(i&#43;1<str.length && ((str[i]-&#39;0&#39;)*10&#43;str[i&#43;1]-&#39;0&#39;<27)){ans&#43;&#61;dp[i&#43;2];}dp[i]&#61;ans;}}return dp[0];}public static void main(String[] args) {String test&#61;"11111";int ans1&#61;convert1(test);int ans2&#61;dp(test);System.out.println(ans1);System.out.println("&#61;&#61;&#61;&#61;&#61;&#61;");System.out.println(ans2);}
}


推荐阅读
  • andr ... [详细]
  • 本文详细介绍了如何构建一个高效的UI管理系统,集中处理UI页面的打开、关闭、层级管理和页面跳转等问题。通过UIManager统一管理外部切换逻辑,实现功能逻辑分散化和代码复用,支持多人协作开发。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文介绍如何利用动态规划算法解决经典的0-1背包问题。通过具体实例和代码实现,详细解释了在给定容量的背包中选择若干物品以最大化总价值的过程。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 主要用了2个类来实现的,话不多说,直接看运行结果,然后在奉上源代码1.Index.javaimportjava.awt.Color;im ... [详细]
  • 本文深入探讨了 Java 中的 Serializable 接口,解释了其实现机制、用途及注意事项,帮助开发者更好地理解和使用序列化功能。 ... [详细]
  • 本文介绍了如何在C#中启动一个应用程序,并通过枚举窗口来获取其主窗口句柄。当使用Process类启动程序时,我们通常只能获得进程的句柄,而主窗口句柄可能为0。因此,我们需要使用API函数和回调机制来准确获取主窗口句柄。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • 本文介绍了如何通过 Maven 依赖引入 SQLiteJDBC 和 HikariCP 包,从而在 Java 应用中高效地连接和操作 SQLite 数据库。文章提供了详细的代码示例,并解释了每个步骤的实现细节。 ... [详细]
  • 本文详细介绍了Java中的访问器(getter)和修改器(setter),探讨了它们在保护数据完整性、增强代码可维护性方面的重要作用。通过具体示例,展示了如何正确使用这些方法来控制类属性的访问和更新。 ... [详细]
  • 本文介绍如何使用阿里云的fastjson库解析包含时间戳、IP地址和参数等信息的JSON格式文本,并进行数据处理和保存。 ... [详细]
  • Scala 实现 UTF-8 编码属性文件读取与克隆
    本文介绍如何使用 Scala 以 UTF-8 编码方式读取属性文件,并实现属性文件的克隆功能。通过这种方式,可以确保配置文件在多线程环境下的一致性和高效性。 ... [详细]
  • 本文提供了使用Java实现Bellman-Ford算法解决POJ 3259问题的代码示例,详细解释了如何通过该算法检测负权环来判断时间旅行的可能性。 ... [详细]
  • 探讨如何通过编程技术实现100个并发连接,解决线程创建顺序问题,并提供高效的并发测试方案。 ... [详细]
author-avatar
joyjz
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有