分享

计算思维实践之路(八)

 宇哥小屋 2020-01-05

             2月19日,多云。“小楼一夜听春雨,深巷明朝卖杏花“。

      例1:输入一个数n,按字典序从小到大的顺序输出1~n的全排列。两个序列的字典序大小关系等价于从开始的第一个不相同位置处的大小关系。例如,(1,3,2) <  (2,1,3)    
       当n = 3时,所有排序的排序结果是(1,2,3)、(1,3,2)、(2,1,3)、(2,3,1)、(3,1,2)、(3,2,1)。

       俺老猪擅长穷举法:寻找问题解的一种可靠的方法是首先列出所有可能候选解,然后依次检查每一个,在检查完所有或部分候选解后,即可找到所需要的解。理论上,当候选解数量有限并且通过检查所有或部分候选解能够得到所需解时,上述方法是可行的。不过在实际应用中,很少使用这种方法,因为候选解的数量通常都非常大,即便采用最快的计算机也只能解决规模很小的问题。

  1. #include <cstring>
  2. #include <cstdio>
  3. #include <iostream>
  4. using namespace std;
  5. int main()
  6. {
  7. int a,b,c;
  8. for(a=1; a<=3; a++)
  9. for(b=1; b<=3; b++)
  10. for(c=1; c<=3; c++)
  11. if((a!=b)&&(a!=c)&&(b!=c))
  12. cout<<a<<" "<<b<<" "<<c<<endl;
  13. return 0;
  14. }

输出:

  1. 1 2 3
  2. 1 3 2
  3. 2 1 3
  4. 2 3 1
  5. 3 1 2
  6. 3 2 1
     八戒,师傅再告诉你一种新的方法-回溯法。
         从问题的某一种可能出发,搜索从这种情况出发所能达到的所有可能, 当这一条路走到“ 尽头 ”而没达到目的地的时候, 再倒回上一个出发点, 从另一个可能出发, 继续搜索. 这种不断“倒回 一步"寻找解的方法, 称作" 回溯法 "。
        为师给出算法框架,悟空,还是你来实现以下。
1.procedure dfs(step:integer)
2.begin
3.   if 到达边界状态 then 输出解;  
4.   else for i:= 1 to n do //顺次尝试每一种可能
5.      begin
6.           if 满足条件 then
7.             begin
8.                 保存结果;
9.                 dfs(step+1);
10.                 恢复:保存结果之前的状态
11.              end;
12.       end;
13.end;
14.Begin
15.     dfs(1);
16.end.
        师傅,让俺老孙试一试吧。
  1. #include <cstring>
  2. #include <cstdio>
  3. #include <iostream>
  4. using namespace std;
  5. int a[20];
  6. bool used[20];
  7. int n;
  8. void print()
  9. {
  10. for(int i=1; i<=n; i++)
  11. cout<<a[i]<<" " ;
  12. cout<<endl;
  13. }
  14. void dfs(int step)
  15. {
  16. if (step ==(n + 1))
  17. print();
  18. else
  19. {
  20. for(int i=1; i<=n; i++)
  21. {
  22. if (used[i] == false)
  23. {
  24. a[step] = i;
  25. used[i] = true;
  26. dfs(step+1);
  27. used[i] = false;
  28. }
  29. }
  30. }
  31. }
  32. int main()
  33. {
  34. memset(used, false, sizeof(used));
  35. while(scanf("%d", &n) == 1)
  36. {
  37. dfs(1);
  38. cout<<endl;
  39. }
  40. return 0;
  41. }
结果:
2
1 2
2 1

3
1 2 3
1 3 2
2 1 3
2 3 1
3 1 2
3 2 1
       

       大家都知道,深度优先的主要思想就是“不撞南墙不回头”,“一条路走到黑”,如果遇到“墙”或者“无路可走”时再去走下一条路。所以先规定好一个走路的规则,比如就按照右下左上顺时针的方式去尝试。

             HDU 1016 Prime Ring Problem是一道简单的搜索题,简单说就是一位一位去试,一位一位地填上数字,看是否满足条件,具体看注释。
        俺沙悟净也试一试
  1. #include <cstring>
  2. #include <cstdio>
  3. using namespace std;
  4. // 素数表,数据较小直接打表也可以
  5. bool prime[41];
  6. void init()
  7. {
  8. memset(prime, true, sizeof(prime));
  9. prime[0] = prime[1] = false;
  10. for(int i=2; i<=40; i++)
  11. {
  12. if(prime[i])
  13. {
  14. for(int j=i*2; j<=40; j+=i)
  15. prime[j] = false;
  16. }
  17. }
  18. }
  19. // 标记
  20. bool vis[21];
  21. int num[21];
  22. int N, t=1;
  23. void dfs(int n)
  24. {
  25. if(n == N && prime[1+num[n]])
  26. {
  27. // 判断最后一位是否满足条件,打印
  28. printf("1");
  29. for(int i=2; i<=N; i++)
  30. printf(" %d", num[i]);
  31. printf("\n");
  32. }
  33. else
  34. {
  35. // 在还未访问的数据中选择满足条件的,标记并访问
  36. for(int i=2; i<=N; i++)
  37. {
  38. if(!vis[i] && prime[i+num[n]])
  39. {
  40. num[n+1] = i;
  41. vis[i] = true;
  42. dfs(n+1);
  43. // 访问结束,重新设为未访问
  44. vis[i] = false;
  45. }
  46. }
  47. }
  48. }
  49. int main()
  50. {
  51. init();
  52. memset(vis, false, sizeof(vis));
  53. vis[1] = true;
  54. num[1] = 1;
  55. while(scanf("%d", &N) == 1)
  56. {
  57. printf("Case %d:\n", t++);
  58. dfs(1);
  59. printf("\n");
  60. }
  61. return 0;
  62. }
    俺白龙马也试一试
  1. #include<iostream>
  2. using namespace std;
  3. int P[38]= {0,0,1,1,0,1,0,1,0,0,0,1,0,1,0,0,0,1,0,1,0,0,0,1,0,0,0,0,0,1,0,1,0,0,0,0,0,1};
  4. // 素数表,数据较小直接打表
  5. int visited[21];
  6. int a[21];
  7. int n;
  8. void DFS(int count)
  9. {
  10. int i;
  11. if(count==n&&P[a[count-1]+1]==1)// // 判断最后一位是否满足条件,打印
  12. {
  13. cout<<a[0];
  14. for( i=1; i<n; i++)
  15. {
  16. cout<<' '<<a[i];
  17. }
  18. cout<<endl;
  19. }
  20. else
  21. {
  22. for(i=2; i<=n; i++)
  23. {
  24. if(!visited[i]&&P[a[count-1]+i]==1)//在还未访问的数据中选择满足条件的,标记并访问
  25. {
  26. a[count]=i;
  27. visited[i]=1;
  28. DFS(count+1);
  29. visited[i]=0; // 访问结束,重新设为未访问
  30. }
  31. }
  32. }
  33. }
  34. int main()
  35. {
  36. a[0]=1;
  37. int A=1;
  38. while(cin>>n)
  39. {
  40. cout<<"Case "<<A++<<':'<<endl;
  41. DFS(1);
  42. cout<<endl;
  43. }
  44. return 0;
  45. }




     





    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多