zoukankan      html  css  js  c++  java
  • [Luogu] CF1443B Saving the City

    (Link)

    Description

    你有一段(01)串,你可以选择花费(B)的代价将一个(0)变为(1),也可以花费(A)的代价将一段连续的(1)变为(0),问你最少需要多少代价,才能把整个串都变为(0)

    Solution

    这道题其实不太像(DP)

    会发现无论对当前点做什么操作,都不会会后面点的选择有影响,即无后效性。

    所以对于一段连续的(1),我们在最后一个点考虑把它全部消掉的代价。可以用(A)直接消掉,也可以把它和下一段连续的(1)一起消掉,这时花费为把它们中间的(0)都推平的代价(即(len imes{B}))。

    这样考虑一定是对的。因为对于下一段连续的(1),用(A)的代价直接消掉,如果之前把它和上一段的(0)都推平了,就等于把它们一起消掉,否则就只是花费(A)把这一段消掉。

    要注意判断这一段(1)后面没有(1)的情况。

    Code

    #include <bits/stdc++.h>
    
    using namespace std;
    
    int t, a, b;
    
    char ch[100005];
    
    int read()
    {
    	int x = 0, fl = 1; char ch = getchar();
    	while (ch < '0' || ch > '9') { if (ch == '-') fl = -1; ch = getchar();}
    	while (ch >= '0' && ch <= '9') {x = (x << 1) + (x << 3) + ch - '0'; ch = getchar();}
    	return x * fl;
    }
    
    int main()
    {
    	t = read();
    	while (t -- )
    	{
    		a = read(); b = read();
    		scanf("%s", ch + 1);
    		int l = strlen(ch + 1), res = 0;
    		for (int i = 1; i <= l; i ++ )
    		{
    			if ((ch[i] == '1' && ch[i + 1] == '0') || (i == l && ch[i] == '1'))
    			{
    				int pos = l + 1;
    				for (int j = i + 1; j <= l; j ++ )
    				{
    					if (ch[j] == '1')
    					{
    						pos = j;
    						break;
    					}
    				}
    				if (pos == l + 1) res += a;
    				else res += min(a, (pos - i - 1) * b);
    				i = pos - 1;
    			}
    		}
    		printf("%d
    ", res);
    	}
    	return 0;
    }
    
  • 相关阅读:
    多线程与线程池
    hdu1506 Largest Rectangle in a Histogram
    安装mathtype出问题卸载后 office2016打开mathtype弹错误窗口
    最小总代价 状压DP
    Sumsets 递推
    不容易系列之(4)——考新郎 递推
    超级楼梯 递推
    阿牛的EOF牛肉串(递推)
    子串查询(二维前缀数组) 2018"百度之星"程序设计大赛
    cf#513 B. Maximum Sum of Digits
  • 原文地址:https://www.cnblogs.com/andysj/p/14015693.html
Copyright © 2011-2022 走看看