POJ 3185 The Water Bowls 反转+点灯游戏

2023-09-12 22:51:33

一、题目大意

有20盏灯(其实20盏灯数据量太少了,1e7我觉得差不多都可以过)放在同一行,点亮某一盏灯,它左右两边的灯也会亮,然后边缘特殊考虑,电亮两端的灯,只会照亮它相邻的那一个。

二、解题思路

这其实就是一个反转问题,我们从第二盏灯开始循环到第20盏灯,对于每一次循环的i,我们计算出i-1的灯是否亮着,如果i-1亮着,我们就continue,如果i-1不亮,那么电亮i,来照亮i-1。

然后计算每盏灯是不是亮着,可以记录一个数组f[i],记录是否有 [i-1,i+1]的区间内被点亮,假如说一个位置i,f[i]==1,f[i-1]==1,那么相当于i位置与起始时的i位置情况相同,如果f[i]和f[i-1]中有一个为0,另一个为1的话,那么相当于i位置与起始时的情况相反。

这样我们用尺取法,每次维护i-1和i-2位置的f[i]的和,这样结合i位置的起始情况,就知道i位置是不是亮着的,然后生成f[i]的值,不断循环即可,循环结束之后,判断下最后一盏灯是不是亮着的,如果是的话,这是一个可行解,然后把f[i]数组求和,就是反转总次数。

因为我们是从第二盏灯开始考虑的,所以我们要循环判断下第一盏灯f[1]为1或0的两种情况,分别求解。

这种反转问题(点灯游戏),特点就是反转两次和没反转一样,然后都是牵一发而动全身,这种问题解题的关键就是找到可以唯一决定某个位置的条件,例如当f[i],f[i-1]为定值时,f[i+1]的值可以直接决定i是不是亮着的,这样的话,每次通过i+1保证i,然后不断循环,最后对无法保证的区间([20,20])做判断,通过判断保存可行解,不通过则找下一种情况。

三、代码

#include <iostream>
using namespace std;
int revCnt[27], ansCnt = 0x3f3f3f3f;
bool drinkable[27];
void input()
{
    int x;
    for (int i = 1; i <= 20; i++)
    {
        scanf("%d", &x);
        if (x == 0)
        {
            drinkable[i] = true;
        }
        else
        {
            drinkable[i] = false;
        }
    }
}
void flushRevCnt()
{
    for (int i = 1; i <= 20; i++)
    {
        revCnt[i] = 0;
    }
}
void saveAns()
{
    int currentAns = 0;
    for (int i = 1; i <= 20; i++)
    {
        currentAns += revCnt[i];
    }
    if (currentAns < ansCnt)
    {
        ansCnt = currentAns;
    }
}
void solve()
{
    int currentSum = revCnt[1];
    for (int i = 2; i <= 20; i++)
    {
        bool isDrinkable = drinkable[i - 1];
        if (currentSum % 2 != 0)
        {
            isDrinkable = !drinkable[i - 1];
        }
        if (!isDrinkable)
        {
            revCnt[i] = 1;
        }
        currentSum += revCnt[i];
        // 我们这个位置i,作为下次的判断条件,那么i-1,i,i+1可以作为比较的因素,i-1-1得去掉
        if (i - 1 - 1 >= 1)
        {
            currentSum -= revCnt[i - 1 - 1];
        }
    }
    int lastRevCnt = revCnt[19] + revCnt[20];
    bool isLastDrinkable = drinkable[20];
    if (lastRevCnt % 2 != 0)
    {
        isLastDrinkable = !drinkable[20];
    }
    if (isLastDrinkable)
    {
        saveAns();
    }
}
int main()
{
    input();
    flushRevCnt();
    solve();
    flushRevCnt();
    revCnt[1] = 1;
    solve();
    printf("%d\n", ansCnt);
    return 0;
}

更多推荐

spring ioc

1.什么是SpringSpring框架是一个分层的、面向切面的Java应用程序的一站式轻量级解决方案,它是Spring技术栈的核心和基础,是为了解决企业级应用开发的复杂性而创建的。>简单来说,Spring是一个轻量级的控制反转(IoC)和面向切面(AOP)的容器框架。介于SpringMVC与Mybatis之间的中间层框

