微信步数C++

题目:

 


样例解释:

【样例 #1 解释】

从 (1,1) 出发将走 2 步,从 (1,2) 出发将走 4 步,从 (1,3) 出发将走 4 步。
从 (2,1) 出发将走 2 步,从 (2,2) 出发将走 3 步,从 (2,3) 出发将走 3 步。
从 (3,1) 出发将走 1 步,从 (3,2) 出发将走 1 步,从 (3,3) 出发将走 1 步。
共计 21 步。

 


思路:

先考虑O(nkw)O(nkw)的30分暴力。

显然,每个维度上走过的位置是一个区间。

只要走的步数确定,那么这个区间关于起点位置的相对位置也就确定了。

只要先算出每个循环向左/右所走的最远距离,以及一个循环的移位即可。

这样,考虑一个算法:

枚举走了多少步结束,并算出贡献(就是算出满足条件的起点数目)。

先枚举走出区域的上一步,走到了循环节中的哪个位置,以及走了多少循环节。

由于不能走出区域,于是可以根据每个维度的区间来算出这个维度的起点所在区间。

设下一步修改的维度为cc。根据对应的dd,容易算出这个维度的起点位置。

那么,这个位置必须在起点区间内。

满足这个条件的基础上,把其他维度的起点区间长度相乘就是起点数目。

考虑优化:

这个算法的主要瓶颈在于对循环节数的枚举。

设走过的循环节数目为xx。

那么,不难发现,每个维度的区间的相对位置(即左右端点与起点的距离)是关于xx的一次函数。

由于这一维度的方案数等于w+1w+1减去区间长度,因此这也是关于xx的一次函数。

根据这个区间长度为正数,可以得出xx的取值范围。

同时,维度cc的起点位置也是关于xx的一次函数。

根据这个位置必须在起点区间内部,进一步缩小xx的取值范围。

这样,答案就是对于每个xx,这些一次函数在xx处的值的乘积的和。

暴力进行多项式乘法并用自然数幂前缀和即可。

时间复杂度O(nk2)O(nk2)。

注意这个写法,可能要对x=0x=0进行特判。

 

 


代码:

