CF1465-D. Grime Zoo

题意:

一个长度为n,由\(0,1,?\)这三个字符构成的字符串,字符串中\(01\)子串贡献\(x\)值,\(10\)的子串贡献\(y\)值,现在让你把\(?\)替换成\(0\)或\(1\),问你整个字符串的总贡献最少可以是多少?

子串是指可以通过删去字符串中的一些字符得到的字符串。不同的\(01\)或\(10\)子串只需要让他们其中的一个\(0\)或\(1\)来自字符串中不同的位置即可。

思路:

先说一个结论:当\(x<y\)的时候,无论两个\(?\)(后面分别称这两个\(?\)为\(c_l\)和\(c_r\))之间的的\(0\)和\(1\)的数量是多少、他们是如何排列的,\(c_l\)取\(0\),\(c_r\)取\(1\)的时候,这个字符串的总贡献一定小于\(c_l\)取\(1\),\(c_r\)取\(0\)的时候的总贡献,即\(w_{c_l=0,c_r=1}<w_{c_l=1,c_r=0}\)。反之\(x>y\)的时候也能得到类似的结论。

证明如下:

假设\(c_l\)和\(c_r\)之间有\(n_0\)个\(0\),\(n_1\)个\(1\),那么当\(c_l=0,c_r=1\)的时候,这段字符串的总贡献\(w_{c_l=0,c_r=1}=(1+n_1)*x+n_0*x+w=(1+n_0+n_1)*x+w\),其中\(w\)是\(c_l,c_r\)之间的字符产生的贡献;当\(c_l=1,c_r=0\)的时候,这段字符串的总贡献\(w_{c_l=1,c_r=0}=(1+n_0)*y+n_1*y+w=(1+n_0+n_1)*y+w\),那么\(w_{c_l=0,c_r=1}-w_{c_l=1,c+r=0}=(1+n_0+n_1)*(x-y)=(r-l)*(x-y)\)。

从上面的式子可以得出结论:\(x<y\)时,\(w_{c_l=0,c_r=1}<w_{c_l=1,c_r=0}\);\(x>y\)时,\(w_{c_l=0,c_r=1}>w_{c_l=1,c+r=0}\)。

这样就可以枚举字符串中的每个位置,将当前枚举的位置之前的所有\(?\)替换为\(1(或0)\),之后所有的\(?\)替换为\(0(或1)\),计算整个字符串的贡献,取最小的一个即可。

但是这样枚举,暴力计算总贡献,时间复杂度为\(O(n^3)\),所以可以通过前缀和进行优化:用前缀和分别维护前\(i\)个字符和后\((n-i+1)\)个字符中\(0\)和\(1\)的数量以及分别维护前\(i\)个字符和后\((n-i+1)\)个字符构成的字符串的总贡献,这样就可以将复杂度从\(O(n^3)\)降为\(O(n)\)。具体的计算方法及实现细节见代码及代码注释。

AC代码:

这段代码使用了一些小技巧,比如将\(x<y\)和\(x>y\)这两种情况归结为一种情况、前缀和的统计方式。具体的在代码以及代码注释中有所体现。

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm> typedef long long ll; const int Maxn = 100005;
const ll INF = 0x3f3f3f3f3f3f3f3f; char s[Maxn];
int char_pre[Maxn], char_suf[Maxn]; // 前i个字符中1的数量,后i个字符中0的数量
ll cnt_pre[Maxn], cnt_suf[Maxn]; // 前i个字符构成的愤怒值,后i个字符构成的愤怒值 void solve() {
ll x, y;
scanf("%s %lld %lld", s + 1, &x, &y);
int n = strlen(s + 1);
if (x > y) { // 让01 > 10 的情况转化为 01 < 10的情况,这样把情况做了统一
std::swap(x, y);
for (int i = 1; i <= n; i++) {
if (s[i] == '1') {
s[i] = '0';
} else if (s[i] == '0') {
s[i] = '1';
}
}
}
for (int i = 1; i <= n; i++) {
if (s[i] == '1') {
// 这里之所以记录1的数量而不是0,是因为在x<y的时候'0'和'?'是被归到一类里面,这样容易统计
char_pre[i] = char_pre[i - 1] + 1;
cnt_pre[i] = cnt_pre[i - 1] + (i - char_pre[i]) * x;
} else { // 这里把'?'看作为0
char_pre[i] = char_pre[i - 1];
cnt_pre[i] = cnt_pre[i - 1] + char_pre[i - 1] * y;
}
}
for (int i = n; i > 0; i--) {
if (s[i] == '0') {
char_suf[i] = char_suf[i + 1] + 1;
cnt_suf[i] = cnt_suf[i + 1] + (n - i + 1 - char_suf[i]) * x;
} else { // 这里把'?'看作为1
char_suf[i] = char_suf[i + 1];
cnt_suf[i] = cnt_suf[i + 1] + char_suf[i] * y;
}
}
ll ans = INF;
for (int i = 0; i <= n; i++) {
// 枚举每个位置,如果这里的字符是'?'那么就会当作'0'处理(见34行)
// 注意这里的i最开始是从0开始的,而不是从1,是为了处理第一个字符为'?'的情况
ans = std::min(ans, cnt_pre[i] + cnt_suf[i + 1] +
1LL * char_pre[i] * char_suf[i + 1] * y +
1LL * (i - char_pre[i]) * (n - i - char_suf[i + 1]) * x);
/*
i以及i之前的字符构成的字符串(所有的?看作0)的总贡献 +
i之后的字符构成的字符串(所有的?看作1)的总贡献 +
i以及i之前的字符中1的数量(所有?看作0) * i之后的字符中0的数量(所有?看作1) +
i以及i之前的字符中0的数量(所有?看作0) * i之后的字符中1的数量(所有?看作1)
*/
}
printf("%lld\n", ans);
} int main() {
solve(); return 0;
}

