LeetCode 372
题目:
Your task is to calculate a^b mod 1337 where a is a positive integer and b is an extremely large positive integer given in the form of an array.
求a的b次方mod 1337,其中b是一个extremely large的数,以至于需要用整数数组来存储。
引理:(a × b)mod M = ((a mod M) * (b mod M)) mod M
证明:
若 a = x·M + m, b = y·M + n (a × b) mod M = (xyMM + myM + nxM + mn) mod M 其中 xyMM, myM, nxM 均能被M整除,故 (a × b) mod M = mn mod M
解题思路:
百度来基本解题思路均为二分,将b作为数字处理。相关链接:http://blog.csdn.net/mebiuw/article/details/51853673
可能我的思路比较特殊,当然方法也比较复杂,但是理论上复杂度应该更低。
举个例子,若b是一个5位数,b = A*1000 + B*1000 + C*100 + D*10 + E
注意到,a^b = a^(A*10000 + B*1000 + C*100 + D*10 + E) = (a^10000)^A + (a^1000)^B + (a^100)^C + (a^10)^D + (a^1)^E
可以利用一个与b等长的数组把a^(10^n)存储下来,之后每次可以直接拿过来用。因为 a^n mod M = ((a^(n/2) mod M) * (a^(n/2) mod M)) mod M, 所以这里可以使用上述引理,执行二分策略。
而对于每一个 a^(10^i)^bi, 又可以看做 ai ^ bi, 这里依旧可以利用引理进行二分。
AC代码如下:
public final int MOD = 1337;
public int modProd(int x, int y){
return ((x%MOD)*(y%MOD))%MOD;
}
public int pow(int a, int b){
// return (a^b)%MOD
if (b == 0) return 1;
if (b == 1) return a%MOD;
int m = pow(a, b/2);
if (b%2 == 1){
return modProd(a, m*m);
}else{
return modProd(m, m);
}
}
public int superPow(int a, int[] b) {
int[] mpow = new int[b.length];
int i, j;
for (i = 0, j = b.length-1; i < j; i++, j--){
// b 中高低位交换,便于后续编码
b[i] = b[i] ^ b[j];
b[j] = b[i] ^ b[j];
b[i] = b[i] ^ b[j];
}
mpow[0] = a%MOD;
for (i = 1; i< b.length; i++){
// 预处理,存储 (a^(10^i)) mod M
// 预处理过程中,对于 a^(10^i)看做 a^(10^(i-1)) ^ 10, 这样每次只需要运行 log 10 次
mpow[i] = pow(mpow[i-1], 10);
}
int res = 1;
for (i = 0; i< b.length; i++){
// 对每一位 bi 计算 a^(bi*(10^i))
res = modProd(res, pow(mpow[i], b[i]));
} return res;
}
预处理阶段,计算 a^(10^i) mod M, 由于每次是提取前一次的结果作为基数,故每次需要4次,预处理的时间复杂度为 4·|b|,其中|b|为字符串b的长度。
运行阶段,每次运行也是最多4次,故其复杂度为 4·|b|。
整个程序运行复杂度为 O(8·|b|)
而对于其他解法,其每次二分均需要对b进行一次处理,故其复杂度为 O(|b|×log(Valueof(b))),其中Valueof(b)表示数组b所代表的值,它将远远大于 |b|。
LeetCode 372的更多相关文章
- C#版(击败100.00%的提交) - Leetcode 372. 超级次方 - 题解
版权声明: 本文为博主Bravo Yeung(知乎UserName同名)的原创文章,欲转载请先私信获博主允许,转载时请附上网址 http://blog.csdn.net/lzuacm. Leetcod ...
- LeetCode——372. Super Pow
题目链接:https://leetcode.com/problems/super-pow/description/ Your task is to calculate ab mod 1337 wher ...
- Java实现 LeetCode 372 超级次方
372. 超级次方 你的任务是计算 ab 对 1337 取模,a 是一个正整数,b 是一个非常大的正整数且会以数组形式给出. 示例 1: 输入: a = 2, b = [3] 输出: 8 示例 2: ...
- Leetcode 372. Super Pow
使用公式 c = ab => c mod d = [a mod d * b mod d] mod d 所以a^423 mod d = (a^100)^4 * (a ^10)^2 * a^3 ...
- Leetcode 372.超级次方
超级次方 你的任务是计算 ab 对 1337 取模,a 是一个正整数,b 是一个非常大的正整数且会以数组形式给出. 示例 1: 输入: a = 2, b = [3] 输出: 8 示例 2: 输入: a ...
- leetcode & lintcode for bug-free
刷题备忘录,for bug-free leetcode 396. Rotate Function 题意: Given an array of integers A and let n to be it ...
- leetcode & lintcode 题解
刷题备忘录,for bug-free 招行面试题--求无序数组最长连续序列的长度,这里连续指的是值连续--间隔为1,并不是数值的位置连续 问题: 给出一个未排序的整数数组,找出最长的连续元素序列的长度 ...
- [LintCode]——目录
Yet Another Source Code for LintCode Current Status : 232AC / 289ALL in Language C++, Up to date (20 ...
- leetcode 50. Pow(x, n) 、372. Super Pow
50. Pow(x, n) 372. Super Pow https://www.cnblogs.com/grandyang/p/5651982.html https://www.jianshu.co ...
随机推荐
- PyQt 自定义信号带参数
import sys from PyQt5.QtCore import pyqtSignal, QObject from PyQt5.QtWidgets import QMainWindow, QAp ...
- JAVA之IO流(字节流)
输入和输出 JAVA的流分为输入流和输出流两部分, 输入流:InputStream或者Reader:从文件中读到程序中: 输出流:OutputStream或者Writer:从程序中输出到文件中: Re ...
- Notepad++编写Markdown
Markdown语法高亮 下载userDefineLang_markdown.xml 打开Notepad++的 Language 菜单,选中底部的 Define your language... 在 ...
- 重写保存按钮save事件
_saveActionEventHandler:function(event,value){ if (!_self.fireEvent("saveBefore")) return; ...
- <<< 三大框架简短介绍
Struts 搞业务 Spring 主要是AOP(面向方面编程)和IOC(控制反转)它里面用到很多的设计模式Hibernate Orm映射工具 实现面向对象的方式操作数据库hibernate 封装了j ...
- APP开发+发布流程
ios开发(证书+应用发布) 证书获取:http://newdocx.appcan.cn/newdocx/docx?type=1297_1291ios应用发布:http://newdocx.appca ...
- 面试题目——《CC150》位操作
面试题5.1:给定两个32位的整数N与M,以及表示比特位置的i与j.编写一个方法,将M插入N,使得M从N的第j位开始,到第i位结束.假定从j位到i位足以容纳M,也即若M=10011,那么j与i之间至少 ...
- 【09-04】java内部类学习笔记
java中的内部类 静态内部类 成员内部类 方法内部类 匿名内部类 1.静态内部类 class Outer { private static String outer = "outer&qu ...
- 万恶的jar包冲突
搭了个spring+struts2+mybatis的项目架子, 好久不用myEclipse和tomcat了,生疏了好多. 建议还是去百度一些框架整合的博客,直接使用博客里面给的jar包列表里的jar包 ...
- java读取项目根路径下和任意磁盘位置下的properties文件
1.读取项目根路径下的properties文件比较简单也是比较常见的一种操作. 具体代码如下: package com.xuanen.util; import java.util.Properties ...