DP(9)--插头DP

DP(9)--插头DP

/*
    Mondriaan’s Dream题目大意:在 N*M 的棋盘内铺满 1*2 或 2*1 的多米诺骨牌,求方案数。
    
    砖只有横放和竖放两种状态,把横放记为两个0,竖放记为上1下0,逐格DP,每次无论前一格放怎么放,
    当前格可竖放或不放,而如果前一格是1,且当前格是0,那么我们可以把前一格改成0,再把当前格也放上0组成一块。
    这样成对合并和生成插头。
*/

#include <iostream>
#include <string.h>
using namespace std;

const int W = 11;
long long f[2][1 << W];

// 将a的第b位取反,最低位编号为0
int flapBit(int a, int b) 

    return a ^ (1 << b);
}

// O(h*w*2^w), 交换为了得到窄列,降低时间复杂度 
long long calc(int h, int w)
{
    if (h < w)
        swap(h, w);

    memset(f, 0, sizeof(f));
    int cur = 0;
    f[cur][0] = 1;
    for (int i = 0; i < h; ++i)
        for (int j = 0; j < w; ++j)
        {
            cur ^= 1;
            memset(f[cur], 0, sizeof(f[cur]));
            for (int k = 0; k < (1 << w); ++k) // 枚举状态
            {
                if (f[cur^1][k] > 0)
                {
                    f[cur][flapBit(k, j)] += f[cur^1][k];  // 竖放或不放
                    if (j != w - 1 && (!((k >> j) & 3))) // 非最后一列且满足能容纳2个单位的宽度
                        f[cur][flapBit(k, j+1)] += f[cur^1][k];  // 横放
                }
            }
        }

    return f[cur][0];
}

int main()
{
    int h, w;
    while (cin >> h >> w && h != 0 && w != 0)
        cout << calc(h, w) << endl;

    return 0;
}

一个方向的插头存在表示这个格子在这个方向可以与外面相连。
对于一个4连通的问题来说,它通常有上下左右4个插头。

插头的连通性(有没有下插头,与轮廓线直接相连的插头)

砖只有横放和竖放两种状态,把横放记为两个0,竖放记为上1下0,逐格DP,每次无论前一格放怎么放,当前格可竖放或不放,
而如果前一格是1,且当前格是0,那么我们可以把前一格改成0,再把当前格也放上0组成一块。

dp[i][j][state] 为位置(i,j),状态为state的方案数

插头存在信息 格子连通信息

优化为位置(i,j)插头的连通状态为S的方案数: f(i, j, S)
f(3, 1, {1, 0, 1, 2, 2}), f(3, 2, {1, 0, 1, 2, 2}), f(3, 3, {1, 0, 0, 0, 1})


情况1 新建一个连通分量,这种情况出现在(i, j)有右插头和下插头。
新建的两个插头连通且不与其他插头连通,这种情况下需要将这两个插头
连通分量标号标记为一个未标记的正数,重新O(n)扫描保证新的状态满足最小表示。
情况2 合并两个连通分量,这种情况出现在(i, j)有上插头和左插头。
如果两个插头不连通,那么将两个插头所处的连通分量合并,标记相同的连通块标号,
O(n)扫描保证最小表示;如果已经连通,相当于出现一个回路,这种情况只能出现在最后一个非障碍格子。
情况3 保持原来的连通分量,这种情况出现在(i, j)的上插头和右插头恰好有一个或下插头和左插头也恰好有一个。
下插头或右插头相当于上插头或左插头的延续,连通块标号相同,并且不会影响到其他的插头的连通块标号,
计算新的状态的时间为O(1).

注意当从一行的最后一个格子转移到下一行的第一个格子的时候,轮廓线需要特殊处理。
值得一提的是,上面三种情况计算新的状态的时间分别为O(n), O(n), O(1), 如果使用前面提到的第二种最小表示法,
情况1只需要O(1),但是情况3可能需要O(n)重新扫描。

对n+1个元素进行编码,将其表示成一个n+1位的p进制数,p可以取能够达到的最大的连通块标号加1,
对本题来说,最多出现n/2<=6个连通块,不妨取p=7,在不超过数据类型范围的前提下,建议将p改为2的幂,
因为位运算比较快,本题最好采用8进制来存储。
如需大范围修改连通块标号,最好将状态O(n)解码到一个数组中,修改后再O(n)计算出新的p进制数,
而对于只需要局部修改几个标号的情况下,可以直接用(x/p^(i-1))%p来获取第i位,用加或者减k*p^(i-1)
直接对第i位进行修改。


唯一特殊的是上一行末到这一行头的处理。上一行末不可能有右插头,那我们直接把上一行末状态的表示最后是否存在右插头的位置去掉,
再添加一个表示没有左插头的位,就表示出了下一行行首的状态,为了方便写,在代码里,用dp[i][0][mask]表示转移后的上一行行末状态。

