【算法笔记】状态压缩dp(noip)

 在acwing学习算法的一点思考和总结


 状态压缩dp可以用来解决两种问题:一种是棋盘式的,也就是表示一行有2^N种摆法,另一种是表示一类集合

状压——棋盘式

 思路:可以类比一下蒙德里安的梦想的解题过程,每一行的状态都只会受到上一层状态的影响。那么我们在更新第i行的状态时,我们枚举一下第i - 1行的状态。也就是当这两行的对应状态是个合法状态的话,我们就进行方案数的累加。

确定状态转移方程:f[i][a] += f[i-1][b],表示前i行,并且第i行是第j种摆法 的最大种植方案数

预处理:为了判断哪两种行状态是对应合法的,我们需要进行预处理,找出相邻两行能进行转移的状态(二进制表达)。具体题目具体分析,在这个题目中,第一要满足左右相邻不能都是1,第二要满足上下相邻不能都是1

/*
前i行,并且第i行是第j种摆法 的最大种植方法
*/
#include<iostream>
#include<cstring>
#include<algorithm>
#include<vector>

using namespace std;
const int N = 14, M = 1<<12, mod = 1e8;
vector<int> head[M]; //存储合法的转移状态
vector<int> state;
int f[N][M];
int n,m;
int g[N];

bool check(int x)
{
    for(int i = 0; i < m; i ++)
    {
        if(( (x >> i) & 1) && (x>> (i+1) & 1) )
            return false;
    }
    return true;
}

int main()
{
    cin>>n>>m;
    for(int i = 1; i <= n; i ++)
        for(int j = 0; j < m; j ++)
        {
            int t;
            cin>>t;
            g[i] += !t * (1 << j);            
        }
        
    for(int i = 0; i < 1<<m; i ++ )
    {
        if(check(i))
        {
            state.push_back(i); //初次筛选:左右相邻不能同时为1
        }
    }
    
    for(int i = 0; i < state.size(); i ++)
    {
        for(int j = 0; j < state.size(); j ++)
        {
            int a = state[i], b = state[j];
            if( (a & b) == 0) //上下相邻不能同时为1
            {
                head[i].push_back(j); //若是合法就加入到可转移数组中
            }
        }
    }
    
    f[0][0] = 1;
    for(int i = 1; i <= n + 1; i ++)
    {
        for(int a = 0; a < state.size(); a ++)
        {
            if(!(state[a] & g[i]))
                for(int b : head[a])
                    f[i][a] =(f[i][a] + f[i-1][b]) % mod;
        }
    }
    cout<<f[n+1][0];
        
}

状压——集合

所有小猪击中状态由一串二进制数来表达,若一个小猪能被击中,那么该小猪对应到二进制表达上的位置就制成1。那么我们接下来要做的就是枚举所有抛物线,并求一下抛物线能击中哪些小猪。并将这个结果存放在path数组里。

state: 二进制表达式,如1100,表示前两只小猪没被击中,后两只被击中了

path[i][j] = state : 含义:经过点i和j的抛物线; 属性:遗传二进制数,表示所有小猪的状态

f[i]: i其实就是state,从0~2^N枚举i的二进制表达,直到枚举到111111(所有位上都是1时)说明所有小猪均已经被击落。 那么f[i]的含义就是所有小猪在该状态下最少可以用多少条抛物线覆盖

那么接下来就是枚举所有状态下的小猪的覆盖方式,更新最小覆盖的抛物线数量

#include<iostream>
#include<cstring>
#include<algorithm>
#include<vector>
#include<cmath>

#define x first
#define y second

using namespace std;
typedef pair<double, double> PDD;
const int N = 18, M = 1<<18;
const double eps = 1e-8;
int f[M];
PDD q[N];
int path[N][N];
int n,m;

int cmp(double x, double y)
{
    if(fabs(x - y) < eps) return 0;
    if(x < y) return -1;
    return 1;
}

