解题思路:

注意事项:

参考代码:import java.io.*;

import java.util.*;

public class Main{

    static int maxn = 200005,n,m,inf=(int)1e9;

    static long INF = (long)2e18,ans = 0,mod = (int)1e9+7;

    static Scanner sc = new Scanner (System.in);

    static BufferedReader bf = new BufferedReader(new InputStreamReader(System.in));

    static StreamTokenizer st  =new StreamTokenizer(bf);

    static PrintWriter pw = new PrintWriter(new OutputStreamWriter(System.out));

    public static void main(String[]args) throws IOException{

        int T = 1;

        while(T-->0) solve();

        pw.flush();

    }

    static final int I() throws IOException {

        st.nextToken();

        return (int)st.nval;

    }

    static int dp[][][] = new int [303][303][3];

    static int num[][] = new int [303][303];

    static int a[] = new int [301];

    static int sum[] = new int [301]; //前缀和

    static int c[] = new int [301];

    static int cost[][] = new int [303][303];

    static void solve() throws IOException{

        n= I();

        for(int i=1;i<=n;i++)

            for(int j=i;j<=n;j++) {

                num[i][j]=j-i+1;

                for(int k=0;k<3;k++) dp[i][j][k]=inf;

            }

        for(int i=1;i<=n;i++) a[i] = I();

        for(int i=1;i<=n;i++) sum[i]=a[i]+sum[i-1]; //前缀和,方便统计区间

        for(int i=1;i<=n;i++) c[i] = I();//颜色

        for(int i=1;i<=n;i++) dp[i][i][c[i]] = 0; //初始石子堆花费显然为0

        for(int len=1;len<=n;len++)

            for(int i=1;i+len-1<=n;i++) {

                int j=i+len-1;

                for(int col=0;col<3;col++) {

                    int min = inf;

                    for(int k=i;k<j;k++) {

                        if(dp[i][k][col]!=inf && dp[k+1][j][col]!=inf) {

                            min = Math.min(min, dp[i][k][col] + dp[k+1][j][col]);

                        }

                    }

                    if(min==inf) continue;

                    num[i][j]=1;

                    dp[i][j][(col+1)%3] = Math.min(dp[i][j][(col+1)%3], min+sum[j]-sum[i-1]);

                }

            }

        for(int i=1;i<=n;i++)

            for(int j=i;j<=n;j++)

                if(num[i][j] == 1) cost[i][j] = Math.min(dp[i][j][0], Math.min(dp[i][j][1], dp[i][j][2]));

        for(int k=1;k<=n;k++) //类似floyd计算花费

            for(int i=1;i<=k;i++)

                for(int j=k+1;j<=n;j++) {

                    if(num[i][j] > num[i][k] + num[k+1][j]) {

                        num[i][j] = num[i][k] + num[k+1][j];

                        cost[i][j] = cost[i][k]+cost[k+1][j];

                    }

                    else if(num[i][j] == num[i][k] + num[k+1][j]) {

                        cost[i][j] = Math.min(cost[i][j], cost[i][k]+cost[k+1][j]);

                    }

                }

        pw.println(num[1][n]+" "+cost[1][n]);

    }

}

点赞(0)
 

0.0分

0 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论