codeforces 495C. Treasure 解题报告
题目链接:http://codeforces.com/problemset/problem/495/C
题目意思:给出一串只有三种字符( ')','(' 和 '#')组成的字符串,每个位置的这个字符 '#'可以替换成不少于 1 个的 ')',问如何对每个'#'进行替换,使得对于字符串的任意一个位置, ')' 的数量始终不大于'(' 的数量。注意,'#'被替换成')'的总数以及原先有的')'的数量之和 == '(' 的总数。
花了两个晚上的一点时间,今天在图书馆里终于想到解决方案了,大感动 ~~~~ >__<
/********************************************* (错误思路请忽略= =)
第一个晚上做的时候,只知道最好的填法就是,除了最后一个'#'之外,其他都替换为一个 ')',最后一个就根据公式: 数量 '(' - 数量 ')' - (总的'#'数 - 1) 来填入。
但是问题是,如何确定有解。我当时很天真地以为,如果 '(' == ')' 但是'#'还有剩('#' > 0)就无解,test7:#)))) 证明此想法是错的,本来答案是-1,我程序输出 -4,因为这个判断是有缺陷的,排除不了这种情况。后来干脆直接保存每个位置的'(' (cnt[]) 和 ')'(cnt1[])的数量还有 '#'的数量,如果是'(',cnt[i] = cnt[i-1]+1,cnt1[i] = cnt[i-1];如果是')',cnt1[i] = cnt1[i-1]+1,cnt[i] = cnt[i-1];如果是'#',cnt[i] = cnt[i-1],cnt1[i] = cnt1[i-1]。'(', ')', '#'总数量分别对应c1, c2, c0,然后加多一个check()函数来判断如果cnt[i] < cnt1[i] 就 返回false,我只能说:有点不知所谓 = =....因为压根就没有考虑'#' 的值!不知道'#'具体填什么【test12 ##((((((() 】。改改改接着就是test14........噩梦啊~~~~煎熬啊
**********************************************/
早上重新想过,终于有头绪了。我的做法应该叫做所谓的反证法吧~~~~容易知道最保守最可行的方法就是:除了最后一个'#'之外的所有 '#' 都替换为 一个')'。
设两个整型数tot 和c(初始值都为 0, c 用来保存'#'的数量),一个数组vis[],保存每个'#'的位置的替换个数,遍历一遍数组s[i]
if 【'('】 tot++
if 【')'】 tot--
if 【'#'】 c++, vis[i] = 1
然后反向遍历数组s[i],找出最后一个'#'的位置,使得vis[i] = tot - (c-1)。但是前期要做一个判断工作,tot-(c-1) <= 0是无解的!,它专门处理 '(' == ')',c >= 1 的情况(类似(#),或者 #)。
最后当然就是判断无解情况啦。开一个整数 cnt = 0,从前往后遍历数组s[],
if 【'('】 cnt++
if 【')'】 cnt--
if 【'#'】 cnt -= vis[i]
需要检查每个位置的cnt的值,if (cnt < 0) 就返回false。即表示 ) > ( ,不满足题目要求。
最后返回true的情况是 cnt == 0,这样才能保证 '#' 替换为 ')' 以及本来存在的 ')' 的数量之和 等于 ‘(' 的数量
如果经得起 check() 函数的考验,那么就输出解。
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
using namespace std; const int maxn = 1e5 + ;
char s[maxn];
int vis[maxn], len; bool check(char s[])
{
int cnt = ;
for (int i = ; i < len; i++)
{
if (s[i] == '(')
cnt++;
else if (s[i] == ')')
cnt--;
else // #
cnt -= vis[i];
if (cnt < ) // 任意位置都要保证 ')' <= '('
return false;
}
if (cnt == )
return true;
return false;
} int main()
{
#ifndef ONLINE_JUDGE
freopen("in.txt", "r", stdin);
#endif // ONLINE_JUDGE while (scanf("%s", s) != EOF)
{
memset(vis, , sizeof(vis)); len = strlen(s);
int tot = , c = ;
for (int i = ; i < len; i++)
{
if (s[i] == '(')
tot++;
else if (s[i] == ')')
tot--;
else // #
{
c++;
vis[i] = ;
}
}
if (tot-(c-) <= )
printf("-1\n");
else
{
int last = tot - (c-); // 最后一个 # 的赋值,至少要为1
for (int i = len-; i >= ; i--)
{
if (s[i] == '#')
{
vis[i] = last;
break;
}
}
if (!check(s))
printf("-1\n");
else
{
for (int i = ; i <= c-; i++)
printf("1\n");
printf("%d\n", last);
}
}
}
return ;
}
codeforces 495C. Treasure 解题报告的更多相关文章
- codeforces 31C Schedule 解题报告
题目链接:http://codeforces.com/problemset/problem/31/C 题目意思:给出 n 个 lessons 你,每个lesson 有对应的 起始和结束时间.问通过删除 ...
- codeforces 499B.Lecture 解题报告
题目链接:http://codeforces.com/problemset/problem/499/B 题目意思:给出两种语言下 m 个单词表(word1, word2)的一一对应,以及 profes ...
- codeforces 490B.Queue 解题报告
题目链接:http://codeforces.com/problemset/problem/490/B 题目意思:给出每个人 i 站在他前面的人的编号 ai 和后面的人的编号 bi.注意,排在第一个位 ...
- CodeForces 166E -Tetrahedron解题报告
这是本人写的第一次博客,学了半年的基础C语言,初学算法,若有错误还请指正. 题目链接:http://codeforces.com/contest/166/problem/E E. Tetrahedro ...
- codeforces 489A.SwapSort 解题报告
题目链接:http://codeforces.com/problemset/problem/489/A 题目意思:给出一个 n 个无序的序列,问能通过两两交换,需要多少次使得整个序列最终呈现非递减形式 ...
- codeforces 485A.Factory 解题报告
题目链接:http://codeforces.com/problemset/problem/485/A 题目意思:给出 a 和 m,a 表示第一日的details,要求该日结束时要多生产 a mod ...
- codeforces 483A. Counterexample 解题报告
题目链接:http://codeforces.com/problemset/problem/483/A 题目意思:给出一个区间 [l, r],要从中找出a, b, c,需要满足 a, b 互质,b, ...
- codeforces 479C Exams 解题报告
题目链接:http://codeforces.com/problemset/problem/479/C 题目意思:简单来说,就是有个人需要通过 n 门考试,每场考试他可以选择ai, bi 这其中一个时 ...
- codeforces 479B Towers 解题报告
题目链接:http://codeforces.com/problemset/problem/479/B 题目意思:有 n 座塔,第 i 座塔有 ai 个cubes在上面.规定每一次操作是从最多 cub ...
随机推荐
- 深入分析,理解jQuery.Deferred源码
前言: 如果你对jQuery.Callback回调对象不了解,或者只掌握其方法,但是没有通过阅读源码理解,可以先阅读 前一章jQuery.Callbacks源码解读二,因为只有完全理解jQuery.C ...
- Git工作流总结
引用自:https://github.com/xirong/my-git/blob/master/git-workflow-tutorial.md 说明: 个人在学习Git工作流的过程中,从原有的 S ...
- Hadoop 面试题之storm 3个
Hadoop 面试题之八 355.metaq 消息队列 zookeeper 集群 storm集群(包括 zeromq,jzmq,和 storm 本身)就可以完成对商城推荐系统功能吗?还有其他的中间件? ...
- sql存储过程几个简单例子
导读:sql存储是数据库操作过程中比较重要的一个环节,对于一些初学者来说也是比较抽象难理解的,本文我将通过几个实例来解析数据库中的sql存储过程,这样就将抽象的事物形象化,比较容易理解. 例1: cr ...
- 个人建了一个APPCAN移动前端开发交流QQ群258213194
QQ群号:258213194,欢迎有兴趣的同志加一下. 二维码如下:
- httpd服务访问控制
客户机地址限制 通过配置Order.Deny from.Allow from 来限制客户机 allow.deny :先"允许"后"拒绝" ,默认拒绝所有为明确的 ...
- Java VisualVM使用:堆OOM
背景 近期遇到了一个java.lang.OutOfMemoryError: Java heap space的问题,排除了堆设置过小的问题,代码走查没有头绪,所以使用VisualVM工具分析堆内存情况. ...
- BZOJ3223——Tyvj 1729 文艺平衡树
1.题目大意:维护序列,只有区间翻转这个操作 2.分析:splay的经典操作就是实现区间翻转,就是在splay中有一个标记,表示这个区间被翻转了 然后就是记得各种的操作访问某个点时,记得下传,顺便交换 ...
- linux中comm命令用法
linux系统中comm命令用法详解 linux系统下的comm命令是一个非常实用的文件对比命令. comm命令功能: 选择或拒绝两个已排序的文件的公共的行. comm命令语法:comm [-12 ...
- gcc编译时对'xxxx'未定义的引用问题
gcc编译时对’xxxx’未定义的引用问题 gcc编译时对’xxxx’未定义的引用问题 原因 解决办法 gcc 依赖顺序问题 在使用gcc编译的时候有时候会碰到这样的问题,编译为.o(obj) 文件没 ...