#include<bits/stdc++.h>
using namespace std;
const long long mod=998244353;
long long f[200007][2],g[200007][2];
long long a[200007],b[200007],c[200007];
int n,k,cnt1,cnt2;
long long qpow(long long a,long long p){
    long long ans=1;
    while(p){
        if(p&1)
            ans=ans*a%mod;
        a=a*a%mod;
        p>>=1;
    }
    return ans;
}
long long solve(long long *a,int n,int k){
    for(int i=2;i<=n;i++)
        if(a[i]==a[i-1]&&a[i]!=-1)//一定有奇数回文串,无法构造出good串
            return 0;
    int p=0;
    for(int i=1;i<=n;i++)
        if(a[i]!=-1)
            p++;
    if(!p)
        return qpow(k-1,n-1)*k%mod;//第一个随意选,剩下的选择和前一个不同的
    else if(p==1)
        return qpow(k-1,n-1);//选的和相邻的不同的
    else{
        if(p==n)
            return 1;//没有构造余地,全部已经安排好了
        if(k==1)//构造不出,一定会存在1?1只羊这样的回文串
            return 0;
        long long res=1;
        int l=0,r=0;
        for(int i=1;i<=n;i++){
            if(a[i]!=-1){
                l=i+1;
                break;
            }
            res=res*(k-1)%mod;//-1只要构造和相邻的不同即可
        }
        for(int i=n;i;i--){
            if(a[i]!=-1){
                r=i;
                break;
            }
            res=res*(k-1)%mod;//-1只要构造和相邻的不同即可
        }
        long long tmp=0;
        long long last=a[l-1];//上一个已经确定的a[i]
        for(int i=l;i<=r;i++){
            if(a[i]!=-1){
                if(!tmp){//前一个也已经确定
                    last=a[i];//更新最新一个确定的a[i]后继续
                    continue;
                }
                if(a[i]!=last)//两端数字不同
                    res=res*f[tmp][0]%mod;
                else//两端数字相同
                    res=res*g[tmp][0]%mod;
                tmp=0;//归零
                last=a[i];//更新
            }
            else
                tmp++;//增加中间-1的长度
        }
        return res;
    }
}
int main(){
    int n,k;
    scanf("%d%d",&n,&k);
    for(int i=1;i<=n;i++){
        scanf("%lld",&a[i]);
    }
    //f和g类似染色,预处理出除了两端不是-1中间全都是-1能有多少种安排情况
    f[1][0]=k-2;//表示两端数字不同时,第一维和结尾不同的方案数,k-2因为要和两边都不同
    f[1][1]=1;//表示两端数字不同时,第一维和结尾相同的方案数,1因为下一个就可以在和上一个相同的那一段中间插入一个
    g[1][0]=k-1;//表示两端数字相同时,第一维和结尾不同的方案数,k-1因为两边是相同的
    g[1][1]=0;//表示两端数字相同时,第一维和结尾相同的方案数,0因为下两个才可以在最初两边都插一个
    for(int i=2;i<=n/2+1;i++){//染色DP,分开讨论,每次在前面的情况后面加上一个
        f[i][0]=(f[i-1][0]*(k-2)%mod+f[i-1][1]*(k-1)%mod)%mod;
        f[i][1]=f[i-1][0];
        g[i][0]=(g[i-1][0]*(k-2)%mod+g[i-1][1]*(k-1)%mod)%mod;
        g[i][1]=g[i-1][0];
    }
    for(int i=1;i<=n;i+=2)
        b[++cnt1]=a[i];//分离出下标为奇数的a[i]
    for(int i=2;i<=n;i+=2)
        c[++cnt2]=a[i];//分离出下标为偶数的a[i]
    long long ans=solve(b,cnt1,k);
    ans=ans*solve(c,cnt2,k)%mod;
    printf("%lld",ans);
    return 0;
}

