Day22.一刷数据结构算法(C语言版) 216组合总和III;17电话号码的字母组合

本文主要是介绍Day22.一刷数据结构算法(C语言版) 216组合总和III;17电话号码的字母组合,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、216组合总和III

        如果把昨天的组合问题理解了,本题就容易一些了。 

        题目链接:组合总和III

        文章讲解:代码随想录

        视频讲解:和组合问题有啥区别?回溯算法如何剪枝?| 216.组合总和III

1.思路分析

        相对于77. 组合,无非就是多了一个限制,本题是要找到和为n的k个数的组合,而整个集合已经是固定的了[1,...,9]。

        本题k相当于树的深度,9(因为整个集合就是9个数)就是树的宽度。

        例如 k = 2,n = 4的话,就是在集合[1,2,3,4,5,6,7,8,9]中求 k(个数) = 2, n(和) = 4的组合。

        选取过程如图:

        图中,可以看出,只有最后取到集合(1,3)和为4 符合条件。

        回溯三部曲:

        1)确定递归函数参数

        本题依然需要一维数组path来存放符合条件的结果,二维数组result来存放结果集。

        这里我依然定义path 和 result以及各自的栈顶指针pathTop、ansTop为全局变量。

        至于为什么取名为path?从上面树形结构中,可以看出,结果其实就是一条根节点到叶子节点的路径。

int* path;
int pathTop;
int** ans;
int ansTop;

        接下来还需要如下参数:

  • targetSum(int)目标和,也就是题目中的n。
  • k(int)就是题目中要求k个数的集合。
  • sum(int)为已经收集的元素的总和,也就是path里元素的总和。
  • startIndex(int)为下一层for循环搜索的起始位置
void backtracking(int targetSum, int k, int sum, int startIndex) 

        还要强调一下,回溯法中递归函数参数很难一次性确定下来,一般先写逻辑,需要啥参数了,填什么参数。 

        2)确定终止条件

        在上面已经说了,k其实就已经限制树的深度,因为就取k个元素,树再往下深了没有意义。

所以如果path.size() 和 k相等了,就终止。

        如果此时path里收集到的元素和(sum) 和targetSum(就是题目描述的n)相同了,就用result收集当前的结果。

    if(pathTop == k) {if(sum == targetSum) {int* tempPath = (int*)malloc(sizeof(int) * k);int j;for(j = 0; j < k; j++)tempPath[j] = path[j];ans[ansTop++] = tempPath;}// 如果path.size() == k 但sum != targetSum 直接返回return;}

        3)单层搜索过程

        本题是集合固定的就是9个数[1,...,9],所以for循环固定i<=9

        如图:

        处理过程就是 path收集每次选取的元素,相当于树型结构里的边,sum来统计path里元素的总和。

int i;//从startIndex开始遍历,一直遍历到9for (i = startIndex; i <= 9; i++) {sum += i; // 处理path[pathTop++] = i; // 处理backtracking(targetSum, k, sum, i + 1); // 注意i+1调整startIndexsum -= i; // 回溯pathTop--;; // 回溯}

        别忘了处理过程 和 回溯过程是一一对应的,处理有加,回溯就要有减! 

2.代码详解

