逃离农场

题目:

牛牛在农场饲养了n只奶牛,依次编号为0到n-1, 牛牛的好朋友羊羊帮牛牛照看着农场.有一天羊羊看到农场中逃走了k只奶牛,但是他只会告诉牛牛逃走的k只奶牛的编号之和能被n整除。你现在需要帮牛牛计算有多少种不同的逃走的奶牛群。因为结果可能很大,输出结果对1,000,000,007取模。

例如n = 7 k = 4:
7只奶牛依次编号为0到6, 逃走了4只
编号和为7的有:{0, 1, 2, 4}
编号和为14的有:{0, 3, 5, 6}, {1, 2, 5, 6}, {1, 3, 4, 6},{2, 3, 4, 5}
4只牛的编号和不会大于18,所以输出5. 
输入描述:
输入包括一行,两个整数n和k(1 ≤ n ≤ 1000),(1 ≤ k ≤ 50),以空格分割。
输出描述:
输出一个整数表示题设所求的种数。
解析:本题可用动态规划,具体分为dp[i][k][t]为前i只奶牛中选k只,模为t的情况,该情况可由以下两种情况组成:
前i-1头奶牛中选取k只,不取第i头奶牛:dp[i-1][k][t]
前i-1头奶牛选取k-1只奶牛,即第i头奶牛必选,第i头奶牛编号为i-1,则若要选取第i头奶牛后模为t,则现在模必须为t+i-1<0?t+i-1+n:t+i-1: dp[i -1][k-1][(t+i-1)%n]
所以动态规划表达式为  dp[i][j][t] = dp[i-1][j][t] + dp[i-1][k-1][(t+i-1+n)%n]

import java.util.Scanner;


public class Main {
	public static void main(String[] args) {
		int N = 0;
		int K = 0;
		Scanner scanner = new Scanner(System.in);
		while (scanner.hasNextInt()) {
			N = scanner.nextInt();
			K = scanner.nextInt();
			Solution(N, K);
		}
	}
	private static void Solution(int n, int k) {
		long[][] dp = new long[55][1000];//不适用三维数组是因为内存空间超出限制
			dp[0][0] = 1;
			
		for (int i = 1; i <= n; i++) {
			for (int j = k; j >= 1; j--) {//从k到1是因为使用的是二维数组,若从小到大,值会被覆盖,计算错误,
				for (int x = 0; x < n; x++) {
					System.out.print( "dp["+(i-1)+"][" + j + "][" + x + "]=" + dp[j][x]+ ",dp["+(i-1)+"][" + (j - 1) + "]["	+ (x - i < 0 ? x - i + n : x - i) + "]="	+ dp[j - 1][x - i < 0 ? x - i + n : x - i]);
					dp[j][x] = (dp[j][x] + dp[j - 1][x - i < 0 ? x - i + n : x- i]);
					System.out.println("表达式为:dp["+(i)+"][" + j + "][" + x + "]=dp["+(i-1)+"][" + j+ "][" + x + "]+dp["+(i-1)+"][" + (j - 1) + "]["+ (x - i < 0 ? x - i + n : x - i) + "]=" + dp[j][x]);
				}
				System.out.println();
			}
			System.out.println("=====i=" + i + "=====");
		}
		System.out.println(dp[k][0]);
	}


	private static void Solution2(int n, int k) {
		int[][][] dp = new int[1000][55][1005];//不适用三维数组是因为内存空间超出限制
		for(int i=0;i<=n;i++){
			dp[i][0][0] = 1;
		}
		
		for (int i = 1; i <= n; i++) {
			for (int j = k; j >= 1; j--) {//从k到1是因为使用的是二维数组,若从小到大,值会被覆盖,计算错误,
				for (int x = 0; x < n; x++) {
					System.out.print( "dp["+(i-1)+"][" + j + "][" + x + "]=" + dp[i-1][j][x]+ ",dp["+(i-1)+"][" + (j - 1) + "]["	+ (x - i < 0 ? x - i + n : x - i) + "]="	+ dp[i-1][j - 1][x - i < 0 ? x - i + n : x - i]);
					dp[i][j][x] = (dp[i-1][j][x] + dp[i-1][j - 1][x - i < 0 ? x - i + n : x- i]);
					System.out.println("表达式为:dp["+(i)+"][" + j + "][" + x + "]=dp["+(i-1)+"][" + j+ "][" + x + "]+dp["+(i-1)+"][" + (j - 1) + "]["+ (x - i < 0 ? x - i + n : x - i) + "]=" + dp[i][j][x]);
				}
				System.out.println();
			}
			System.out.println("=====i=" + i + "=====");
		}
		System.out.println(dp[n][k][0]);
	}
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值