Educational Codeforces Round 62 (Rated for Div. 2)E(染色DP,构造,思维,组合数学)的更多相关文章

  1. Educational Codeforces Round 73 (Rated for Div. 2)D(DP,思维)

    #define HAVE_STRUCT_TIMESPEC#include<bits/stdc++.h>using namespace std;long long a[300007],b[3 ...

  2. Educational Codeforces Round 69 (Rated for Div. 2)D(DP,思维)

    #include<bits/stdc++.h>using namespace std;int a[300007];long long sum[300007],tmp[300007],mx[ ...

  3. Educational Codeforces Round 102 (Rated for Div. 2) B. String LCM (构造,思维)

    题意:给你两个字符串\(a\)和\(b\),找出它们的\(lcm\),即构造一个新的字符串\(c\),使得\(c\)可以由\(x\)个\(a\)得到,并且可以由\(y\)个\(b\)得到,输出\(c\ ...

  4. Educational Codeforces Round 62 (Rated for Div. 2) Solution

    最近省队前联考被杭二成七南外什么的吊锤得布星,拿一场Div. 2恢复信心 然后Div.2 Rk3.Div. 1+Div. 2 Rk9,rating大涨200引起舒适 现在的Div. 2都怎么了,最难题 ...

  5. Educational Codeforces Round 62 (Rated for Div. 2)

    A. Detective Book 题意:一个人读书  给出每一章埋的坑在第几页可以填完 . 一个人一天如果不填完坑他就会一直看 问几天能把这本书看完 思路:模拟一下 取一下过程中最大的坑的页数  如 ...

  6. Educational Codeforces Round 62 (Rated for Div. 2) C 贪心 + 优先队列 + 反向处理

    https://codeforces.com/contest/1140/problem/C 题意 每首歌有\(t_i\)和\(b_i\)两个值,最多挑选m首歌,使得sum(\(t_i\))*min(\ ...

  7. Educational Codeforces Round 62 (Rated for Div. 2)C

    题目链接 :C. Playlist #include<bits/stdc++.h> using namespace std; #define maxn 300005 #define LL ...

  8. Educational Codeforces Round 62 (Rated for Div. 2) - C Playlist

    当时题意看错了...不过大致思路是对的,唯一没有想到的就是用优先队列搞这个东西,真是不该啊... 题意大概就是,有N首歌,N首歌有两个东西,一个是长度Ti,一个是美丽值Bi,你最多可以选择K首歌, 这 ...

  9. C. Playlist Educational Codeforces Round 62 (Rated for Div. 2) 贪心+优先队列

    C. Playlist time limit per test 2 seconds memory limit per test 256 megabytes input standard input o ...

随机推荐

  1. Java标准I/O流介绍

    1.I/O是什么? I/O 是Input/Output(输入.输出)的简称,输入流可以理解为向内存输入,输出流是从内存输出. 2.流 流是一个连续的数据流,可以从流中读取数据,也可以往流中写数据.流与 ...

  2. oracle 12c使用问题总结

    1.无法登录 安装完毕只能使用system和sys用户,用安装时配置的密码登录:不能使用默认密码 2.远程无法访问 1)检测服务器配置 lsnrctl status 看到(DESCRIPTION=(A ...

  3. 安全测试回顾(一)补充:burp 的基本操作

    浏览器设置; 拦截请求: Spider 伪造请求: 对这个url 伪造请求 拦截一个页面后,进入攻击模块 sniper  两个参数值 保证一个不变  另一个 进行枚举 battering ram 两个 ...

  4. xp远程桌面连接最大用户数怎么设置?

    1.首先到网上去百度下载“补丁UniversalTermsrvPatch”,这个补丁主要目的是在于去除“单用户登陆的限制”,允许多人多用户同时并行访问登录;2.然后根据自己的系统运行对应的程序:系统是 ...

  5. Centos7 第三方仓库 yum 方式安装 PHP7.2

    1.卸载原先安装的PHP yum remove php rpm -qa|grep php #列出所有的php相关的rpm包 rpm -e xxx #xxx指的是上一个命令列出的rpm包的包名,复制即可 ...

  6. JS判断IE,FF,Opera,Safari等浏览器类型

    第一种,只区分浏览器,不考虑版本 function myBrowser(){ var userAgent = navigator.userAgent; //取得浏览器的userAgent字符串 var ...

  7. delphi 天气预报

    天气预报 var astream : tmemorystream; sStream : TStringStream; jv : TJSONValue; begin astream := tmemory ...

  8. Mycat主从模式下的读写分离与自动切换

    1. 机器环境 192.168.2.136 mycat1 192.168.2.134 mydb1 192.168.2.135 mydb2 2在mysql1.mysql2上安装mysql 更改root用 ...

  9. C++深度解析教程学习笔记(5)面向对象

    1. 面向对象基本概念 (1)面向对象的意义在于 ①将日常生活中习惯的思维方式引入程序设计中 ②将需求中的概念直观的映射到解决方案中 ③以模块为中心构建可复用的软件系统 ④提高软件产品的可维护性和可扩 ...

  10. SonarQube在CentOS上的安装

    1 简介 SonarQube 是一个用于代码质量管理的开放平台.通过插件机制,Sonar 可以集成不同的测试工具,代码分析工具,以及持续集成工具.与持续集成工具(例如 Hudson/Jenkins 等 ...