彩色的石子 题目和解法 (附源代码)

 

/*题目描述:-----------------------------------------------------------------------------------------------*/

 

把含有k种不同颜色的m个石子放成一条线上。现在要问你怎么才能取走 

最少的石子,使得没有两个相同颜色的石子之间含有其它的颜色 

 

输入: 

有多组测试数据,每组测试数据有两行,第一行是m和k,1 <=m <=1000, 

1 <=k <=5,第二行就是那m个石子的颜色,用1到k表示,最后输入0 0表示 

程序结束。 

 

输出: 

对于每组测试数据,请你找出最少取走的石子数目并输出 

 

样例输入: 

10 3 

2 1 2 2 1 1 3 1 3 3 

0 0 

 

样例输出: 

 

 

 

/*算法描述与分析 ----------------------------------------------------------------------------------------*/

思路:动态规划法。

假设前n个石子的取走方式已知,思考对于第n+1个石子的取法。

首先,假设第n个石子与第n+1个相同,则n+1个石子的取法与n个石子的取法相同。

其次,若第n个石子与第n+1个不同,则对于n+1个石子有两种取的方法。

其一、取走第n+1个石子,则取的数目=n个石子的取的数据 + 1;

其二、前n个石子的取法中,查找这样的取法:“取完后没有与第n+1石子颜色相同的石子”。去所有这些取法中,取走数目最小的一个。

 

接下来,说明动态规划中所使用的状态。数组dp[1000][32][5]。dp[n][i][j]中,n表示加入第n个石子后。

i的二进制数有5位,第x位表示第x种,当前取法中,取完后,是否包含第x种颜色。1表示包含,0表示不包含。

j表示取法取完后,最后一种颜色是j。例如:对于样例中的数据,

输入2后,dp[0][2][1]=0;

输入1后,dp[1][2][1]=1; dp[1][1][0]=1; dp[1][3][0]=0;

输入2后,dp[2][2][1]=1; dp[2][1][0]=2; dp[2][3][0]=1; dp[2][3][1]=1;

dp[n][i][j]=min(dp[n-1][i-(1<<j)][k]), min(dp[n-1][i][k]+1));(0<=k<=4)

实际算法中,可以省略n,用dp[i][j]。

 

逐一加入石子,计算dp[i][j]。加入所有石子后,打印min(dp[i][j]);

 

算法时间复杂度O(m*(2^k)*k)

 

 

/*源代码 及测试例子------------------------------------------------------------------------------------*/

 

static const int poo=~(1<<31);

static int bound=5;

 

#define min(x,y) ((x>y) ? x=y : x)

 

class ColorStones{

public:

void doit();

};

 

 

void ColorStones::doit(){

int stonenums, colors, i,j,k;

int tin, ans;

int dp[1<<bound][bound];

 

while(cin>>stonenums>>colors){

if(!colors) break;

 

memset(dp, poo, sizeof(dp));

 

for(i = 0; i < stonenums; i++){

cin>>tin;tin--;

for(j=0;j<bound;j++){

if(j==tin) continue;

for(k=(1<<bound)-1;k>0;k--){

if(!(k&(1<<tin))) {

min(dp[k|(1<<tin)][tin],dp[k][j]);

}

if(dp[k][j]!=poo) dp[k][j]++;

}

}

if(dp[1<<tin][tin]==poo) dp[1<<tin][tin]=i;

}//end for

 

 

ans=poo;

for(j=(1<<bound)-1;j>0;j--)

for(k=0;k<bound;k++){

min(ans,dp[j][k]);

}

cout<<ans<<endl;

}//end while

return;

}

 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
1. 题目描述:有一组气球,每个气球都有一个起始点和一个结束点,同一时间内只能有一个气球飞行。现在需要尽可能多地放置气球,求最多可以放置多少个气球。 解法:对于每个气球,按照结束点从小到大排序,依次遍历每个气球,如果当前气球的起始点大于等于上一个气球的结束点,则可以放置,否则不能放置。采用贪心思想,每次选择结束点最靠前的气球,可以得到最多可以放置的气球数量。 Java代码: ```java public int maxBalloons(int[][] balloons) { if (balloons == null || balloons.length == 0) { return 0; } Arrays.sort(balloons, (a, b) -> a[1] - b[1]); int count = 1; int end = balloons[0][1]; for (int i = 1; i < balloons.length; i++) { if (balloons[i][0] >= end) { count++; end = balloons[i][1]; } } return count; } ``` 2. 题目描述:有一组任务,每个任务有一个开始时间和结束时间以及需要花费的时间,同一时间只能完成一个任务。现在需要尽可能多地完成任务,求最多可以完成多少个任务。 解法:对于每个任务,按照结束时间从小到大排序,依次遍历每个任务,如果当前任务的开始时间大于等于上一个任务的结束时间,则可以完成,否则不能完成。采用贪心思想,每次选择结束时间最早的任务,可以得到最多可以完成的任务数量。 Java代码: ```java public int maxTasks(int[][] tasks) { if (tasks == null || tasks.length == 0) { return 0; } Arrays.sort(tasks, (a, b) -> a[1] - b[1]); int count = 1; int end = tasks[0][1]; for (int i = 1; i < tasks.length; i++) { if (tasks[i][0] >= end) { count++; end = tasks[i][1]; } } return count; } ``` 3. 题目描述:有一组会议,每个会议有一个开始时间和结束时间,同一时间只能参加一个会议。现在需要在有限的时间内参加尽可能多的会议,求最多可以参加多少个会议。 解法:对于每个会议,按照结束时间从小到大排序,依次遍历每个会议,如果当前会议的开始时间大于等于上一个会议的结束时间,则可以参加,否则不能参加。采用贪心思想,每次选择结束时间最早的会议,可以得到最多可以参加的会议数量。 Java代码: ```java public int maxMeetings(int[] start, int[] end, int limit) { if (start == null || start.length == 0 || end == null || end.length == 0 || start.length != end.length) { return 0; } int[][] meetings = new int[start.length][2]; for (int i = 0; i < start.length; i++) { meetings[i][0] = start[i]; meetings[i][1] = end[i]; } Arrays.sort(meetings, (a, b) -> a[1] - b[1]); int count = 0; int time = 0; for (int i = 0; i < meetings.length; i++) { if (meetings[i][0] >= time && time + (meetings[i][1] - meetings[i][0]) <= limit) { count++; time += meetings[i][1] - meetings[i][0]; } } return count; } ``` 以上是三个贪心算法题目和Java代码实现。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值