/**
题目:C. Divide by Six
链接:https://oj.ejq.me/problem/24
题意:给定一个数,这个数位数达到1e5,可能存在前导0.问为了使这个数是6的倍数,且没有前导0,删除尽量少的位数,可以任意位置删除。
输出剩余的位数。如果找不到,输出-1s。
思路:
一开始心想,这么多个位置,又不知道删除几个,排列组合的情况下,简直不可能完成。
这题是我的队友tzq想出来的。在此重新思考一下。
常规想法不可能完成。一定存在某些特殊的地方。观察题目要求,6的倍数。一个数是6的倍数有什么特殊性?
尾数一定是偶数。那么枚举尾数为偶数,仍然很大。
假设确定了一定是2的倍数,那么只要判断是否是3的倍数就行了。而判断一个数是否是3的倍数,只要各个数位的和是3的倍数这个数就一定是3的倍数。很容易证明。 如果找到一个尾数为偶数,且从它为尾的数%3==0.那么这个可能是答案。
%3==1.那么要删除一个某个位的数%3==1的位。或者删除两个数位上的数%3==2的位。为了避免删除后出现前导0,为了答案最优,从右边开始找。如果找不到,所以这种情况无解。
如果删除后出现前导0,去掉前导0,不会有影响。注意把数删完了的情况。此时无解。
%3==2.那么删除一个某个位的数%3==2的位,或者删除两个数位上的数%3==1的位,同上。 问题是:尾数为偶数有很多种情况。我们知道为了让剩余的位数越来越多。所以选择尾数为偶数应该尽量右才好。
如果某两个位作为尾数。以他们为尾数的数%3==0,则显然取尾数更右的更优。
同理%3==1, %3==2,都是选更右的更优,这样删除的选择更多,且剩余的数位也更多。 至此,本体解法为分析+模拟。
细节:只要有前导0,肯定有答案。即剩余的数位0.那么输出1;
*/
#include<bits/stdc++.h>
#define LL long long
using namespace std;
typedef long long ll;
const int maxn = 1e5+;
char s[maxn];
int r[maxn];
int rr[maxn];
char temp[maxn];
int r0, r1, r2;///%3==0, %3==1, %3==2的尾数位置。
int leadzero, len, ans;
int getR(int type)
{
for(int i = len-; i>= ; i--){
if(r[i]==type&&(s[i]-'')%==){
return i;
}
}
return -;
}
void f(int r,int type)
{
if(r==-) return ;
///type==1
if(type==)
{
/// 删除一个1
for(int i = r; i >= &&r>; i--){
if(rr[i]==){
if(i==){
int t = i+;
while(t<=r&&temp[t]==''){
t++;
}
if(t>r){
ans = max(ans,);
}else
{
ans = max(ans,r-t+);
}
}else
{
ans = max(ans,r);
}
}
}
/// 删除两个2
int cnt = , pos = -;
char tt;
for(int i = r; i >= &&r>; i--){
if(rr[i]==){
cnt--;
if(cnt==){
if(i==){
int t = i+;
while(t<=r&&temp[t]==''){
t++;
}
if(t>r){
ans = max(ans,);
}else
{
ans = max(ans,r-t+);
}
}else
{
ans = max(ans,r);
}
}else
{
pos = i;
tt = temp[i];
temp[i]='';
}
}
}
if(pos!=-) temp[pos] = tt; }else
///type==2
{
///删除一个2
for(int i = r; i >= &&r>; i--){
if(rr[i]==){
if(i==){
int t = i+;
while(t<=r&&temp[t]==''){
t++;
}
if(t>r){
ans = max(ans,);
}else
{
ans = max(ans,r-t+);
}
}else
{
ans = max(ans,r);
}
}
}
///删除两个1
int cnt = , pos = -;
char tt;
for(int i = r; i >= &&r>; i--){
if(rr[i]==){
cnt--;
if(cnt==){
if(i==){
int t = i+;
while(t<=r&&temp[t]==''){
t++;
}
if(t>r){
ans = max(ans,);
}else
{
ans = max(ans,r-t+);
}
}else
{
ans = max(ans,r);
}
}else
{
pos = i;
tt = temp[i];
temp[i]='';
}
}
}
if(pos!=-) temp[pos] = tt;
}
}
int main()
{
while(scanf("%s",s)==)
{
r0 = r1 = r2 = -;
leadzero = ;
while(s[leadzero]!='\0'&&s[leadzero]==''){
leadzero++;
}
if(s[leadzero]=='\0'){
printf("1\n"); continue;
}
if(leadzero>) ans = ;///只要有前导0,那么肯定可以一个0作为结果。
strcpy(temp,s+leadzero);
// printf("temp = %s\n", temp);
strcpy(s,temp);
len = strlen(s);
int p = ;
for(int i = ; s[i]!='\0'; i++){
p = p+(s[i]-'');
p %= ;
r[i] = p;
rr[i] = (s[i]-'')%;
}
ans = -;
r0 = getR();
if(r0!=-) ans = r0+;
r1 = getR();
r2 = getR();
f(r1,);
//strcpy(temp,s);
f(r2,);
if(ans!=-){
printf("%d\n",ans);
}else printf("-1s\n");
}
return ;
}