//https://www.luogu.com.cn/problem/P5056

#include <iostream>
#include <string.h>
using namespace std;
const int M = 15;
const int offset = 3, mask = (1 << offset) - 1;
int n, m;
long long ans;
//MaxSZ: 合法状态的上界,可以估计,也可以预处理出较为精确的值。
//Prime: 一个小于 MaxSZ 的大素数
const int MaxSZ = 16796, Prime = 9973;
bool path[M][M];

struct hashTable 
{
/*
    head[] 表头节点的指针。
    next[] 后续状态的指针。
    state[] 节点的状态。
    key[] 节点的关键字,在本题中是方案数。
*/
    int head[Prime], next[MaxSZ], sz;
    long long state[MaxSZ];
    long long key[MaxSZ];

/* 初始化函数,和手写邻接表类似,我们只需要初始化表头节点的指针 */
    inline void init() 
    {
        sz = 0;
        memset(head, -1, sizeof(head));
    }
/*  
    状态转移函数,其中 d 表示每次状态s转移所带来的增量。
    如果找到的话就 +=,否则就创建一个状态为 s,关键字为 d 的新节点
*/
    inline void push(long long s, long long d) 
    {
        int x = s % Prime;
        for (int i = head[x]; ~i; i = next[i])
        {
            if (state[i] == s) // s状态存在,直接更新key[]
            {
                key[i] += d;
                return;
            }
        }
        // 添加新状态
        state[sz] = s;
        key[sz] = d;
        next[sz] = head[x];
        head[x] = sz++;
    }
}H[2];

/*
    code[]: 轮廓线上的插头的状态编码
    arr[]: 最小表示法的编码过程中,每个数字被映射到的最小数字。0表示插头不存在,不能被映射到其他值。
*/
int code[M + 1], arr[M + 1];

/*
    最小表示法 m<=12 最多只有6个不同的连通分量,
    对m+1个元素进行编码,将其表示成一个m+1位的p进制数,p可以取能够达到的最大的连通块标号加1,
    本题最好采用8进制来存储, 将插头连通状态数组code进行8进制压缩,转为8进制数s。
*/
long long encode() 
{
    long long s = 0;
    memset(arr, -1, sizeof(arr));
    // 最小表示法,连通块编号从1开始
    int bn = 1;
    arr[0] = 0;
    for (int i = 0; i <= m; ++i)
    {
        if (!~arr[code[i]]) // arr[] 为 -1, 即出现一个新的连通分量,添加新编号
            arr[code[i]] = bn++;
        s <<= offset;  // 逐位进行8进制压缩
        s |= arr[code[i]];
    }

    return s;
}

// 将8进制压缩码解析到code数组
void decode(long long s)
{
    for (int i = m; i >= 0; --i)
    {
        code[i] = s & mask;
        s >>= offset;
    }
}

void push(int cur, int j, int dn, int rt, long long d) 
{
    code[j-1] = dn;
    code[j] = rt;
    H[cur].push(encode(), d);
}

int main() 
{
    cin >> n >> m;
    char str[32] =  { '\0' };
    int row = 0, colum = 0;
    for (int i = 1; i <= n; ++i)
    {
        cin >> str+1;
        for (int j = 0; j <= m; ++j)
            if (str[j] == '.')
            {
                path[i][j] = true;
                row = i;
                colum = j;
            }
            else
                path[i][j] = false;
    }

    if (!row)
    {
        cout << 0 << endl;
        return 0;
    }

    int cur= 0;
    H[cur].init();
    long long d = 1; // 初始状态0的增量delta为1
    H[cur].push(0, d);
    for (int i = 1; i <= n; ++i) 
    {
        for (int j = 1; j <= m; ++j) 
        {
            if (path[i][j])
            {
                cur ^= 1;
                H[cur].init();
                for (int s = 0; s < H[cur^1].sz; ++s) 
                {
                    decode(H[cur^1].state[s]);  // 取出状态,并解码
                    d =  H[cur^1].key[s];        // 得到增量 delta
                    int lt = code[j-1], up = code[j]; // 左插头,上插头

                    if (lt && up) // 如果左、上均有插头
                    {
                        if (lt == up) // 来自同一个连通块
                        { 
                            if (i == row && j == colum)  // 只有在最后一个格子时,才能合并,封闭回路。
                                push(cur, j, 0, 0, d);
                        } 
                        else  // 否则,必须合并这两个连通块,因为本题中需要回路覆盖
                        {
                            for (int k = 0; k <= m; ++k)
                                if (code[k] == lt) 
                                    code[k] = up;
                            push(cur, j, 0, 0, d);
                        }
                    } 
                    else if (lt || up)  // 如果左、上之中有一个插头
                    {
                        int t = lt | up;// 得到这个插头
                        if (path[i+1][j]) // 如果可以向下延伸
                            push(cur, j, t, 0, d);
                        if (path[i][j+1]) // 如果可以向右延伸
                            push(cur, j, 0, t, d);
                    }
                    else // 如果左、上均没有插头
                    {
                        if (path[i+1][j] && path[i][j+1])  // 生成一对新插头
                            push(cur, j, 7, 7, d);         // 插头连通分量最大值不超过7
                    }
                }
            }
        }
        /*  迭代完一整行之后,滚动轮廓线 */
        for (int j =0; j < H[cur].sz; ++j)
            H[cur].state[j] >>= offset;

    }
    

    cout << (H[cur].sz > 0 ? H[cur].key[0] : 0) << endl;

    return 0;
}