#include <stdio.h>
#define inf 1999999999
#define md 1000000007
#define min(a,b) a<b?a:b
#define max(a,b) a>b?a:b
int az[12],al[12],ar[12],w[12],aa[500010][12];
int B[12],C[500010],D[500010],la[500010][12],ra[500010][12];
int ksm(int a,int b) {int jg=1;while(b>0) {if(b&1)jg=1ll*jg*a%md;a=1ll*a*a%md;b=(b>>1);}return jg;
}
int Cc[12][12],xs[12][12];
void GetB(int k)//伯努利数 
{B[0]=1;for (int i=1;i<=k+1;i++) {for (int j=0;j<=i;j++) {if(j==0||j==i)Cc[i][j]=1; else Cc[i][j]=(Cc[i-1][j]+Cc[i-1][j-1])%md;}}for (int i=1;i<=k;i++) {int h=0;for (int j=0;j<i;j++)h=(h+1ll*Cc[i+1][j]*B[j])%md;B[i]=1ll*(md-h)*ksm(i+1,md-2)%md;}for (int i=1;i<=k;i++) {int ny=ksm(i+1,md-2);for (int j=0;j<=i;j++)xs[i][i+1-j]=1ll*Cc[i+1][j]*B[j]%md*ny%md;}
}
struct SLi//一次函数 
{int k,b;SLi() {}SLi(int K,int B) {k=K;b=B;}SLi(int Z) {k=0;b=Z;}
};
SLi operator-(const SLi &x,const SLi &y) {return SLi(x.k-y.k,x.b-y.b);
}
int Sum(int k,int n) {if(k==0)return n+1;int jg=0;for (int i=0,j=1;i<=k+1;i++) {jg=(jg+1ll*j*xs[k][i])%md;j=1ll*j*(n+1)%md;}return jg;
}
struct DXS//多项式 
{int sz[12],n;void operator=(const DXS &a) {n=a.n;for (int i=0;i<=n;i++)sz[i]=a.sz[i];}void clear() {for (int i=1;i<=10;i++)sz[i]=0;sz[0]=1;n=0;}int sum(int l,int r) {int ans=0;for (int i=0;i<=n;i++) {int t=(Sum(i,r)-Sum(i,l-1)+md)%md;ans=(ans+1ll*sz[i]*t)%md;}return ans;}
};
DXS operator*(const DXS&x,const SLi&y) {DXS rt;rt.n=x.n+1;rt.sz[rt.n]=0;for (int i=0;i<=x.n;i++)rt.sz[i]=1ll*y.b*x.sz[i]%md;for (int i=0;i<=x.n;i++)rt.sz[i+1]=(rt.sz[i+1]+1ll*y.k*x.sz[i])%md;return rt;
}
struct SQj//维护区间 
{int l,r;SQj() {}SQj(int L,int R) {l=L;r=R;}
};
SQj jiao(const SQj&a,const SQj&b) {return SQj(max(a.l,b.l),min(a.r,b.r));
}
int floor(int,int);
int ceil(int x,int y) {if(y<0)x=-x,y=-y;if(x>=0)return (x+y-1)/y;return -floor(-x,y);
}
int floor(int x,int y) {if(y<0)x=-x,y=-y;if(x>=0)return x/y;return -ceil(-x,y);
}
SQj Less(SLi a,SLi b) {int x=a.k-b.k,y=b.b-a.b;if(x==0)return y>=0?SQj(-inf,inf):SQj(inf,-inf);if(x>0)return SQj(-inf,floor(y,x));return SQj(ceil(y,x),inf);
}
SQj More(SLi a,SLi b) {int x=a.k-b.k,y=b.b-a.b;if(x==0)return y<=0?SQj(-inf,inf):SQj(inf,-inf);if(x>0)return SQj(ceil(y,x),inf);return SQj(-inf,floor(y,x));
}
int cal0(int n,int i,int k) {int l[12],r[12],jg=1;for (int c=1;c<=k;c++)l[c]=la[i][c],r[c]=ra[i][c];for (int c=1;c<=k;c++) {int zl=1-l[c],zr=w[c]-r[c];if(c!=C[(i+1)%n]) {int s=zr-zl+1;if(s<0)s=0;jg=1ll*jg*s%md;} else {int t=aa[i][c],s=0;if(D[(i+1)%n]==1) {int o=w[c]-t;if(o>=zl&&o<=zr)s=1;} else {int o=1-t;if(o>=zl&&o<=zr)s=1;}jg=1ll*jg*s%md;}}return 1ll*(i+2)*jg%md;
}
int main() {int n,k;scanf("%d%d",&n,&k);GetB(k);for (int i=1;i<=k;i++)scanf("%d",&w[i]);for (int i=0;i<n;i++) {scanf("%d%d",&C[i],&D[i]);if(i>0) {for (int j=1;j<=k;j++)aa[i][j]=aa[i-1][j];}aa[i][C[i]]+=D[i];//循环节中某一前缀的偏移量az[C[i]]+=D[i];if(az[C[i]]<al[C[i]])//最左移位al[C[i]]=az[C[i]];if(az[C[i]]>ar[C[i]])//最右移位ar[C[i]]=az[C[i]];for (int j=1;j<=k;j++) {la[i][j]=al[j];ra[i][j]=ar[j];}}bool zd=false;for (int i=1;i<=k;i++) {if(az[i]!=0||ar[i]-al[i]>=w[i]) {zd=true;break;}}if(!zd)//走不出去 {printf("-1");return 0;}int ans=1;for (int i=1;i<=k;i++) {if(i!=C[0])ans=1ll*ans*w[i]%md;}for (int i=0;i<n;i++) {ans=(ans+cal0(n,i,k))%md;//特殊处理x=0的情况SLi l[12],r[12],d[12];for (int c=1;c<=k;c++)//算出对应维度的一次函数 {if(az[c]>=0) {l[c]=al[c];int t=az[c]+ra[i][c];if(ar[c]>t)t=ar[c];r[c]=SLi(az[c],t-az[c]);} else {r[c]=ar[c];int t=az[c]+la[i][c];if(al[c]<t)t=al[c];l[c]=SLi(az[c],t-az[c]);}d[c]=r[c]-l[c];}int tc=C[(i+1)%n];SLi o;SLi tz=SLi(az[tc],aa[i][tc]);if(D[(i+1)%n]==1)o=w[tc]-tz; else o=1-tz;SLi zl=1-l[tc],zr=w[tc]-r[tc];//tc这一维度起点的范围SQj qj=jiao(More(o,zl),Less(o,zr));//tc这一维度起点是确定的,需要满足条件for (int i=1;i<=k;i++) {d[i]=w[i]-d[i];qj=jiao(qj,More(d[i],1));//方案数>0}qj=jiao(qj,SQj(1,inf));if(qj.l>qj.r)continue;DXS ji;ji.clear();ji=ji*SLi(n,i+2);for (int c=1;c<=k;c++)//对应维度相乘 {if(c!=tc)ji=ji*d[c];}ans=(ans+ji.sum(qj.l,qj.r))%md;}printf("%d",(ans%md+md)%md);return 0;
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

图论day55|深度优先搜索理论基础、98. 所有可达路径(卡码网)

图论day55|深度优先搜索理论基础、98. 所有可达路径(卡码网&#xff09; 思维导图汇总深度优先搜索理论基础98.所有可达路径(卡码网)1.邻接矩阵法2.邻接表法 思维导图汇总 深度优先搜索理论基础 深度优先搜索&#xff08;dfs&#xff09;与广度优先搜索&#xff08;bfs&#xf…

QSerialPort 串口通信示例

之前使用过MFC写过串口通信的示例&#xff0c;今年学了Qt&#xff0c;特意使用Qt写了串口通信的示例&#xff0c;发现比MFC要容易一些&#xff0c; MFC串口示例如下&#xff1a; Qt示例如下&#xff1a; Qt这个做的很简单&#xff0c;主要还是想验证一下api&#xff0c; 核心…

Python中对象obj类型确定最pythonic的方式——isinstance()函数

python中确定对象obj的类型&#xff0c;isinstance函数最是优雅&#xff0c;type、issubclass等函数也可以&#xff0c;但终究“曲折”。 (笔记模板由python脚本于2024年10月07日 19:42:38创建&#xff0c;本篇笔记适合喜欢python的coder翻阅) 【学习的细节是欢悦的历程】 Pyth…

算法专题四: 前缀和

目录 1. 前缀和2. 二维前缀和3. 寻找数组的中心下标4. 除自身以外数组的乘积5. 和为k的子数组6. 和可被K整除的子数组7. 连续数组8. 矩阵区域和 博客主页:酷酷学!!! 感谢关注~ 1. 前缀和 算法思路: 根据题意, 创建一个前缀和数组, dp[i] dp[i -1] arr[i], 再使用前缀和数组,…

排查和解决JVM OOM实战

JVM OOM介绍 Java内存区域布局 下面的分析中都是基于JDK 8开始的。关于JMM不过多介绍每个区域的作用。OOM不单只会发生在堆内存&#xff0c;也可能是因为元空间或直接内存泄漏导致OOM&#xff0c;此时在OOM的详细信息中会有不同体现。 Java OOM的类别 java.lang.OutOfMemory…

王者农药更新版

一、启动文件配置 二、GPIO使用 2.1基本步骤 1.配置GPIO&#xff0c;所以RCC开启APB2时钟 2.GPIO初始化&#xff08;结构体&#xff09; 3.给GPIO引脚设置高/低电平&#xff08;WriteBit&#xff09; 2.2Led循环点亮&#xff08;GPIO输出&#xff09; 1.RCC开启APB2时钟。…

HarmonyOS/OpenHarmony 自定义弹窗页面级层级控制解决方案

关键词&#xff1a;CuntomDialog自定义弹窗、SubWindow子窗口、页面级、弹窗层级控制、鸿蒙、弹窗展示层级异常 问题存在API版本&#xff1a;API10 - API12&#xff08;该问题已反馈&#xff0c;期望后续官方能增加页面级控制能力&#xff09; 在正常的鸿蒙app开发过程中&…

TIM(Timer)定时器的原理

一、介绍 硬件定时器的工作原理基于时钟信号源提供稳定的时钟信号作为计时器的基准。计数器从预设值开始计数&#xff0c;每当时钟信号到达时计数器递增。当计数器达到预设值时&#xff0c;定时器会触发一个中断信号通知中断控制器处理相应的中断服务程序。在中断服务程序中&a…

LeetCode讲解篇之239. 滑动窗口最大值

文章目录 题目描述题解思路题解代码题目链接 题目描述 题解思路 我们维护一个长度为k的窗口&#xff0c;然后窗口从数组最左边一直移动到最右边&#xff0c;记录过程中窗口中的最大值&#xff0c;就是答案 我们每次查询长度为k的窗口最大值是什么时间复杂度是O(k)的&#xff0…

rust中async/await的使用

在Rust中,async/await 用于编写异步代码。它允许您以同步的方式编写异步代码,使得异步操作更易于理解和编写。 安装依赖: cargo add futures cargo add async-std 使用示例: 示例1: use async_std::task::block_on;fn main() {block_on(hello()); }async fn hello() …

MoveIt2-humble----Planning Around Objects

1 添加Planning Scene Interface头文件 #include <moveit/planning_scene_interface/planning_scene_interface.h>2 改变目标位姿 // Set a target Pose auto const target_pose [] {geometry_msgs::msg::Pose msg;msg.orientation.w 1.0;msg.position.x 0.28;msg.p…

Github 2024-10-06 php开源项目日报 Top10

根据Github Trendings的统计,今日(2024-10-06统计)共有10个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量PHP项目10Blade项目2Laravel:表达力和优雅的 Web 应用程序框架 创建周期:4631 天开发语言:PHP, BladeStar数量:75969 个Fork数量:24281 次…

Linux安装部署MySQL8.0加遇着问题解决

1.首先我先给个URL下载MySQL官方网站https://downloads.mysql.com/archives/community/ 2.选择Linux的红帽系统 3.接着选择红帽系统的7版本,x86 4.接着选择MySQL版本,此时我选择8.4.0,下载rpm bundle这个,下载下面这个就好 5.Windows文件上传到Linux系统 rz上传文件命令,找到…

【unity游戏开发】彻底理解AnimatorStateInfo,获取真实动画长度

前言 前置知识&#xff1a;设置参数后&#xff0c;下一个循环才会切换对应动画&#xff0c;所以在下一个循环获取真实的动画长度 AnimatorStateInfo是结构体&#xff01;值类型&#xff0c;要不断重复获取才是最新的 主要是自动设置trigger切换的动画自动切回上一个动画&#x…

Java中如何实现定时任务?

目录 一、定时任务 概念 作用 二、简单定时任务实现方式 1. Thread线程等待&#xff08;最原始最简单方式&#xff09; 2. 使用java.util.Timer Timer 优缺点分析 3. 使用JDK自带的ScheduledExecutorService schedule和scheduleAtFixedRate的区别 schedule侧重保持间隔…

B 私域模式升级:开源技术助力传统经销体系转型

一、引言 1.1 研究背景 随着市场竞争加剧&#xff0c;传统经销代理体系面临挑战。同时&#xff0c;开源技术发展迅速&#xff0c;为 B 私域升级带来新机遇。在当今数字化时代&#xff0c;企业面临着日益激烈的市场竞争。传统的经销代理体系由于管理效率低下、渠道局限、库存压…

贝锐蒲公英网盘首发,秒建私有云,高速远程访问

虽然公共网盘带来了不少便利&#xff0c;但是大家对隐私泄露和重要数据泄密的担忧也随之增加。如果想要确保数据安全&#xff0c;自建私有云似乎是一条出路&#xff0c;然而面对搭建私有云的复杂步骤&#xff0c;许多人感到力不从心&#xff0c;NAS设备的成本也往往让人望而却步…

项目——超级马里奥——Day(2)

争取今天晚上能搞一半啊&#xff0c;啊啊啊啊&#xff0c;感觉事多的忙不过来 设计思路&#xff1a; 1&#xff09;创建并完成常量类 ------->一张图片的情况 先完成对图片的封装------>把图片加载一遍 &#xff08;老实说&#xff0c;我也不太知道为什么&#xff0…

【项目安全设计】软件系统安全设计规范和标准(doc原件)

1.1安全建设原则 1.2 安全管理体系 1.3 安全管理规范 1.4 数据安全保障措施 1.4.1 数据库安全保障 1.4.2 操作系统安全保障 1.4.3 病毒防治 1.5安全保障措施 1.5.1实名认证保障 1.5.2 接口安全保障 1.5.3 加密传输保障 1.5.4终端安全保障 资料获取&#xff1a;私信或者进主页。…

约数个数约数之和

好久没发文章了.......不过粉丝还是一个没少...... 今天来看两道超级恶心的数论题目&#xff01; No.1 约数个数 No.2 约数之和 先来看第一道&#xff1a;约数个数 题目描述 给定 n 个正整数 ai​,请你输出这些数的乘积的约数个数,答案对 10^97 取模 输入格式 第一行包含…