2017 Wuhan University Programming Contest (Online Round) C. Divide by Six 分析+模拟的更多相关文章

  1. 2017 Wuhan University Programming Contest (Online Round) Lost in WHU 矩阵快速幂 一个无向图,求从1出发到达n最多经过T条边的方法数,边可以重复经过,到达n之后不可以再离开。

    /** 题目:Lost in WHU 链接:https://oj.ejq.me/problem/26 题意:一个无向图,求从1出发到达n最多经过T条边的方法数,边可以重复经过,到达n之后不可以再离开. ...

  2. 2017 Wuhan University Programming Contest (Online Round) B Color 树形dp求染色方法数

    /** 题目:Color 链接:https://oj.ejq.me/problem/23 题意:给定一颗树,将树上的点最多染成m种颜色,有些节点不可以染成某些颜色.相邻节点颜色不同.求染色方法数. 思 ...

  3. 2017 Wuhan University Programming Contest (Online Round) D. Events,线段树区间更新+最值查询!

    D. Events 线段树区间更新查询区间历史最小值,看似很简单的题意写了两天才写出来. 题意:n个数,Q次操作,每次操作对一个区间[l,r]的数同时加上C,然后输出这段区间的历史最小值. 思路:在线 ...

  4. zoj 4020 The 18th Zhejiang University Programming Contest Sponsored by TuSimple - G Traffic Light(广搜)

    题目链接:The 18th Zhejiang University Programming Contest Sponsored by TuSimple - G Traffic Light 题解: 题意 ...

  5. The 16th Zhejiang University Programming Contest-

    Handshakes Time Limit: 2 Seconds      Memory Limit: 65536 KB Last week, n students participated in t ...

  6. 2017 Bangladesh National High School Programming Contest ( National Round, Senior Group ), NHSPC 2017 题解

    [题目链接] A. Charm Is Not Always Enough 模拟一下就可以了. #include <bits/stdc++.h> using namespace std; i ...

  7. The 18th Zhejiang University Programming Contest Sponsored by TuSimple

    Pretty Matrix Time Limit: 1 Second      Memory Limit: 65536 KB DreamGrid's birthday is coming. As hi ...

  8. 写完代码就去吃饺子|The 10th Henan Polytechnic University Programming Contest

    河南理工大学第十届校赛 很久没有组队打比赛了,好吧应该说很久没有写题了, 三个人一起玩果然比一个人玩有趣多了... 前100分钟过了4题,中途挂机100分钟也不知道什么原因,可能是因为到饭点太饿了?, ...

  9. The 15th Zhejiang University Programming Contest

    a  ZOJ 3860 求和大家不一样的那个数,签到,map水之 #include<cstdio> #include<map> using namespace std; map ...

随机推荐

  1. zk介绍

    1. 配置管理 Zookeeper提供了这样的一种服务:一种集中管理配置的方法,我们在这个集中的地方修改了配置,所有对这个配置感兴趣的都可以获得变更.这样就省去手动拷贝配置了,还保证了可靠和一致性. ...

  2. 关于ArrayList的一些源码分析

    集合是Java中非常重要而且基础的内容,因为任何数据必不可少的就是该数据是如何存储的,集合的作用就是以一定的方式组织.存储数据.这里写的集合,一部分是比较常见的.一部分是不常用但是我个人平时见到过的, ...

  3. Linux中C语言的编程

    编译的过程 编译的概念:编译程序读取源程序(字符流),对之进行词法与语法的分析,将高级语言指令转换成功能等效的汇编代码,再由汇编程序转换成机器语言,并且按照操作系统对可执行文件格式的要求链接成可执行程 ...

  4. 10道典型的JavaScript面试题

    问题1: 作用域(Scope) 考虑以下代码: (function() { ; })(); console.log(b); 上述代码会打印出5.这个问题的陷阱就是,在立即执行函数表达式(IIFE)中, ...

  5. 【AS3 Coder】任务八:没剧情还玩毛RPG

    使用框架:AS3任务描述:了解RPG游戏中剧情播放器的制作原理及流程难度系数:3(了解原理,能根据XML文件播放剧情) / 5(会制作剧情编辑器) 本章源码下载:http://www.iamseven ...

  6. scrapy-splash抓取动态数据例子十一

    一.介绍 本例子用scrapy-splash抓取活动树网站给定关键字抓取活动信息. 给定关键字:数字:融合:电视 抓取信息内如下: 1.资讯标题 2.资讯链接 3.资讯时间 4.资讯来源 二.网站信息 ...

  7. 怎样编写高效android代码

    基于Android相关设备作为嵌入式设备范畴,在书写App应用的时候要格外关注效率.而且受电池电量的限制.这就导致嵌入式设备有诸多考虑.有限处理能力.因此就要求我们尽量去写高效的代码. 本文讨论了非常 ...

  8. service 和 Controller 差别

    service  层能够看做是还有一个 DAO 层,仅仅是在里面封装了还有一些逻辑. 而 Controller 和 service 差别就大了.Controller 要处理请求映射, service ...

  9. Android 之UI自适应解决方式

    1.概况 作为Android开发者,最头疼的莫过于让自己开发的程序在不同终端上面的显示效果看起来尽量一致(当然.假设要充分利用大屏幕的优势另当别论).在全球范围内来讲.android有着数以亿计的设备 ...

  10. css样式布局中position的那些事儿

    哎,页面布局及设计开发.对于一个一直从事后台开发来说屌丝来说,确实是件非常费时.费力,非常艰难的一件事. 今晚是想实现把多张重叠在一起.或是标记一张图片中不同的位置然后赋以超链接.花了一晚上的时间,才 ...