zoukankan      html  css  js  c++  java
  • 算法题--扔棋子

    题目如下:“有一个100层高的大厦,你手中有两个相同的玻璃围棋子。从这个大厦的某一层扔下围棋子就会碎,用你手中的这两个玻璃围棋子,找出一个最优的策略,来得知那个临界层面。“
    先说下扩展:n层k个球

    这道题有一个dp解,因存在递归。假设第一次扔在第r层,碎了就在1~r之间寻找,此时还剩k-1个球;没碎就在r+1~n之间寻找,还剩k个球。

    代码如下:

     1 #include <iostream>
     2 #include <fstream>
     3 #include <sstream>
     4 #include <string>
     5 #include <cmath>
     6 #include <iomanip>
     7 #include <vector>
     8 #include <deque>
     9 #include <list>
    10 #include <queue>
    11 #include <stack>
    12 #include <map>
    13 #include <algorithm>
    14 #include <limits>
    15 #include <utility>
    16 #include <ctime>
    17 #include <bitset>
    18 using namespace std;
    19 
    20 #define MAX_FLOOR 512
    21 #define MAX_BALL  100
    22 
    23 int dp(int n, int k)
    24 {
    25      if(k<1 || n<1) return -1;    //错误输入
    26 
    27      if(k==1) return n-1;        //去掉一些trivial case
    28      if(n==1) return 0;
    29 
    30      int M[MAX_BALL][MAX_FLOOR];
    31      int i,j,r;
    32      int temp, min;
    33 
    34      for(i=0;i<=k;i++) M[i][0]=M[i][1]=0;    //F(1,k)=F(0,k)=0
    35      for(j=2;j<=n;j++) M[1][j]=j-1;            //F(n,1)=n-1
    36 
    37      /*
    38      状态转移方程:
    39      F(n,k)=min{max{F(r,k-1)+1, F(n-r,k)+1}, 1<=r<=n}
    40      */
    41      for(i=2;i<=k;i++)
    42          for(j=2;j<=n;j++)
    43          {
    44              min = numeric_limits<int>::max();
    45              for(r=1;r<=j;r++)
    46              {
    47                  temp = max(M[i-1][r], M[i][j-r])+1;
    48                  if(temp<min)
    49                      min = temp;
    50              }
    51              M[i][j] = min;
    52          }
    53 
    54      return M[k][n];//F(n,k)
    55  }
    56 
    57  int main()
    58  {
    59      int n,k;
    60 
    61      cin>>n>>k;
    62      cout<<dp(n, k)<<endl;
    63 
    64      return 0;
    65  }

    这道题对于第一问还有一个解:第一个球先找出临界层所在的段,第二个球从段中找层。由于找段的次数不可避免的会增加,那么就减少每次找层的次数。(其实我没太理解)

    x+(x-1)+(x-2)....

  • 相关阅读:
    Andorid开发中如何去除标题栏title
    Andorid自动读取短信验证码
    1020. Tree Traversals (25)
    1019. General Palindromic Number (20)
    1003. Emergency (25)
    1014. Waiting in Line (30)
    ubuntu14.04上java jdk & mvn安装
    LVM基本概念及工作原理
    利用Screen重启DevStack服务
    ubuntu14.04 桌面版/服务器版安装DevStack教程
  • 原文地址:https://www.cnblogs.com/cane/p/3898946.html
Copyright © 2011-2022 走看看