当前位置:   article > 正文

代码随想录算法训练营第22天-leetcode-回溯算法part01:

代码随想录算法训练营第22天-leetcode-回溯算法part01:

#回溯算法理论基础

能解决的问题:

  • 组合问题:N个数里面按一定规则找出k个数的集合
  • 切割问题:一个字符串按一定规则有几种切割方式
  • 子集问题:一个N个数的集合里有多少符合条件的子集
  • 排列问题:N个数按一定规则全排列,有几种排列方式
  • 棋盘问题:N皇后,解数独等等

第77题. 组合

力扣题目链接(opens new window)

给定两个整数 n 和 k,返回 1 ... n 中所有可能的 k 个数的组合。

示例: 输入: n = 4, k = 2 输出: [ [2,4], [3,4], [2,3], [1,2], [1,3], [1,4], 

注意:

1、对于是递归回溯问题,用树图来考虑问题!+使用基本结构

        同时,也要积极分析如何剪枝

2、路径类问题的标准套路

        在函数外开辟path 和 ans 一层空间 ans=(int**)malloc:一层空间,二层空间还没开辟

        终止条件 if(一条路径完成了) 把path放入ans数组

                先开辟ans的二层空间,ans【i】=(int*)malloc

                放入path的过程需要用循环一个个放入,直接=path的话,后面会随path修改而修改

        递归体:填充path

3、报错分析:

遇见heap堆错误,找malloc相关的;遇见stack栈报错,找函数内数组是否越界

4、returnsize 和 return column

*returnsize 在函数调用中无需&,且指向个数,而非下标

column的赋值过程:*column是正常数组,先为*column开辟空间,*column【第几个,<returnsize】=ans【第几个】有多少个二层元素

分析:

代码:

  1. void bf(int *path,int n,int start,int k,int *pathlength,int **ans,int *returnSize){
  2. if(*pathlength == k-1){//路径类问题的标准输出
  3. ans[++(*returnSize)]=(int *)malloc(sizeof(int)*k);
  4. for (int i=0;i<k;i++){
  5. ans[*returnSize][i]=path[i];
  6. }
  7. return;
  8. }
  9. for(int i=start;i<=n-(k-*pathlength-2);i++){
  10. //遍历各个树
  11. //剪枝:如果后面全放进去,也达不到k个个数,那么就不考虑了
  12. path[++(*pathlength)]=i;
  13. bf(path,n,i+1,k,pathlength,ans,returnSize);
  14. (*pathlength)--;//回溯 步骤!!
  15. }
  16. }
  17. int** combine(int n, int k, int* returnSize, int** returnColumnSizes) {
  18. int **ans=(int **)malloc(sizeof(int *)*200001);
  19. *returnSize=-1;
  20. int pathlength=-1;//考虑路径,用到path,pathlength
  21. int *path=(int *)malloc(sizeof(int )*k);
  22. bf(path,n,1, k, &pathlength,ans,returnSize);//returnsize不需要&
  23. (*returnSize)++;//returnsize指向数组的实际大小
  24. *returnColumnSizes=(int*)malloc(sizeof(int )*(*returnSize));//column的意义
  25. for(int i=0;i<(*returnSize);i++){
  26. (*returnColumnSizes)[i]=k;
  27. }
  28. return ans;
  29. }

216.组合总和III

力扣题目链接(opens new window)

找出所有相加之和为 n 的 k 个数的组合。组合中只允许含有 1 - 9 的正整数,并且每种组合中不存在重复的数字。

说明:

  • 所有数字都是正整数。
  • 解集不能包含重复的组合。

示例 1: 输入: k = 3, n = 7 输出: [[1,2,4]]

示例 2: 输入: k = 3, n = 9 输出: [[1,2,6], [1,3,5], [2,3,4]

  1. void bp(int **ans,int *size,int *path,int *p,int k,int n,int start,int sum){
  2. if(sum>n) return;//剪枝
  3. if(sum==n && *p==k){
  4. ans[*size]=(int *)malloc(sizeof(int)*k);
  5. for (int i=0;i<k;i++){
  6. ans[*size][i]=path[i];
  7. }
  8. (*size)++;
  9. return;//不要忘记写return
  10. }
  11. else if(*p==k){//另外一种终止情况
  12. return;
  13. }
  14. for(int i=start;i<=9;i++){
  15. path[(*p)++]=i;
  16. bp(ans, size, path, p, k, n, i+1, sum+i);//i+1,而不是start+1
  17. (*p)--;
  18. }
  19. }
  20. int** combinationSum3(int k, int n, int* returnSize, int** returnColumnSizes) {
  21. int **ans=(int **)malloc(sizeof(int *)*500);
  22. int size=0;
  23. int *path=(int *)malloc(sizeof(int)*k);
  24. int p=0;
  25. bp(ans, &size,path, &p, k, n, 1, 0);
  26. *returnSize=size;
  27. *returnColumnSizes=(int *)malloc(sizeof(int)*(size));
  28. for (int i=0;i<size;i++){
  29. (*returnColumnSizes)[i]=k;//要加括号
  30. }
  31. return ans;
  32. }

17.电话号码的字母组合

力扣题目链接(opens new window)

给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。

给出数字到字母的映射如下(与电话按键相同)。注意 1 不对应任何字母。

  1. char phoneMap[11][5] = {"\0", "\0", "abc\0", "def\0", "ghi\0", "jkl\0", "mno\0", "pqrs\0", "tuv\0", "wxyz\0"};
  2. void bp(char**ans,int *size,char *path,int *p,int len,char* digits,int d){
  3. if(*p==len){
  4. ans[*size] =(char*)malloc(sizeof(char)*(len+1));
  5. for(int i=0;i<len;i++){
  6. ans[*size][i]=path[i];
  7. printf("%c",path[i]);
  8. }
  9. ans[*size][len]='\0';
  10. printf("\n");
  11. (*size)++;
  12. return;
  13. }
  14. int number= digits[d]-'0';
  15. char * nowd=phoneMap[number];
  16. int dlen=strlen(nowd);
  17. for(int i=0;i<dlen;i++){
  18. char new=nowd[i];
  19. path[(*p)++]=new;
  20. bp(ans, size, path,p, len, digits,d+1);
  21. (*p)--;
  22. }
  23. }
  24. char** letterCombinations(char* digits, int* returnSize) {
  25. int len=strlen(digits);
  26. char**ans=(char**)malloc(sizeof(char*)*pow(4,len));
  27. int size=0;
  28. if (len==0) {
  29. *returnSize=0;
  30. return ans;
  31. }
  32. char *path=(char*)malloc(sizeof(char)*(len+1));
  33. int p=0;
  34. bp(ans,&size, path, &p, len, digits, 0);
  35. *returnSize=size;
  36. return ans;
  37. }

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小桥流水78/article/detail/912169
推荐阅读
相关标签
  

闽ICP备14008679号