/*** Return an array of arrays of size *returnSize.* The sizes of the arrays are returned as *returnColumnSizes array.* Note: Both returned array and *columnSizes array must be malloced, assume caller calls free().*/
int* path;
int pathTop;
int** ans;
int ansTop;
int getPathSum() {int i;int sum = 0;for(i = 0; i < pathTop; i++) {sum += path[i];}return sum;
}void backtracking(int targetSum, int k, int sum, int startIndex) {if(pathTop == k) {if(sum == targetSum) {int* tempPath = (int*)malloc(sizeof(int) * k);int j;for(j = 0; j < k; j++)tempPath[j] = path[j];ans[ansTop++] = tempPath;}// 如果path.size() == k 但sum != targetSum 直接返回return;}int i;//从startIndex开始遍历,一直遍历到9for (i = startIndex; i <= 9; i++) {sum += i; // 处理path[pathTop++] = i; // 处理backtracking(targetSum, k, sum, i + 1); // 注意i+1调整startIndexsum -= i; // 回溯pathTop--;; // 回溯}
}int** combinationSum3(int k, int n, int* returnSize, int** returnColumnSizes){//初始化辅助变量path = (int*)malloc(sizeof(int) * k);ans = (int**)malloc(sizeof(int*) * 20);pathTop = ansTop = 0;backtracking(n, k, 0, 1);//设置返回的二维数组中元素个数为ansTop*returnSize = ansTop;//设置二维数组中每个元素个数的大小为k*returnColumnSizes = (int*)malloc(sizeof(int) * ansTop);int i;for(i = 0; i < ansTop; i++) {(*returnColumnSizes)[i] = k;}return ans;
}

 二、17电话号码的字母组合

        本题大家刚开始做会有点难度,先自己思考20min,没思路就直接看题解。 

        题目链接:电话号码的字母组合

        文章讲解:代码随想录

        视频讲解:还得用回溯算法!| LeetCode:17.电话号码的字母组合

1.思路分析

        这道题要解决如下三个问题:

        数字和字母如何映射

        两个字母就两个for循环,三个字符我就三个for循环,以此类推,然后发现代码根本写不出来

        输入1 * #按键等等异常情况(力扣测试数据并没有涉及到这类情况,所以代码中暂时也不考虑,不过我们要知道有这些异常)

        数字和字母如何映射:

        这里定义一个二维数组,代码如下: 

char* letterMap[10] = {"", //0"", //1"abc", //2"def", //3"ghi", //4"jkl", //5"mno", //6"pqrs", //7"tuv", //8"wxyz", //9
};

        回溯法来解决n个for循环的问题:

        例如:输入:"23",抽象为树形结构,如图所示:

        图中可以看出遍历的深度,就是输入"23"的长度,而叶子节点就是我们要收集的结果,输出["ad", "ae", "af", "bd", "be", "bf", "cd", "ce", "cf"]。

        回溯三部曲:

        1)确定输入参数

        首先需要一个字符串s来收集叶子节点的结果,然后用一个字符串数组result保存起来,这两个变量我依然定义为全局。别忘了还有对应栈顶指针。

        再来看参数,参数指定是有题目中给的string digits,然后还要有一个参数就是int型的index。

        注意这个index可不是 77.组合 和216.组合总和III 中的startIndex了。

        这个index是记录遍历第几个数字了,就是用来遍历digits的(题目中给出数字字符串),同时index也表示树的深度。

char* path;
int pathTop;
char** result;
int resultTop;void backTracking(char* digits, int index)

        2)确定终止条件

        例如输入用例"23",两个数字,那么根节点往下递归两层就可以了,叶子节点就是要收集的结果集。

        那么终止条件就是如果index 等于 输入的数字个数了(本来index就是用来遍历digits的)。

        然后收集结果,结束本层递归。

//若当前下标等于digits数组长度if(index == strlen(digits)) {//复制digits数组,因为最后要多存储一个0,所以数组长度要+1char* tempString = (char*)malloc(sizeof(char) * strlen(digits) + 1);int j;for(j = 0; j < strlen(digits); j++) {tempString[j] = path[j];}//char数组最后要以0结尾tempString[strlen(digits)] = 0;result[resultTop++] = tempString;return ;}

         3)确定单层遍历逻辑

        首先要取index指向的数字,并找到对应的字符集(手机键盘的字符集)。

        然后for循环来处理这个字符集,代码如下:

//将字符数字转换为真的数字int digit = digits[index] - '0';//找到letterMap中对应的字符串char* letters = letterMap[digit];int i;for(i = 0; i < strlen(letters); i++) {path[pathTop++] = letters[i];//递归,处理下一层数字backTracking(digits, index+1);pathTop--;}

        注意这里for循环,可不像是在组合问题中从startIndex开始遍历的

        因为本题每一个数字代表的是不同集合,也就是求不同集合之间的组合,而组合问题是求同一个集合中的组合!

2.代码详解

/*** Note: The returned array must be malloced, assume caller calls free().*/
char* path;
int pathTop;
char** result;
int resultTop;
char* letterMap[10] = {"", //0"", //1"abc", //2"def", //3"ghi", //4"jkl", //5"mno", //6"pqrs", //7"tuv", //8"wxyz", //9
};
void backTracking(char* digits, int index) {//若当前下标等于digits数组长度if(index == strlen(digits)) {//复制digits数组,因为最后要多存储一个0,所以数组长度要+1char* tempString = (char*)malloc(sizeof(char) * strlen(digits) + 1);int j;for(j = 0; j < strlen(digits); j++) {tempString[j] = path[j];}//char数组最后要以0结尾tempString[strlen(digits)] = 0;result[resultTop++] = tempString;return ;}//将字符数字转换为真的数字int digit = digits[index] - '0';//找到letterMap中对应的字符串char* letters = letterMap[digit];int i;for(i = 0; i < strlen(letters); i++) {path[pathTop++] = letters[i];//递归,处理下一层数字backTracking(digits, index+1);pathTop--;}
}char ** letterCombinations(char * digits, int* returnSize){//初始化path和resultpath = (char*)malloc(sizeof(char) * strlen(digits));result = (char**)malloc(sizeof(char*) * 300);*returnSize = 0;//若digits数组中元素个数为0,返回空集if(strlen(digits) == 0) return result;pathTop = resultTop = 0;backTracking(digits, 0);*returnSize = resultTop;return result;
}

        今天难度有点大,慢慢消化吧。 

 

 

 

 

 

 

这篇关于Day22.一刷数据结构算法(C语言版) 216组合总和III;17电话号码的字母组合的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/941054

相关文章

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

hdu4869(逆元+求组合数)

//输入n,m,n表示翻牌的次数,m表示牌的数目,求经过n次操作后共有几种状态#include<iostream>#include<algorithm>#include<cstring>#include<stack>#include<queue>#include<set>#include<map>#include<stdio.h>#include<stdlib.h>#includ

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

秋招最新大模型算法面试,熬夜都要肝完它

💥大家在面试大模型LLM这个板块的时候,不知道面试完会不会复盘、总结,做笔记的习惯,这份大模型算法岗面试八股笔记也帮助不少人拿到过offer ✨对于面试大模型算法工程师会有一定的帮助,都附有完整答案,熬夜也要看完,祝大家一臂之力 这份《大模型算法工程师面试题》已经上传CSDN,还有完整版的大模型 AI 学习资料,朋友们如果需要可以微信扫描下方CSDN官方认证二维码免费领取【保证100%免费

6.1.数据结构-c/c++堆详解下篇(堆排序,TopK问题)

上篇:6.1.数据结构-c/c++模拟实现堆上篇(向下,上调整算法,建堆,增删数据)-CSDN博客 本章重点 1.使用堆来完成堆排序 2.使用堆解决TopK问题 目录 一.堆排序 1.1 思路 1.2 代码 1.3 简单测试 二.TopK问题 2.1 思路(求最小): 2.2 C语言代码(手写堆) 2.3 C++代码(使用优先级队列 priority_queue)

dp算法练习题【8】

不同二叉搜索树 96. 不同的二叉搜索树 给你一个整数 n ,求恰由 n 个节点组成且节点值从 1 到 n 互不相同的 二叉搜索树 有多少种?返回满足题意的二叉搜索树的种数。 示例 1: 输入:n = 3输出:5 示例 2: 输入:n = 1输出:1 class Solution {public int numTrees(int n) {int[] dp = new int