DP 动态规划

p1269 马棚

题目:

每天,小明和他的马外出,然后他们一边跑一边玩耍。当他们结束的时候,必须带所有的马返回马棚,小明有K个马棚。他把他的马排成一排然后跟随它走向马棚,因为他们非常疲劳,小明不想让他的马做过多的移动。因此他想了一个办法:将马按照顺序放在马棚中,后面的马放的马棚的序号不会大于前面的马放的马棚的序号。而且,他不想他的K个马棚中任何一个空置,也不想任何一匹马在外面。已知共有黑、白两种马,而且它们相处得并不十分融洽。如果有i个白马和j个黑马在一个马棚中,那么这个马棚的不愉快系数将是i*j。所有k个马棚不愉快系数的和就是系数总和。确定一种方法把n匹马放入k个马棚,使得系数总和最小。

输入:

在第一行有两个数字:n(1≤n≤500)和k(1≤k≤n)。在接下来的n行是n个数。在这些行中的第i行代表队列中的第i匹马的颜色:1意味着马是黑色的,0意味着马是白色的。

6 3
1
1
0
1
0
1
{6匹马,3个马棚}
{第1匹马为黑马}
{第3匹马为白马}

输出:只输出一个单一的数字,代表系数总和可能达到的最小值

2    {最小系数总和}

 

 

这道题我是先直接求出从牧场i到j的不愉快值,然后再用DP状态转移f[i][j]=min(f[i][j],f[i-1][k]+b[k+1][j]);然后输出f[k][n]就好了

代码:

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<ctime>
using namespace std;
int a[10000];
int f[1100][1010];
int b[1000][1000];
int main()
{
    int n,k;
    cin>>n>>k;
    memset(f,10,sizeof(f));
    for(int i=1;i<=n;i++)
        cin>>a[i];
    int bl,w;
    for(int i=1;i<=n;i++)
    {
        for(int j=i;j<=n;j++)
        {
            bl=0;w=0;
            for(int k=i;k<=j;k++)
            {
                if(a[k]==1)
                {
                    bl++;
                }
                if(a[k]==0)
                {
                    w++;
                }
            }
            b[i][j]=bl*w;
        }
    }
    for(int i=0;i<=n;i++)
    {
        f[1][i]=b[1][i];
    }
    /*for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=n;j++)
        {
            cout<<f[i][j]<<' ';
        }
        cout<<endl;
    }*/
    for(int i=2;i<=n;i++)
    {
        for(int j=1;j<=n;j++)
        {
            for(int k=0;k<=j;k++)
            {
                f[i][j]=min(f[i][j],f[i-1][k]+b[k+1][j]);
            }
        }
    }
    /*for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=n;j++)
        {
            cout<<f[i][j]<<' ';
        }
        cout<<endl;
    }*/
    cout<<f[k][n]<<endl;
    return 0;
}

 

posted @ 2017-02-27 09:32  列車員lcy  阅读(275)  评论(0编辑  收藏  举报