【洛谷P2054洗牌】AC代码(扩展欧几里得+二分快速幂+二分龟速乘)

2024-06-20 20:18

本文主要是介绍【洛谷P2054洗牌】AC代码(扩展欧几里得+二分快速幂+二分龟速乘),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目描述

题目链接
为了表彰小联为Samuel星球的探险所做出的贡献,小联被邀请参加Samuel星球近距离载人探险活动。

由于Samuel星球相当遥远,科学家们要在飞船中度过相当长的一段时间,小联提议用扑克牌打发长途旅行中的无聊时间。玩了几局之后,大家觉得单纯玩扑克牌对于像他们这样的高智商人才来说太简单了。有人提出了扑克牌的一种新的玩法。

对于扑克牌的一次洗牌是这样定义的,将一叠N(N为偶数)张扑克牌平均分成上下两叠,取下面一叠的第一张作为新的一叠的第一张,然后取上面一叠的第一张作为新的一叠的第二张,再取下面一叠的第二张作为新的一叠的第三张……如此交替直到所有的牌取完。

如果对一叠6张的扑克牌1 2 3 4 5 6,进行一次洗牌的过程如下图所示:

从图中可以看出经过一次洗牌,序列1 2 3 4 5 6变为4 1 5 2 6 3。当然,再对得到的序列进行一次洗牌,又会变为2 4 6 1 3 5。

游戏是这样的,如果给定长度为N的一叠扑克牌,并且牌面大小从1开始连续增加到N(不考虑花色),对这样的一叠扑克牌,进行M次洗牌。最先说出经过洗牌后的扑克牌序列中第L张扑克牌的牌面大小是多少的科学家得胜。小联想赢取游戏的胜利,你能帮助他吗?

输入格式

输入文件中有三个用空格间隔的整数,分别表示N,M,L(其中0<N≤10^10 ,0 ≤M≤10^10,且N为偶数)。

输出格式

单行输出指定的扑克牌的牌面大小。

输入输出样例

  • 输入 #1
6 2 3
  • 输出 #1
6
  • 说明/提示
0<N≤10^100≤M≤10^10,且N为偶数

前置知识

扩展欧几里得算法

贝祖定理

  • 若有整数a,b,c,在方程 a x + b y = c ax+by=c ax+by=c中,未知数x,y有整数解当且仅当c是gcd(a,b)的倍数,其中gcd(a,b)表示a,b的最大公因数。

欧几里得算法

  • gcd(a,b)=gcd(b,a%b)=…=gcd(d,0)=d

算法模板

int gcd(int a,int b){if(b==0) return a;else return gcd(b,a%b);
}

扩展欧几里得算法

  • 用于求解 a x + b y = g c d ( a , b ) ax+by=gcd(a,b) ax+by=gcd(a,b)

算法思想
按照欧几里得算法展开方程 a x + b y = g c d ( a , b ) = g c d ( b , a % b ) = b x ′ + ( a % b ) y ′ = . . . = 1 d + 0 y = g c d ( a , b ) ax+by=gcd(a,b)=gcd(b,a\%b)=bx'+(a\%b)y'=...=1d+0y=gcd(a,b) ax+by=gcd(a,b)=gcd(b,a%b)=bx+(a%b)y=...=1d+0y=gcd(a,b)
其中 b x ′ + ( a % b ) y ′ = b x ′ + ( a − a / b ∗ b ) y ′ = a y ′ + ( x ′ − a / b ∗ y ′ ) b = a x + b y bx'+(a\%b)y'=bx'+(a-a/b*b)y'=ay'+(x'-a/b*y')b=ax+by bx+(a%b)y=bx+(aa/bb)y=ay+(xa/by)b=ax+by
x = x ′ , y = x ′ − a / b ∗ y ′ x=x',y=x'-a/b*y' x=x,y=xa/by
最终 g c d ( a , b ) x 0 + 0 y 0 = g c d ( a , b ) gcd(a,b)x_0+0y_0=gcd(a,b) gcd(a,b)x0+0y0=gcd(a,b),即 x 0 = 1 , y 0 ∈ R x_0=1,y_0\in R x0=1,y0R,所求得 x , y x,y x,y只是一组特解,通解为 ( x + k ∗ b / d , y − k ∗ b / d ) (x+k*b/d,y-k*b/d) (x+kb/d,ykb/d),推导方法略

