NNT

转载自:http://blog.csdn.net/acdreamers/article/details/39026505

今天,我将来介绍另一种计算多项式乘法的算法,叫做快速数论变换(NTT),在离散正交变换的理论中,已经证明在

复数域内,具有循环卷积特性的唯一变换是DFT,所以在复数域中不存在具有循环卷积性质的更简单的离散正交变换。

因此提出了以数论为基础的具有循环卷积性质的快速数论变换

 

回忆复数向量,其离散傅里叶变换公式如下

 

  

 

离散傅里叶逆变换公式为

 

   

 

今天的快速数论变换(NTT)是在上进行的,在快速傅里叶变换(FFT)中,通过次单位复根来运算的,即满

,而对于快速数论变换来说,则是可以将看成是的等价,这里是模素数

的原根(由于是素数,那么原根一定存在)。即

 

       

 

所以综上,我们得到数论变换的公式如下

 

   

 

数论变换的逆变换公式为

 

    

 

这样就把复数对应到一个整数,之后一切都是在系统内考虑。

 

上述数论变换(NTT)公式中,要求是素数且必须是的因子。由于经常是2的方幂,所以可以构造形

的素数。通常来说可以选择费马素数,这样的变换叫做费马数数论变换

 

这里我们选择,这样得到模的原根值为

 

 

题目:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1028

 

分析:题目意思就是大数相乘,此处用快速数论变换(NTT)实现。

 

