# 快速冪
假設要算3^11^
先把11分解成二進制
11 = 1011(2)
11 = 2^3^×1 + 2^2^×0 + 2^1^×1 +2^0^×1
11 = 8 + 2 + 1
所以我們原本要求的3^11^就變成3^(8+2+1)^
3^(8+2+1)^ = 3^8^ × 3^2^ × 3^1^
那藉由二進制的升冪可以發現,每當次方數增加1就會多乘2
2^0^-> 2^1^-> 2^2^-> 2^3^-> 2^4^
1 -> 2 -> 4 -> 8 -> 16
因此,帶回去次方的位置,指數乘2會等於整個數平方
3^2^ -> 3^4^ = (3^2^)^2^
所以就可以得到下數列
3^1^ -(平方)-> 3^2^ -(平方)-> 3^4^ -(平方)-> 3^8^ -(平方)-> 3^16^
#### 結論(1) : 利用自己乘自己得到以二進制為次方的數列
觀察二進制1011
2^3^×1 + 2^2^×0 + 2^1^×1 +2^0^×1
當是0的時候,任何數的0次方都是1,所以可以略過
當是1的時候次方才有意義
帶回底數
3^11^ = (3^2的3次方×1^) × (3^2的2次方×0^) × (3^2的1次方×1^) × (3^2的0次方×1^)
#### 結論(2) : 利用次方的二進制是0或1判斷此數該不該乘到答案裡
```cpp=
int powf(int a,int b){
int ans=1; //答案
while(b>0){
if(b%2==1){ //結論(2)
ans*=a; //乘到答案裡
}
b/=2; //左移一個
a*=a; //結論(1)
}
return ans;
}
```
bitswise + mod 精簡版
```cpp=
#define m 10000007
int powf(int a,int b){
int ans=1;
while(b){
if(b & 1) ans=(ans*a)%m;
a=(a*a)%m;
b>>=1;
}
return ans;
}
```
## 大數取模
當指數太大超過long long範圍時,要用string讀
求(a^b^)%m 假設a=123
a=((0*10+1)*10+2)*10+3
取模就是:
a=((((((0*10+1)%m)*10+2)%m)*10+3)%m
```cpp=
int a=0;
for(int i=0;i<str.size();i++){
a*=10;
a+=str[i]-'0';
a%=m;
}
```
## 無理數快速冪
注意:ans要先放一個(a,b)進去
<a href="https://judge.tcirc.tw/ShowProblem?problemid=d022">無理數的快速冪</a>
```cpp=
#include <bits/stdc++.h>
#define int long long
#define pii pair<int,int>
#define x first
#define y second
#define m 1000000009
using namespace std;
int a,b,n;
pii mul(pii i,pii j){
pii ans={0,0};
ans.x+=i.x*j.x;
ans.y+=i.x*j.y;
ans.y+=i.y*j.x;
ans.x+=2*i.y*j.y;
ans.x%=m;
ans.y%=m;
return ans;
}
void solve(){
pii ans={a,b};
pii A={a,b};
n--;
while(n){
if(n & 1) ans=mul(ans,A);
A=mul(A,A);
n>>=1;
}
cout << ans.x << ' ' << ans.y << '\n';
}
signed main(){
cin >> a >> b >> n;
solve();
return 0;
}
```