算法模板

typedef long long ll;
ll exgcd(ll a,ll b,ll &x,ll &y){if(b==0){x=1;//gcd(a,b)x=gcd(a,b)y=0;//y随意return a;}ll gcd=exged(b,a%b,x,y),r;r=x;x=y;//x=y'y=r-a/b*y;//y=x'-a/b*y'return gcd;
}

用途

  1. 求乘法逆元
  2. 求解 a x + b y = c ax+by=c ax+by=c
  3. 求同余方程

乘法单位元:任何数乘以单位元等于这个数,显然乘法的单位元是1。
乘法逆元:任何数乘以他的乘法逆元等于单位元,比如2*0.5=1,0.5就是1的乘法逆元,但这不是我们需要的东西,我们需要的是在模运算下的整数乘法逆元。

二分快速幂与二分龟速乘

二分快速幂

二分幂
  • 二分幂,就是平常所用的幂运算化简方法,一般采用递归实现,不建议使用


a b = { a ∗ ( a 2 ) b 2 , b 为奇数 ( a 2 ) b 2 , b 为偶数 a^b=\left\{ \begin{array}{l} a*\left( a^2 \right) ^{\frac{b}{2}},b\text{为奇数}\\ \left( a^2 \right) ^{\frac{b}{2}},b\text{为偶数}\\ \end{array} \right. ab={a(a2)2b,b为奇数(a2)2b,b为偶数
转化为递归方程为
f ( a , b ) = { 1 , b = 0 a ∗ f ( a 2 , b 2 ) , b 为奇数 f ( a 2 , b 2 ) , b 为偶数 f\left( a,b \right) =\left\{ \begin{array}{l} 1,b=0\\ a*f\left( a^2,\frac{b}{2} \right) ,b\text{为奇数}\\ f\left( a^2,\frac{b}{2} \right) ,b\text{为偶数}\\ \end{array} \right. f(a,b)=1,b=0af(a2,2b),b为奇数f(a2,2b),b为偶数
算法模板

typedef long long ll;
ll efm(ll a,ll b,ll n){if(b==0) return 1;if(b&1) return a*efm(a*a%n,b>>1,n);else return efm(a*a%n,b>>1,n);
}
快速幂
  • 采用二分的思想利用二进制快速求解 a b a^b ab

算法思想
这里用b表示二进制数,
3 13 = 3 b 1101 = 3 1 ∗ b 1000 ∗ 3 1 ∗ b 100 ∗ 3 0 ∗ b 10 ∗ 3 1 ∗ b 1 3^{13}=3^{b1101}=3^{1*b1000}*3^{1*b100}*3^{0*b10}*3^{1*b1} 313=3b1101=31b100031b10030b1031b1
3 b 10 = ( 3 b 1 ) 2 3^{b10}=(3^{b1})^2 3b10=(3b1)2
3 b 100 = ( 3 b 10 ) 2 3^{b100}=(3^{b10})^2 3b100=(3b10)2
3 b 1000 = ( 3 b 100 ) 2 3^{b1000}=(3^{b100})^2 3b1000=(3b100)2
可以看出,每一项都是前一项的平方,原本需要计算13次的算法被优化到了计算4次,本算法时间复杂度是 O ( l o g 2 b ) O(log_2b) O(log2b)

算法模板

  • 通常,本算法的幂非常大,所求结果常需要取模
typedef long long ll;
ll ksm(ll a,ll b,ll n){//a^b%nll ret=1;//累乘结果计算while(b>0){//非0次幂则计算if(b&1){//判断最低为是否为1,0不需要乘入ret=ret*a%n;}a=a*a%n;//由上述推导可得平方关系b>>=1;//b右移一位}return ret;
}

二分龟速乘

  • 对于快速幂算法的改进

二分快速幂算法存在的问题
在使用二分快速幂计算乘法时,尽管采用了%n来防止溢出,但仍然会有溢出现象,因为x*x%n,在x*x时就有可能溢出。

二分乘
  • 其实就是龟速乘的二分版本,一般用递归实现,不建议使用

乘法可以写成累加的形式,诸如
3 ∗ 5 = 3 + 3 ∗ 4 = 3 + ( 2 ∗ 3 ) ∗ 2 = 3 + 6 ∗ 2 = 3 + 12 3*5=3+3*4=3+(2*3)*2=3+6*2=3+12 35=3+34=3+(23)2=3+62=3+12
转化为递归方程为
f ( a , b ) = { 0 , b = 0 a + f ( 2 a , b 2 ) , b 为奇数 f ( 2 a , b 2 ) , b 为偶数 f\left( a,b \right) =\left\{ \begin{array}{l} 0,b=0\\ a+f\left( 2a,\frac{b}{2} \right) ,b\text{为奇数}\\ f\left( 2a,\frac{b}{2} \right) ,b\text{为偶数}\\ \end{array} \right. f(a,b)=0,b=0a+f(2a,2b),b为奇数f(2a,2b),b为偶数
算法模板

typedef long long ll;
ll gsc(ll a,ll b,ll n){if(b==0) return 0;if(b&1) return (a+gsc((a<<1)%n,b>>1))%n;else return gsc((a<<1)%n,b>>1)%n;
}
龟速乘

我们可以让x*y也变成类似于快速幂的运算形式,诸如
3 ∗ 5 = 3 ∗ b 101 = 3 ∗ ( 1 ∗ b 1 + 0 ∗ b 10 + 1 ∗ b 100 ) 3*5=3*b101=3*(1*b1+0*b10+1*b100) 35=3b101=3(1b1+0b10+1b100)
= 1 ∗ 3 ∗ b 1 + 0 ∗ 3 ∗ b 10 + 1 ∗ 3 ∗ b 100 =1*3*b1+0*3*b10+1*3*b100 =13b1+03b10+13b100
其中
3 ∗ b 10 = 3 ∗ b 1 + 3 ∗ b 1 3*b10=3*b1+3*b1 3b10=3b1+3b1
3 ∗ b 100 = 3 ∗ b 10 + 3 ∗ b 10 3*b100=3*b10+3*b10 3b100=3b10+3b10
不难发现,每一项都是前一项的二倍,由于本算法甚至慢于for循环相加,故得名龟速乘,时间复杂度为 O ( l o g 2 b ) O(log_2b) O(log2b)

算法模板

typedef long long ll;
ll gsc(ll a,ll b,ll n){ll ret=0;//累加结果计算while(b>0){//非0乘则计算if(b&1){//判断最低为是否为1,0不需要加入ret=(ret+a)%n;}a=(a+a)%n;//由上述推导可得平方关系b>>=1;//b右移一位}return ret;
}

快速幂的改进

  • 引入了龟速乘后,我们便可以改进快速幂算法
typedef long long ll;
ll ksm(ll a,ll b,ll n){ll ret=1;while(b>0){if(b&1){ret=gsc(ret,a,n);//修改位}a=gsc(a,a,n);//修改位b>>=1;}return ret;
}

题解

根据题目不难看出在 n n n张牌中第 x x x张牌经过 m m m伦洗牌后与所在位置 l l l满足:
x ∗ 2 m = l ( m o d n + 1 ) x*2^m=l(mod\ n+1) x2m=l(mod n+1)
两种解法:

  1. 通过同余方程求逆元解答
  2. 直接求同余方程解

由于本文旨在学习更多的知识,采用逆元解法
这是线性同余方程,需要快速幂结合扩展欧几里得算法求解。
由于 a x = b ( m o d n ) ax=b(mod\ n) ax=b(mod n),令 d = g c d ( a , n ) , t = n / d d=gcd(a,n),t=n/d d=gcd(a,n),t=n/d,则 x x x的最小正整数解为 x = ( x % t + t ) % t x=(x\%t+t)\%t x=(x%t+t)%t,在本题中 a a a为偶数, n + 1 n+1 n+1为奇数,则 t = n t=n t=n

快速幂+龟速乘代码

#include <iostream>
using namespace std;
typedef long long ll;
ll n,m,l,x,y;
ll gsc(ll x,ll m){ll ret=0;while(m){if(m&1) ret=(ret+x)%n;x=(x+x)%n;m>>=1;}return ret;
}
ll ksm(ll x,ll m){ll ret=1;while(m){if(m&1) ret=gsc(ret,x);x=gsc(x,x);m>>=1;}return ret;
}
ll exgcd(ll a,ll b,ll &x,ll &y){if(b==0){x=1;y=0;return a;}ll r=exgcd(b,a%b,x,y);ll c=x;x=y;y=c-a/b*y;return r;
}
int main(){scanf("%lld%lld%lld",&n,&m,&l);n++;ll k=ksm(2,m);exgcd(k,n,x,y);x=gsc(l,x%n+n);printf("%lld",x%n);return 0;
}

二分幂+二分乘代码

#include <iostream>
using namespace std;
typedef long long ll;
ll n,m,l,x,y;
ll efc(ll x,ll m){if(m==0) return 0;if(m&1) return (x+efc((x<<1)%n,m>>1))%n;else return efc((x<<1)%n,m>>1)%n;
}
ll efm(ll x,ll m){if(m==0) return 1;if(m&1) return efc(x,efm(efc(x,x)%n,m>>1));else return efm(efc(x,x)%n,m>>1);
}
ll exgcd(ll a,ll b,ll &x,ll &y){if(b==0){x=1;y=0;return a;}ll r=exgcd(b,a%b,x,y);ll c=x;x=y;y=c-a/b*y;return r;
}
int main(){scanf("%lld%lld%lld",&n,&m,&l);n++;ll k=efm(2,m);exgcd(k,n,x,y);x=efc(l,x%n+n);printf("%lld",x%n);return 0;
}

扩展

快速乘

  • 经Cyan_rose介绍,《论程序底层优化的一些方法与技巧》这篇论文中提到了快速乘

  • 可以按此代码实现快速乘
cin>>a>>b>>mod;
cout<<((a*b-(long long)((long double)a*b/mod)*mod+mod)%mod);

这篇关于【洛谷P2054洗牌】AC代码(扩展欧几里得+二分快速幂+二分龟速乘)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1079147

相关文章

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

使用Python快速实现链接转word文档

《使用Python快速实现链接转word文档》这篇文章主要为大家详细介绍了如何使用Python快速实现链接转word文档功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 演示代码展示from newspaper import Articlefrom docx import

Python中顺序结构和循环结构示例代码

《Python中顺序结构和循环结构示例代码》:本文主要介绍Python中的条件语句和循环语句,条件语句用于根据条件执行不同的代码块,循环语句用于重复执行一段代码,文章还详细说明了range函数的使... 目录一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

CSS3中使用flex和grid实现等高元素布局的示例代码

《CSS3中使用flex和grid实现等高元素布局的示例代码》:本文主要介绍了使用CSS3中的Flexbox和Grid布局实现等高元素布局的方法,通过简单的两列实现、每行放置3列以及全部代码的展示,展示了这两种布局方式的实现细节和效果,详细内容请阅读本文,希望能对你有所帮助... 过往的实现方法是使用浮动加

JAVA调用Deepseek的api完成基本对话简单代码示例

《JAVA调用Deepseek的api完成基本对话简单代码示例》:本文主要介绍JAVA调用Deepseek的api完成基本对话的相关资料,文中详细讲解了如何获取DeepSeekAPI密钥、添加H... 获取API密钥首先,从DeepSeek平台获取API密钥,用于身份验证。添加HTTP客户端依赖使用Jav

Java实现状态模式的示例代码

《Java实现状态模式的示例代码》状态模式是一种行为型设计模式,允许对象根据其内部状态改变行为,本文主要介绍了Java实现状态模式的示例代码,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来... 目录一、简介1、定义2、状态模式的结构二、Java实现案例1、电灯开关状态案例2、番茄工作法状态案例

nginx-rtmp-module模块实现视频点播的示例代码

《nginx-rtmp-module模块实现视频点播的示例代码》本文主要介绍了nginx-rtmp-module模块实现视频点播,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录预置条件Nginx点播基本配置点播远程文件指定多个播放位置参考预置条件配置点播服务器 192.

CSS自定义浏览器滚动条样式完整代码

《CSS自定义浏览器滚动条样式完整代码》:本文主要介绍了如何使用CSS自定义浏览器滚动条的样式,包括隐藏滚动条的角落、设置滚动条的基本样式、轨道样式和滑块样式,并提供了完整的CSS代码示例,通过这些技巧,你可以为你的网站添加个性化的滚动条样式,从而提升用户体验,详细内容请阅读本文,希望能对你有所帮助...