代码:

  1. #include <iostream>  
  2. #include <string.h>  
  3. #include <stdio.h>  
  4.   
  5. using namespace std;  
  6. typedef long long LL;  
  7.   
  8. const int N = 1 << 18;  
  9. const int P = (479 << 21) + 1;  
  10. const int G = 3;  
  11. const int NUM = 20;  
  12.   
  13. LL  wn[NUM];  
  14. LL  a[N], b[N];  
  15. char A[N], B[N];  
  16.   
  17. LL quick_mod(LL a, LL b, LL m)  
  18. {  
  19.     LL ans = 1;  
  20.     a %= m;  
  21.     while(b)  
  22.     {  
  23.         if(b & 1)  
  24.         {  
  25.             ans = ans * a % m;  
  26.             b--;  
  27.         }  
  28.         b >>= 1;  
  29.         a = a * a % m;  
  30.     }  
  31.     return ans;  
  32. }  
  33.   
  34. void GetWn()  
  35. {  
  36.     for(int i=0; i<NUM; i++)  
  37.     {  
  38.         int t = 1 << i;  
  39.         wn[i] = quick_mod(G, (P - 1) / t, P);  
  40.     }  
  41. }  
  42.   
  43. void Prepare(char A[], char B[], LL a[], LL b[], int &len)  
  44. {  
  45.     len = 1;  
  46.     int len_A = strlen(A);  
  47.     int len_B = strlen(B);  
  48.     while(len <= 2 * len_A || len <= 2 * len_B) len <<= 1;  
  49.     for(int i=0; i<len_A; i++)  
  50.         A[len - 1 - i] = A[len_A - 1 - i];  
  51.     for(int i=0; i<len - len_A; i++)  
  52.         A[i] = '0';  
  53.     for(int i=0; i<len_B; i++)  
  54.         B[len - 1 - i] = B[len_B - 1 - i];  
  55.     for(int i=0; i<len - len_B; i++)  
  56.         B[i] = '0';  
  57.     for(int i=0; i<len; i++)  
  58.         a[len - 1 - i] = A[i] - '0';  
  59.     for(int i=0; i<len; i++)  
  60.         b[len - 1 - i] = B[i] - '0';  
  61. }  
  62.   
  63. void Rader(LL a[], int len)  
  64. {  
  65.     int j = len >> 1;  
  66.     for(int i=1; i<len-1; i++)  
  67.     {  
  68.         if(i < j) swap(a[i], a[j]);  
  69.         int k = len >> 1;  
  70.         while(j >= k)  
  71.         {  
  72.             j -= k;  
  73.             k >>= 1;  
  74.         }  
  75.         if(j < k) j += k;  
  76.     }  
  77. }  
  78.   
  79. void NTT(LL a[], int len, int on)  
  80. {  
  81.     Rader(a, len);  
  82.     int id = 0;  
  83.     for(int h = 2; h <= len; h <<= 1)  
  84.     {  
  85.         id++;  
  86.         for(int j = 0; j < len; j += h)  
  87.         {  
  88.             LL w = 1;  
  89.             for(int k = j; k < j + h / 2; k++)  
  90.             {  
  91.                 LL u = a[k] % P;  
  92.                 LL t = w * (a[k + h / 2] % P) % P;  
  93.                 a[k] = (u + t) % P;  
  94.                 a[k + h / 2] = ((u - t) % P + P) % P;  
  95.                 w = w * wn[id] % P;  
  96.             }  
  97.         }  
  98.     }  
  99.     if(on == -1)  
  100.     {  
  101.         for(int i = 1; i < len / 2; i++)  
  102.             swap(a[i], a[len - i]);  
  103.         LL Inv = quick_mod(len, P - 2, P);  
  104.         for(int i = 0; i < len; i++)  
  105.             a[i] = a[i] % P * Inv % P;  
  106.     }  
  107. }  
  108.   
  109. void Conv(LL a[], LL b[], int n)  
  110. {  
  111.     NTT(a, n, 1);  
  112.     NTT(b, n, 1);  
  113.     for(int i = 0; i < n; i++)  
  114.         a[i] = a[i] * b[i] % P;  
  115.     NTT(a, n, -1);  
  116. }  
  117.   
  118. void Transfer(LL a[], int n)  
  119. {  
  120.     int t = 0;  
  121.     for(int i = 0; i < n; i++)  
  122.     {  
  123.         a[i] += t;  
  124.         if(a[i] > 9)  
  125.         {  
  126.             t = a[i] / 10;  
  127.             a[i] %= 10;  
  128.         }  
  129.         else t = 0;  
  130.     }  
  131. }  
  132.   
  133. void Print(LL a[], int n)  
  134. {  
  135.     bool flag = 1;  
  136.     for(int i = n - 1; i >= 0; i--)  
  137.     {  
  138.         if(a[i] != 0 && flag)  
  139.         {  
  140.             printf("%d", a[i]);  
  141.             flag = 0;  
  142.         }  
  143.         else if(!flag)  
  144.             printf("%d", a[i]);  
  145.     }  
  146.     puts("");  
  147. }  
  148.   
  149. int main()  
  150. {  
  151.     GetWn();  
  152.     while(scanf("%s%s", A, B)!=EOF)  
  153.     {  
  154.         int len;  
  155.         Prepare(A, B, a, b, len);  
  156.         Conv(a, b, len);  
  157.         Transfer(a, len);  
  158.         Print(a, len);  
  159.     }  
  160.     return 0;  


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
程序完全免费版,并带有正版升级ID 3000个!官方提供的。<br><br>本系统由飞跃网络倾力打造,其主要宗旨是让公司管理更轻松,让每个员工都参与,从而提高公司的办公效率。<br>NNT公司管理系统是基于网络技术构建,与以往的局域网OA程序完全不同,她是完全基于互联网的一套软件,<br>因此不怕病毒感染,系统崩溃与文件丢失。并且能在全球范围内随时随地的完成信息共享。<br>其方便快捷的效率,使工作更加高效,使管理更加轻松,只要有网络,一切都可以轻松完成。<br>她可以让一个公司的业务和资源结合得更加紧密,因而可以将诸如信息录入,查询统计等功能与具体业务密切关联。<br>操作人员只需点击一个按钮就可以得到想要的员工信息,客户信息与业务信息。<br><br>###################################################################### <br>NNT公司管理系统 免费版 v2008030415<br><br><br>开发者:飞跃网络<br><br>客户服务:QQ 320174239 技术服务: QQ 320174234<br><br>官方网站: oa.nnt.cn <br><br>#######################################################################<br>#######################################################################<br><br>更多使用说明请登陆官方页面帮助中心 http://oa.nnt.cn<br><br>管理地址: 登陆页面登陆<br>管理员: boobuy<br>密码: 123456<br><br>NNT官方使用帮助中心:<br><br>http://oa.nnt.cn/help/list.asp?wlei=nntoa<br><br>请大家第一次使用时,一定要先去看帮助中心里的帮助,<br>由于帮助信息不断更新,<br>系统包内没有很详细的帮助。 <br>=======================================================================<br>客户中心<br>快速添加查找所需要的目标客户,并且拥有独特的潜在客户库,对于业务拓展有绝对性的帮助!<br><br>职员中心<br>列出了所有在职的职员信息,方便管理层查看,备注以及员工相互交流,当职员离职时他的信息将转到历史职员库,可供以后查询联系!<br>并且拥有公司人才库,为公司储备足够的可用人才,在需要时方便转为公司新力量.<br><br><br>业务中心<br>提交各种不同类型的新业务,不在需要纸张,方便统计查阅.并且列出本月所提交的业务,并可以完成审核流程.并可查询以前的每一项业务!<br><br><br>系统升级中心<br>完全自主开发的自动升级接口,让您的系统实现全自动的智能升级!保持系统随时更新到最安全,最优秀的版本!<br><br><br>公司U盘 个人U盘<br>将公司资料上传到公司U盘和个人U盘,可以随时随地的上传与下载,让工作更加轻松便捷!更可以让员工资源共享<br><br>公司部门<br>无限的公司部门及职位设置,给每一个职位分配不同的权限有业务类型,满足不同职员的管理权限及业务需求!让公司管理更轻松,让每个职员都参与!<br><br>首页快捷操作面板<br>每个职员都可以按自己的喜好,定义自己的常用快捷按钮!让系统操作起来更加随心所欲!<br><br>更多的功能,请您在使用中体验!!!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值