CF1465-D. Grime Zoo的更多相关文章

  1. zookeeper的zoo.cfg的配置

    zookeeper的默认配置文件为zookeeper/conf/zoo_sample.cfg,需要将其修改为zoo.cfg.其中各配置项的含义,解释如下: tickTime:CS通信心跳时间 Zook ...

  2. (转)The Neural Network Zoo

    转自:http://www.asimovinstitute.org/neural-network-zoo/ THE NEURAL NETWORK ZOO POSTED ON SEPTEMBER 14, ...

  3. Mininet建立topology zoo中的拓扑

    以前用Mininet建立拓扑都是在别人的代码上进行需求上的修改,这次从头开始将topology zoo(http://www.topology-zoo.org/)中的拓扑用Mininet建立,不失一般 ...

  4. zoo.cfg配置

    zookeeper的默认配置文件为zookeeper/conf/zoo_sample.cfg,需要将其修改为zoo.cfg.其中各配置项的含义,解释如下: 1.tickTime:CS通信心跳时间 Zo ...

  5. Codeforces Round #250 (Div. 1) B. The Child and Zoo 并查集

    B. The Child and Zoo Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/438/ ...

  6. cf437D The Child and Zoo

    D. The Child and Zoo time limit per test 2 seconds memory limit per test 256 megabytes input standar ...

  7. CF437D(The Child and Zoo)最小生成树

    题目: D. The Child and Zoo time limit per test 2 seconds memory limit per test 256 megabytes input sta ...

  8. Codeforces 437D The Child and Zoo(贪心+并查集)

    题目链接:Codeforces 437D The Child and Zoo 题目大意:小孩子去參观动物园,动物园分非常多个区,每一个区有若干种动物,拥有的动物种数作为该区的权值.然后有m条路,每条路 ...

  9. Codeforces 437 D. The Child and Zoo 并查集

    题目链接:D. The Child and Zoo 题意: 题意比较难懂,是指给出n个点并给出这些点的权值,再给出m条边.每条边的权值为该条路连接的两个区中权值较小的一个.如果两个区没有直接连接,那么 ...

随机推荐

  1. CentOS6.8安装及各种坑

    实现目的:用U盘安装CentOS 6.2 32位系统 所需工具: 一.UltraISO(用来制作U盘启动) 下载地址:http://www.newhua.com/soft/614.htm 二.Cent ...

  2. Django-发上云服务器遇到的问题

    1.服务器启动后外网访问显示A server error occurred. Please contact the administrator. 解决方法:原文:https://www.cnblogs ...

  3. windows上传ipa到苹果开发者中(app store)的方法

    假如你已经使用过苹果开发者中心上架app,你肯定知道在苹果开发者中心的web界面,无法直接提交ipa文件,而是需要使用第三方工具,将ipa文件上传到构建版本,开发者中心才能在构建版本里选择构建版本上架 ...

  4. 用动图讲解分布式 Raft

    一.Raft 概述 Raft 算法是分布式系统开发首选的共识算法.比如现在流行 Etcd.Consul. 如果掌握了这个算法,就可以较容易地处理绝大部分场景的容错和一致性需求.比如分布式配置系统.分布 ...

  5. JAVA SSM整合流程以及注意点

    1.搭建整合环境 整合说明:SSM整合可以使用多种方式,咱们会选择XML + 注解的方式 先搭建整合的环境 先把Spring的配置搭建完成 再使用Spring整合SpringMVC框架 最后使用Spr ...

  6. 研发过程及工具支撑 DevOps 工具链集成

    https://mp.weixin.qq.com/s/NYm63nkCymIV3DbL4O01dg 腾讯重新定义敏捷 |Q推荐 小智 InfoQ 2020-09-03 敏捷开发奠基人 Robert C ...

  7. @functools.lru_cache()

    django.views.debug.get_default_exception_reporter_filter @functools.lru_cache()def get_default_excep ...

  8. DPDK CAS(compare and set)操作

    前言 rte_ring是一个无锁队列,无锁队列的出队入队操作是rte_ring实现的关键.因此,本文主要讲解dpdk是怎样使用无锁机制实现rte_ring的多生产者入队操作. rte_atomic32 ...

  9. CF1416D 做题心得

    CF1416D 做题心得 上次在某trick中提到了这个题,一开始觉得太毒瘤没有写,现在把它补上了. 感觉实现这个东西,比单纯收获一个trick,收获的东西多太多了. 主要思路 它的主要trick是& ...

  10. MIT 6.S081 Lab File System

    前言 打开自己的blog一看,居然三个月没更新了...回想一下前几个月,开题 + 实验室杂活貌似也没占非常多的时间,还是自己太懈怠了吧,掉线城和文明6真的是时间刹手( 不过好消息是把15445的所有l ...