【java】【SpringBoot】【四】原理篇 bean、starter、核心原理

目录一、自动配置1、bean加载方式(复习)1.1加载方式-xml方式生命bean1.2加载方式-xml+注解方式声明bean1.3注解方式声明配置类1.4FactoryBean1.5proxyBeanMethod属性1.6使用@Import注解导入1.7使用上下文对象在容器初始化完毕后注入bean1.8导入实现了Im

Django Web开发入门基础

官方有很详细的文档,但是看过几遍之后如果要翻找还是有点麻烦,本文算作是学习笔记,提取一些关键点记录下来,另附上官方教程WritingyourfirstDjangoapp注:文中的指令使用py,是在Windows上,macOS要使用python31.安装DjangoDjango是一个基于Python的Web开发框架,安装

git使用说明

配置hosts配置C:\Windows\System32\drivers\etc\hosts192.168.**.**git.wl.com本地git账号配置(xxx在gitlab个人profile中)打开gitbashgitconfig--globaluser.namexxxxgitconfig--globaluser

使用springcloud-seata解决分布式事务问题-2PC模式

目录一、建立undo_log表二、安装事务协调器:seata-server三、整合可以查看官网:快速启动|Seata一、建立undo_log表--注意此处0.3.0+增加唯一索引ux_undo_logCREATETABLE`undo_log`(`id`bigint(20)NOTNULLAUTO_INCREMENT,`b

华为OD机试 - 滑动窗口最大和 - 滑动窗口(Java 2023 B卷 100分)

目录专栏导读一、题目描述二、输入描述三、输出描述四、解题思路五、Java算法源码六、效果展示1、输入2、输出3、说明华为OD机试2023B卷题库疯狂收录中,刷题点这里专栏导读本专栏收录于《华为OD机试(JAVA)真题(A卷+B卷)》。刷的越多,抽中的概率越大,每一题都有详细的答题思路、详细的代码注释、样例测试,发现新题

【技术分享】NetLogon于域内提权漏洞(CVE-2020-1472)

一、漏洞介绍CVE-2020-1472是一个Windows域控中严重的远程权限提升漏洞。攻击者在通过NetLogon(MS-NRPC)协议与AD域控建立安全通道时,可利用该漏洞将AD域控的计算机账号密码置为空,从而控制域控服务器。该漏洞适用于Win2008及后的所有版本。二、漏洞原理Netlogon使用的AES认证算法

区块链(3):区块链去中心化

1点对点同步区块链的流程流程图如下:流程讲解:(1)连接节点(2)向该节点请求最新区块(3)请求到区块以后,根据返回的最新区块前置hash是否和我本身的区块hash相等,分为两种情况:第一种情况:最新区块前置hash和我本身的区块hash相等并合法有效,则最新区块是新区块,同时添加到我的链中。然后广播到我这个节点连接的

期权交易保证金比例一般是多少?

期权交易是一种非常受欢迎的投资方式之一,它为期权市场带来了更为多样化和灵活化的交易形式。而其中的期权卖方保证金比例是期权交易中的一个重要指标,直接关系到投资者的风险与收益,下文介绍期权交易保证金比例一般是多少?本文来自:期权酱一、期权的交易保证金如何计算?期权交易保证金分为开仓保证金和维持保证金。采用非线性保证金的方式

Spring Authorization Server入门 (十七) Vue项目使用授权码模式对接认证服务

Vue单页面项目使用授权码模式对接流程说明以下流程摘抄自官网在本例中为授权代码流程。授权码流程的步骤如下:客户端通过重定向到授权端点来发起OAuth2请求。如果用户未通过身份验证,授权服务器将重定向到登录页面。身份验证后,用户将再次重定向回授权端点。如果用户未同意所请求的范围并且需要同意,则会显示同意页面。一旦用户同意

Spring中@Component和@Bean的区别

前言Spring是一个流行的Java开发框架,它提供了一种简化应用程序开发的方式。在Spring中,@Component和@Bean是两个常用的注解,用于定义和管理对象的创建和依赖注入。虽然它们都用于创建和管理对象,但有一些关键区别。@Component注解@Component是Spring框架的核心注解之一,它用于标

热文推荐