int main()
{
    int T;
    cin>>T;
    while(T--)
    {
        cin>>n>>m;
        
        for(int i = 0; i<n; i ++) cin>>q[i].x>>q[i].y;
        
        memset(path, 0, sizeof path);
        
        for(int i = 0; i < n; i ++)  //枚举所有的二次函数方程(两点确定一个抛物线,因为抛物线过原点),这里是找第一个点
        {
            path[i][i] = 1 << i;
            for(int j = i; j < n; j ++) //找这个函数的第二个点
            {
                double x1 = q[i].x, x2 = q[j].x;
                double y1 = q[i].y, y2 = q[j].y;
                
                if(!cmp(x1,x2)) continue;
                double a = (y1 / x1 - y2 / x2) / (x1 - x2); //通过两点,解出抛物线方程
                double b = y1 / x1 - a * x1;
                
                if(cmp(a, 0) >= 0) continue;
                int state = 0;
                        
                for(int k = 0; k < n; k ++) //计算有多少点在这条抛物线上
                {   
                    double x = q[k].x, y = q[k].y;
                    if(!cmp(a*x*x + b * x, y)) state +=1 << k;
                }
                path[i][j] = state;
            }
        }
        
        memset(f, 0x3f, sizeof f);
        f[0] = 0;                          
        for(int i = 0; i < 1<<n; i ++) //枚举所有小猪击中状态
        {
            int x = 0;
            for(int j = 0; j < n; j ++) //若找到没被击中的小猪,那就要更新f[i]。
            {
                if(!( i >> j & 1))
                {
                    x = j;
                    break; //注意到这里是break,即只用更新一次,因为后面还会进行一次更新,避免掉重复的更新工作
                }
            }
            
            for(int k = 0; k < n; k ++)
            {
                f[i | path[x][k]] = min(f[i | path[x][k]], f[i] + 1);
            }
        }
        cout<<f[(1<<n) - 1]<<endl;
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/314807.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数据库悲观锁 select for update的详解

一 作用 1.1 结论 在mysql中&#xff0c;select ... for update 仅适用于InnoDB&#xff0c;且必须在事务块中才能生效。Innodb引擎默认是行锁。 Select .... from where .... for update 如果在where的查询条件字段使用了【主键|索引】&#xff0c;则此命令上行锁。否…

“Frontiers”系列多本期刊分区下跌,1本SCI被踢,2本SCI升为Top,还可投吗?

近期&#xff0c;2023年中科院分区正式发布&#xff0c;不少学者都很关心期刊变动情况。此次分区更新中&#xff0c;Frontiers出版社旗下的医学期刊表现让人大跌眼镜。 据汇总来看&#xff0c;32本大类医学SCI期刊中&#xff0c;Frontiers of Hormone Research直接从原来的医学…

C语言操作符详解与进制

目录 一&#xff1a;操作符的分类 二&#xff1a;二进制和进制转换 2.1 2进制转10进制 2.1.1 10进制转2进制数字 2.2 2进制转8进制和16进制 2.2.1 2进制转8进制 2.2.2 2进制转16进制 三&#xff1a; 原码、反码、补码 四&#xff1a;移位操作符 4.1左移操作符 4.2 右…

AOT-GAN-for-Inpainting项目解读|使用AOT-GAN进行图像修复

项目地址&#xff1a; https://github.com/researchmm/AOT-GAN-for-Inpainting 基于pytorch实现 论文地址&#xff1a; https://arxiv.org/abs/2104.01431 开源时间&#xff1a; 2021年 项目简介&#xff1a; AOT-GAN-for-Inpainting是一个开源的图像修复项目&#xff0c;其对 …

c++学习笔记-STL案例-机房预约系统3-登录模块

前言 衔接上一篇“c学习笔记-STL案例-机房预约系统2-创建身份类”&#xff0c;本文主要设计登录模块&#xff0c;建立globalFile.h头文件定义使用的文件名字符串&#xff0c;登录函数封装&#xff0c;并对学生登录、老师登录、管理员登录进行了具体实现。 目录 6 登录模块 6…

外卖骑手与行人之间的非零和博弈

一、背景 自2013年成立以来&#xff0c;美团外卖一直保持着高速增长&#xff0c;通过提供便捷、高效的外卖服务&#xff0c;满足了大量消费者的需求。美团外卖的服务不仅限于基础的送餐服务&#xff0c;还涵盖了多种生活服务&#xff0c;如超市便利、药品配送等&#xff0c;满…

记录汇川:H5U与Fctory IO测试10

主程序&#xff1a; 子程序&#xff1a; IO映射 子程序&#xff1a; 自动程序 Fctory IO配置&#xff1a; HMI配置&#xff1a; 实际动作如下&#xff1a; Fctory IO测试10

档案数字化怎样快速整理资料

对于机构和组织来说&#xff0c;档案数字化是一个重要的信息管理和保护措施。要快速整理资料进行档案数字化&#xff0c;可以遵循以下步骤&#xff1a; 1. 准备工具和设备&#xff1a;确保有一台计算机、扫描仪和相关软件。 2. 分类和组织资料&#xff1a;先将资料分类&#xf…

一文解析Web缓存代理

Web缓存代理在现代网络架构中起着非常重要的作用&#xff0c;它可以减少网络传输延迟&#xff0c;提高网站的性能和用户体验。本文将深入解析Web缓存代理的原理、工作方式以及优势&#xff0c;帮助读者更好地理解和应用这一技术。 在Web应用中&#xff0c;数据的快速传输是至关…

[足式机器人]Part2 Dr. CAN学习笔记-Advanced控制理论 Ch04-5稳定性stability-李雅普诺夫Lyapunov

本文仅供学习使用 本文参考&#xff1a; B站&#xff1a;DR_CAN Dr. CAN学习笔记-Advanced控制理论 Ch04-5稳定性stability-李雅普诺夫Lyapunov Stability in the sense of Lyapunov Assympototic Stability

关于白盒测试,这些技巧你得游刃有余~

对于很多刚开始学习软件测试的小伙伴来说&#xff0c;如果能尽早将黑盒、白盒测试弄明白&#xff0c;掌握两种测试的结论和基本原理&#xff0c;将对自己后期的学习有较好的帮助。今天&#xff0c;我们就来聊聊黑盒、白盒测试的相关话题。 1、黑盒测试的方法和小结 最常见黑盒…

Elasticsearch倒排索引详解

倒排索引&#xff1a; 组成 term index(词项索引 &#xff0c;存放前后缀指针) Term Dictionary&#xff08;词项字典&#xff0c;所有词项经过文档与处理后按照字典顺序组成的一个字典&#xff08;相关度&#xff09;&#xff09; Posting List&#xff08;倒排表&#xf…

Asp .Net Core 系列: 集成 Consul 实现 服务注册与健康检查

文章目录 什么是 Consul?安装和运行 ConsulAsp .Net Core 如何集成 Consul 实现服务注册和健康检查Consul.AspNetCore 中的 AddConsul 和 AddConsulServiceRegistration 方法 究竟做了什么&#xff1f;AddConsul 方法AddConsulServiceRegistration 方法 配置 Consul 检查服务封…

16 张动图讲透网络原理

网络其实存在于日常生活中的每一个角落。 你的电脑&#xff0c;打印机&#xff0c;手机&#xff0c;甚至电视等等都属于网络设备。通常&#xff0c;你需要将这些设备通过网络连接起来&#xff0c;这样就可以实现数据的传输和共享&#xff0c;让工作生活更加便捷。 如果你的连接…

云服务器哪家强?当属阿里云腾讯云or华为云?

云服务器哪家强?当属阿里云腾讯云or华为云&#xff1f;云服务器哪家便宜&#xff1f;2024最新整理你要的都在这&#xff01;头部云厂商阿里云、腾讯云、华为云、京东云、UCloud等都在降价&#xff0c;阿腾云atengyun.com分享2024年云服务器租用价格给你惊喜&#xff01; 便宜云…

乱 弹 篇(一)

题记 对于“乱弹”这个词汇的释义&#xff0c;《辞海》上仅有“ 戏曲剧种&#xff0c;亦指声腔 ”8个字。而由于“乱弹 ”的“ 弹”谐音“谈”&#xff0c;这就容易让人联想到“乱谈”。不过从文体上看&#xff0c;“乱谈”也非乱七八糟之谈&#xff0c;反倒是“东西南北&…

系分笔记数据库反规范化、SQL语句和大数据

文章目录 1、概要2、反规范化3、大数据4、SQL语句5、总结 1、概要 数据库设计是考试重点&#xff0c;常考和必考内容&#xff0c;本篇主要记录了知识点&#xff1a;反规范化、SQL语句及大数据。 2、反规范化 数据库遵循范式的设计&#xff0c;使得多表查询和连接表查询较多的时…

Tomcat简介及搭建

1、Tomcat概述 自2017年11月编程语言排行榜 Java 占比 13%&#xff0c;高居榜首&#xff0c;Tomcat也一度成为Java开发人员的首选。其开源、占用系统资源少、跨平台等特性深受广大程序员喜爱。本篇文章主要讲解如何部署 Tomcat 服务&#xff0c;根据生产环境实现多个虚拟主机的…

一种具有轨迹优化的无人驾驶车实时运动规划器 论文阅读

论文题目&#xff1a;A Real-Time Motion Planner with Trajectory Optimization for Autonomous Vehicles Abstract 本文的实时规划器首先将空间离散化&#xff0c;然后基于一组成本函数搜索出最佳轨迹。迭代优化所得到的轨迹的Path和Speed。post-optimization计算复杂度低&…

蚂蚁爱购--靠谱的SpringBoot项目

简介 这是一个靠谱的SpringBoot项目实战&#xff0c;名字叫蚂蚁爱购。从零开发项目&#xff0c;视频加文档&#xff0c;十天就能学会开发JavaWeb项目。 教程路线是&#xff1a;搭建环境> 安装软件> 创建项目> 添加依赖和配置> 通过表生成代码> 编写Java代码&g…