/*
测试数据
4 4
**..
....
....
....
2


4 4
....
....
....
....
6

12 12
..**********
...*********
....********
*....*******
**....******
***....*****
****....****
*****....***
******....**
*******....*
********....
*********...
1


*/
参考:

https://oi-wiki.org/dp/plug/

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

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

相关文章

详解MySQL慢SQL定位、分析

目录 1.概述 2.慢SQL定位 3.SQL性能分析 3.1.例子 3.2.SQL性能分析 3.3.参数说明 3.3.1.id 3.3.2.select_type 3.3.3.key_len 3.3.4.rows 3.3.5.type 3.3.6.extra 1.概述 解决慢SQL的问题无非3步&#xff1a; 定位慢SQL分析慢SQL优化慢SQL 本文将按顺序介绍前两…

【MySQL】SQL优化

上一篇索引是针对查询语句进行优化,但在MySQL中可不仅有查询语句,针对其他的SQL语句同样也能进行优化 文章目录 1.插入数据2.主键优化3.order by 优化4.group by优化5.limit优化6.update优化 1.插入数据 插入数据所使用的关键字为insert,SQL语句为 insert into 表名(字段1,字…

恢复item2和oh-my-zsh的配置

1. 首先正常安装item2 2. 加载onedrive里的传家宝iterm2_default_profile.json&#xff0c;让iterm2的配置生效 2. 然后正常安装oh-my-zsh (官方步骤&#xff1a; sh -c "$(curl -fsSL https://raw.githubusercontent.com/ohmyzsh/ohmyzsh/master/tools/install.sh)&q…

BUUCTF ciscn_2019_c_1

小白垃圾做题笔记而已&#xff0c;不建议阅读。 1前期&#xff1a; 其实刚开始拿到程序的时候我还以为是逆向题放错地方了。唉&#xff0c;做题太少了。啥也不会。我是大笨蛋。 题目中用的是ubuntu18&#xff0c;我的ubuntu没怎么用过&#xff0c;vmtools都不能用&#xff0c…

什么是GPT模型,GPT下载和国内镜像

什么是GPT模型&#xff0c;GPT模型是通过预训练的方式&#xff0c;采用无监督学习方式&#xff0c;大量语料输入&#xff0c;经过多次训练后得到模型。它能够自动学习并理解自然语言中的语义、句法和语法信息&#xff0c;并可以用于文本生成、对话系统、情感分析、机器翻译等自…

零死角玩转stm32中级篇3-SPI总线

本篇博文目录: 一.基础知识1.什么是SPI2.SPI和IIC有什么不同3.SPI的优缺点4.SPI是怎么实现通信的5.SPI 数据传输的步骤6.SPI菊花链7.通过SPI实现数据的读和写 二.STM32F103C8T6芯片SPI协议案例代码 一.基础知识 1.什么是SPI SPI&#xff08;Serial Peripheral Interface&#…

Flask开发之环境搭建

目录 1、安装flask 2、创建Flask工程 ​编辑 3、初始化效果 4、运行效果 5、设置Debug模式 6、设置Host 7、设置Port 8、在app.config中添加配置 1、安装flask 如果电脑上从没有安装过flask&#xff0c;则在命令行界面输入以下命令&#xff1a; pip install flask 如果电…

给大家介绍几个手机冷门但好用的小技巧

技巧一&#xff1a;拍照识别植物 手机的拍照识别植物功能是指在使用手机相机时&#xff0c;可以通过对植物进行拍照&#xff0c;并通过植物识别技术&#xff0c;获取植物的相关信息和资料。其主要优点如下&#xff1a; 方便实用&#xff1a;使用拍照识别植物功能&#xff0c;…

【Java笔试强训 18】

&#x1f389;&#x1f389;&#x1f389;点进来你就是我的人了博主主页&#xff1a;&#x1f648;&#x1f648;&#x1f648;戳一戳,欢迎大佬指点! 欢迎志同道合的朋友一起加油喔&#x1f93a;&#x1f93a;&#x1f93a; 目录 一、选择题 二、编程题 &#x1f525;统计每…

基于springcloud微服务的java课程资源在线学习考试系统

在我国&#xff0c;由于计算机与网络技术的不断发展&#xff0c;信息化建设的不断深入&#xff0c;不管是企业、学校或个人都在结合计算机网络技术队现有的管理或生活中的一些环节进行开发研究&#xff0c;运用计算机进行一些必要的数据信息管理&#xff0c;分析及发布&#xf…

拷贝构造函数和赋值重载函数详解

1.拷贝构造函数 1.1拷贝构造函数的概念 拷贝构造函数&#xff1a;只有单个形参&#xff0c;该形参是对本类类型对象的引用(一般常用const修饰)&#xff0c;在用已存在的类类型对象创建新对象时由编译器自动调用。拷贝构造函数也是特殊的成员函数&#xff0c;其特征如下&#…

第三十一章 Unity骨骼动画

关于骨骼动画的原理&#xff0c;我们这里不再详细介绍&#xff0c;有不清楚的可以回去看DirectX课程和3dsMAX课程。接下来&#xff0c;我们来讲解一下Unity的骨骼动画系统。Unity 的动画系统基于动画剪辑&#xff08;Animation Clip&#xff09;的概念&#xff0c;它的本质就是…

LeetCode - 239 滑动窗口最大值

目录 题目来源 题目描述 示例 提示 题目解析 算法源码 题目来源 239. 滑动窗口最大值 - 力扣&#xff08;LeetCode&#xff09; 题目描述 给你一个整数数组 nums&#xff0c;有一个大小为 k 的滑动窗口从数组的最左侧移动到数组的最右侧。你只可以看到在滑动窗口内的 k…

springboot+vue前后端分离项目打包成jar包及运行

将 Spring Boot 和 Vue.js 项目打包成 jar 包需要按照以下步骤操作&#xff1a; 在项目的根目录中&#xff0c;使用命令行进入 Vue.js 项目的根目录&#xff0c;然后运行以下命令&#xff1a; npm run build这个命令将会构建 Vue.js 项目&#xff0c;并在项目的 dist 目录中生…

鸿蒙Hi3861学习八-Huawei LiteOS(事件标记)

一、简介 事件是一种实现任务间通信的机制&#xff0c;可用于实现任务间的同步。但事件通信只能是事件类型的通信&#xff0c;无数据传输。一个任务可以等待多个事件的发生&#xff1a;可以是任意一个事件发生时唤醒任务进行事件处理&#xff1b;也可以是几个事件都发生后才唤醒…

华为网络设备+WinRadius 实现用户统一管理设备

一、直接贴配置 ###配置VTY用户界面所支持的协议、验证方式 user-interface vty 0 4 protocol inbound telnet authentication-mode aaa quit ###配置RADIUS认证 ###&#xff08;1&#xff09;配置RADIUS服务器模板&#xff0c;指定服务器的IP地址与端口号、共享密钥 radius-s…

Unity - Render Doc - 解决 Waiting For Debugger 导致连接不了 APP 的问题

环境 Unity : 2020.3.37f1 Pipeline : BRP RDC : 1.26 问题 平常有一些公司内的游戏发布在移动端运行会有各种异常&#xff0c;但是 unity editor (android opengl es / dx) 下正常 如果没有真机抓帧分析&#xff0c;是搞不定的 然后 RenderDoc 在抓发布出来的调试包也抓不…

漫画 | Linux之父:财务自由以后,我失眠了!

前言&#xff1a;今年是Linux诞生的30周年&#xff01; 1991年的8月&#xff0c; Linus在新闻组中公布了他正在开发的一个免费的操作系统&#xff0c;这也是以后风靡世界的Linux操作系统的雏形。 今天翻到这篇漫画&#xff0c;看到Linux的诞生过程&#xff0c;很是感慨&#x…

SuperMap GIS基础产品云GIS FAQ集锦(2)

SuperMap GIS基础产品云GIS FAQ集锦&#xff08;2&#xff09; 【iManager】云套件ispeco-dashboard-api的日志等级只有到info&#xff0c;如何设置才能查看到debug级别的日志&#xff1f; 【解决方案】可以在ispeco-dashboard-api的deployment中添加以下环境变量&#xff0c;…

vue框架快速入门

vue 1、第一个Vue程序1.1、什么是Vue程序1.2、为什么要使用MVVM1.3、Vue1.4、第一个vue程序 2、基础语法2.1、v-bind2.2、v-if&#xff0c; v-else2.3、v-for2.4、v-on 3、Vue表单双绑、组件3.1、什么是双向数据绑定3.2、在表单中使用双向数据绑定3.3、什么是组件 4、Axios异步…