2015 ICL, Finals, Div. 1 Ceizenpok’s formula && Gym - 100633J

上来先感谢大佬的文章:https://blog.youkuaiyun.com/clove_unique/article/details/54571216

题目链接:http://codeforces.com/gym/100633/problem/J

题意:

Ckn%p C n k % p
的值,其中 1n1018 1 ≤ n ≤ 10 18 , 0kn 0 ≤ k ≤ n , 2p1000000 2 ≤ p ≤ 1000000 .

分析:

若p为质数,可以用lucas定理直接得,但题目中p不一定为质数,则需要用拓展lucas定理+中国剩余定理。

结论:

拓展lucas定理:

可求方程:

ansc(modpk11) a n s ≡ c ( m o d p 1 k 1 )

中国剩余定理(又名孙子定理)

可求

ansc1(modm1)ansc2(modm2)....anscq(modmq) { a n s ≡ c 1 ( m o d m 1 ) a n s ≡ c 2 ( m o d m 2 ) . . . . a n s ≡ c q ( m o d m q )

的最小正整数解,解为
(i=1qciMmiinv(Mmi,mi))%M ( ∑ i = 1 q ∗ c i ∗ M m i ∗ i n v ( M m i , m i ) ) % M
其中
M=1qmi M = ∏ 1 q m i

题解:

对于p,将它表示成

p=pk11pk22...pkqq p = p 1 k 1 ∗ p 2 k 2 . . . p q k q
的形式,对于每个 pi p i ,用lucas定理求解,再用中国剩余定理求和。

唯一的问题在于在求

n!(nm)!m!%pkii n ! ( n − m ) ! m ! % p i k i
的时候,由于分母与 pkii p i k i 不一定互质,没法求逆元,所以需要先把分子分母中 pi p i 全部除去,最后再乘回来。
代码实现:

long long x=0,ans;
for(long long i=n;i;i/=pi) x+=i/pi;
for(long long i=k;i;i/=pi) x-=i/pi;
for(long long i=n-k;i;i/=pi) x-=i/pi;
ans=a*mod_inverser(b,pk)%pk*mod_inverser(c,pk)*pow_mod(pi,x,pk)%pk;

解释一下 ipi i p i 的含义:
以求 n! n ! 为例:假设 n=100,pi=2 n = 100 , p i = 2
100!=1009998...1 100 ! = 100 ∗ 99 ∗ 98 ∗ . . . ∗ 1
=(999795...1)250(123...50) = ( 99 ∗ 97 ∗ 95 ∗ . . . ∗ 1 ) ∗ 2 50 ∗ ( 1 ∗ 2 ∗ 3 ∗ . . . ∗ 50 )
=(999795...1)250(13...49)225(12...25) = ( 99 ∗ 97 ∗ 95 ∗ . . . ∗ 1 ) ∗ 2 50 ∗ ( 1 ∗ 3 ∗ . . . ∗ 49 ) ∗ 2 25 ∗ ( 1 ∗ 2 ∗ . . . ∗ 25 )
……
所以,一直除 pi p i 可将 n! n ! 表示为

pxia p i x ∗ a
a为其余的项的积。通过这个方法可求得 n!(nm)!m!%pkii n ! ( n − m ) ! m ! % p i k i

代码:

#include <iostream>
#include <bits/stdc++.h>
using namespace std;

long long exgcd(long long a,long long b,long long &x,long long &y)
{
    long long d=a;
    if(b!=0)
    {
        d=exgcd(b,a%b,y,x);
        y-=(a/b)*x;
    }else{
        x=1;
        y=0;
    }
    return d;
}
long long pow_mod(long long a,long long b,long long p)
{
    long long res=1;
    while(b)
    {
        if(b&1) res=res*a%p;
        a=a*a%p;
        b>>=1;
    }
    return res;
}
long long mod_inverser(long long a,long long m)
{
    long long x,y;
    exgcd(a,m,x,y);
    return (m+x%m)%m;
}
long long cal(long long n,long long pi,long long pk)
{
    if(n==0) return 1;
    long long ans=1;
    if(n/pk)
    {
         for(long long i=2;i<=pk;i++)
         {
             if(i%pi)
                ans=ans*i%pk;
         }
         ans=pow_mod(ans,n/pk,pk);
    }
    for(long long i=2;i<=n%pk;i++)
    {
        if(i%pi)
            ans=ans*i%pk;
    }
    return ans*cal(n/pi,pi,pk)%pk;
}
long long C(long long n,long long k,long long p,long long pi,long long pk)
{
    if(k>n) return 0;
    long long a=cal(n,pi,pk),b=cal(k,pi,pk),c=cal(n-k,pi,pk);
    long long x=0,ans;
    for(long long i=n;i;i/=pi) x+=i/pi;
    for(long long i=k;i;i/=pi) x-=i/pi;
    for(long long i=n-k;i;i/=pi) x-=i/pi;
    ans=a*mod_inverser(b,pk)%pk*mod_inverser(c,pk)*pow_mod(pi,x,pk)%pk;
    return ans*(p/pk)%p*mod_inverser(p/pk,pk)%p;
}

int main()
{
    long long n,k,p;
    cin>>n>>k>>p;
    long long ans=0;
    for(long long x=p,i=2;i<=p;i++)
    {
        if(x%i==0)
        {
            long long pk=1;
            while(x%i==0)
            {
                pk*=i;
                x/=i;
            }
            ans=(ans+C(n,k,p,i,pk))%p;
        }
    }
    cout<<ans<<endl;
    return 0;
}
欧姆龙FINS(工厂集成网络系统)协议是专为该公司自动化设备间数据交互而设计的网络通信标准。该协议构建于TCP/IP基础之上,允许用户借助常规网络接口执行远程监控、程序编写及信息传输任务。本文档所附的“欧ronFins.zip”压缩包提供了基于C与C++语言开发的FINS协议实现代码库,旨在协助开发人员便捷地建立与欧姆龙可编程逻辑控制器的通信连接。 FINS协议的消息框架由指令头部、地址字段、操作代码及数据区段构成。指令头部用于声明消息类别与长度信息;地址字段明确目标设备所处的网络位置与节点标识;操作代码定义了具体的通信行为,例如数据读取、写入或控制器指令执行;数据区段则承载实际交互的信息内容。 在采用C或C++语言实施FINS协议时,需重点关注以下技术环节: 1. **网络参数设置**:建立与欧姆龙可编程逻辑控制器的通信前,必须获取控制器的网络地址、子网划分参数及路由网关地址,这些配置信息通常记载于设备技术手册或系统设置界面。 2. **通信链路建立**:通过套接字编程技术创建TCP连接至控制器。该过程涉及初始化套接字实例、绑定本地通信端口,并向控制器网络地址发起连接请求。 3. **协议报文构建**:依据操作代码与目标功能构造符合规范的FINS协议数据单元。例如执行输入寄存器读取操作时,需准确配置对应的操作代码与存储器地址参数。 4. **数据格式转换**:协议通信过程中需进行二进制数据的编码与解码处理,包括将控制器的位状态信息或数值参数转换为字节序列进行传输,并在接收端执行逆向解析。 5. **异常状况处理**:完善应对通信过程中可能出现的各类异常情况,包括连接建立失败、响应超时及错误状态码返回等问题的处理机制。 6. **数据传输管理**:运用数据发送与接收函数完成信息交换。需注意FINS协议可能涉及数据包的分割传输与重组机制,因单个协议报文可能被拆分为多个TCP数据段进行传送。 7. **响应信息解析**:接收到控制器返回的数据后,需对FINS响应报文进行结构化解析,以确认操作执行状态并提取有效返回数据。 在代码资源包中,通常包含以下组成部分:展示连接建立与数据读写操作的示范程序;实现协议报文构建、传输接收及解析功能的源代码文件;说明库函数调用方式与接口规范的指导文档;用于验证功能完整性的测试案例。开发人员可通过研究这些材料掌握如何将FINS协议集成至实际项目中,从而实现与欧姆龙可编程逻辑控制器的高效可靠通信。在工程实践中,还需综合考虑网络环境稳定性、通信速率优化及故障恢复机制等要素,以确保整个控制系统的持续可靠运行。 资源来源于网络分享,仅用于学习交流使用,请勿用于商业,如有侵权请联系我删除!
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值