C语言 递归打印出每个数组的组成部分之和等于给定数

zed5wv10  于 2023-10-16  发布在  其他
关注(0)|答案(2)|浏览(140)

所以我们给出一个数组v[]={2,6} and d=10,我们必须打印出每个子数组,以便总和等于d
例如,对于给予d和v[],我们应该打印出这些数组

  1. 2 2 2 2 2
  2. 6 2 2
  3. 2 6 2
  4. 2 2 6

这是我目前的程序,但我似乎不能打印任何东西。
我假设问题是当我回到跟踪,但我没有任何目前的想法

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. void findSolutions(int v[], int n, int d, int current[], int index) {
  4. if(index>n)
  5. {
  6. current=realloc(current,index*sizeof (int));
  7. }
  8. if (d == 0) {
  9. for (int i = 0; i < index; i++) {
  10. printf("%d ", current[i]);
  11. }
  12. printf("\n");
  13. return;
  14. }
  15. if (d < 0 || index >= n) {
  16. return;
  17. }
  18. current[index] = v[index];
  19. findSolutions(v, n, d - v[index], current, index + 1);
  20. findSolutions(v, n, d, current, index + 1);
  21. }
  22. int main() {
  23. int v[] = {2, 6};
  24. int n = 2;
  25. int d = 10;
  26. int *current= malloc(n*sizeof(int));
  27. findSolutions(v, n, d, current, 0);
  28. return 0;
  29. }
64jmpszr

64jmpszr1#

下面的注解似乎解决了您的代码中的几个问题。一致的格式将使您的生活更轻松。

  1. if(index>n) // We'll see this never is true...
  2. {
  3. current=realloc(current, index*sizeof (int)); // so this never happens.
  4. /* Never happening is actually good.
  5. * realloc() may return a new address of heap memory
  6. * You don't pass 'current' back up to higher levels, so their stale copy of
  7. * the pointer 'current' could point to de-allocated memory. (UB awaits!)
  8. */
  9. }
  10. if (d == 0) { // 'd' is not a descriptive variable name
  11. for (int i = 0; i < index; i++) // remove unnecessary braces
  12. printf("%d ", current[i]);
  13. printf("\n");
  14. return;
  15. }
  16. if (d < 0 || index >= n) // Once index = 2, game over. Data requires at least 3
  17. return;
  18. current[index] = v[index]; // probably want "index - 1" for base 0 indexing
  19. findSolutions(v, n, d - v[index], current, index + 1); // close!!!
  20. findSolutions(v, n, d, current, index + 1); // Looks like vestigial attempt
  21. }
  22. int main() {
  23. int v[] = {2, 6}; // Not the best variable names!
  24. int n = 2;
  25. int d = 10;
  26. int *current= malloc(n*sizeof(int)); // could be NULL and let realloc() do the work
  27. findSolutions(v, n, d, current, 0);
  28. return 0;
  29. }

很明显,这段代码正在将目标值(从10)向零(或更低)(即 * 基本情况 *)收缩。聪明,因为这意味着少了一个参数!
下面是一个注解版本。我希望这有助于澄清一些事情。

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. /* Helper function to find minimum value in array */
  4. int arrMinVal( const unsigned arr[], const size_t a_sz ) {
  5. size_t minInd = 0;
  6. for( size_t i = 1; i < a_sz; i++ )
  7. if( arr[ minInd ] > arr[ i ] )
  8. minInd = i;
  9. return arr[ minInd ];
  10. }
  11. /* The recursive function.
  12. * Notice the sequence of parameters:
  13. * - remaining balance of target
  14. * - source array and its size
  15. * - array of values (incl. duplicates) used so far (and its size)
  16. */
  17. int solve_ex( const int remain, const unsigned arr[], const size_t a_sz, int soln[], int len ) {
  18. if( remain < 0 ) // Overshoot! No good, so return...
  19. return 0;
  20. if( !remain ) { // Bull's eye! Print (in reverse) the working combination found
  21. while( len-- )
  22. printf( "%d ", soln[ len ] );
  23. putchar( '\n' );
  24. return 1; // return 1 to increase tally of found combinations
  25. }
  26. // Recurse trying out each value of the source array one at a time
  27. int cnt = 0;
  28. for( size_t i = 0; i < a_sz; i++ ) {
  29. soln[ len ] = arr[ i ]; // Remember this value (for printing)
  30. // Tally the wanted combinations found at "lower levels"
  31. // First parameter 'shrinks' the target toward zero (as you had)
  32. // Last parameter indicates one more value has been added to "soln[]"
  33. cnt += solve_ex( remain - arr[ i ], arr, a_sz, soln, len + 1 );
  34. }
  35. // exhausted possibilities at this level, so return tally
  36. return cnt;
  37. }
  38. /* 'Bridging' function so that main() is less cluttered */
  39. int solve( const int target, const unsigned arr[], const size_t a_sz ) {
  40. // determine max required size of solution array
  41. // For example: with { 2, 6 }, finding 10 requires at most 5 entries (5*2=10)
  42. // Add 1 to allow for "overshoot"...
  43. int n = target / arrMinVal( arr, a_sz ) + 1;
  44. int *soln = malloc( n * sizeof *soln ); // Could be VLA, instead.
  45. /* for brevity, malloc presumed to work */
  46. int cnt = solve_ex( target, arr, a_sz, soln, 0 ); // get into it...
  47. free( soln ); // clean up
  48. return cnt; // return
  49. }
  50. int main( void ) { // Uncluttered...
  51. const unsigned arr[] = { 2, 6 };
  52. const int target = 10;
  53. // Eventually, code to get user input may be added.
  54. // Sometimes, no solutions can be found. Give the user feedback.
  55. printf( "%d solutions found\n", solve( target, arr, sizeof arr/sizeof arr[0] ) );
  56. return 0;
  57. }

我希望这对你有帮助。。

展开查看全部
3phpmpom

3phpmpom2#

您的代码使用malloc(),但没有使用free()来释放内存,这将导致内存泄漏。
我不知道你为什么在findSolutions()中使用两次findSolutions(v, n, d - v[index], current, index + 1);findSolutions(v, n, d, current, index + 1);
所以我重写了一下,这是我的答案:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. void findSolutions(int *a , int* current , int num , int minusResult , int len, int sum)
  5. {
  6. if(sum > minusResult)
  7. return;
  8. else if(sum == minusResult){
  9. for(int i = 0; i < len; i++){
  10. printf("%d ", current[i]);
  11. }
  12. printf("\n");
  13. //printf("len is %d\n",len);
  14. return;
  15. }else{
  16. for(int j = 0; j < num; j++){
  17. sum = sum + a[j];
  18. current[len] = a[j];
  19. findSolutions(a, current, num, minusResult, len+1, sum);
  20. sum = sum - a[j];
  21. }
  22. }
  23. }
  24. int main() {
  25. int v[] = {2, 6};
  26. int n = 2;
  27. int d = 10;
  28. int sum = 0;
  29. int minusResult = 0;
  30. int *current= malloc(1000*sizeof(int));
  31. if(NULL == current){
  32. printf("malloc current failed\n");
  33. return -1;
  34. }
  35. memset(current, 0, 1000*sizeof(int));
  36. findSolutions(v, current, n, d, minusResult, sum);
  37. free(current);
  38. current = NULL;
  39. return 0;
  40. }

运行它将输出:

  1. 2 2 2 2 2
  2. 2 2 6
  3. 2 6 2
  4. 6 2 2

如果你认为2 2 62 6 26 2 2是相同的结果,然后想立即输出2 2 6,你可以使用下面的代码:

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <string.h>
  4. void findSolutions(int *a , int* current , int num , int minusResult , int len, int sum , int loc)
  5. {
  6. if(sum > minusResult)
  7. return;
  8. else if(sum == minusResult){
  9. for(int i = 0; i < len; i++){
  10. printf("%d ", current[i]);
  11. }
  12. printf("\n");
  13. return;
  14. }else{
  15. for(int j = loc; j < num; j++){
  16. sum = sum + a[j];
  17. current[len] = a[j];
  18. findSolutions(a, current, num, minusResult, len+1, sum, loc);
  19. sum = sum - a[j];
  20. loc++;
  21. }
  22. }
  23. }
  24. int main() {
  25. int v[] = {2, 6};
  26. int n = 2;
  27. int d = 10;
  28. int sum = 0;
  29. int loc = 0;
  30. int minusResult = 0;
  31. int *current= malloc(1000*sizeof(int));
  32. if(NULL == current){
  33. printf("malloc current failed\n");
  34. return -1;
  35. }
  36. memset(current, 0, 1000*sizeof(int));
  37. findSolutions(v, current, n, d, minusResult, sum, loc);
  38. free(current);
  39. current = NULL;
  40. return 0;
  41. }

运行它将输出:

  1. 2 2 2 2 2
  2. 2 2 6
展开查看全部

相关问题