记录编号 310691 评测结果 AAAAAAAAAA
题目名称 [HAOI 2016]放棋子 最终得分 100
用户昵称 Gravatarsxysxy 是否通过 通过
代码语言 C++ 运行时间 0.011 s
提交时间 2016-09-22 21:40:08 内存使用 0.00 MiB
显示代码纯文本
  1. #include <cstdio>
  2. #include <cstring>
  3. #include <cstdlib>
  4. #include <algorithm>
  5. #include <vector>
  6. #include <list>
  7. #include <map>
  8. #include <cstring>
  9. #include <string>
  10. #include <iostream>
  11. #include <fstream>
  12. #include <cmath>
  13. using namespace std;
  14.  
  15. #define BIGNUM_DEBUG 2333
  16.  
  17. class Bignum
  18. {
  19. void mknum(const char *s, int len = -1)
  20. {
  21. sign = 0;
  22. if(*s == '-')
  23. {
  24. mknum(s+1);
  25. sign = 1;
  26. return;
  27. }
  28. int l;
  29. if(len == -1)
  30. l = strlen(s);
  31. else
  32. l = len;
  33. l = strlen(s);
  34. bits.clear();
  35. bits.resize(l);
  36. for(int i = l-1; i >= 0; i--)
  37. bits[l-i-1] = s[i] - '0';
  38. maintain();
  39. }
  40. void mknum(string &s)
  41. {
  42. mknum(s.c_str(), s.length());
  43. }
  44. // -------------
  45. void us_addto(Bignum &b) // unsigned add to
  46. {
  47. int mlen = max(b.bits.size(), bits.size());
  48. int slen = bits.size();
  49. int olen = b.bits.size();
  50. bits.resize(mlen);
  51. for(int i = 0; i < mlen; i++)
  52. {
  53. int s = 0;
  54. if(i < slen)s += bits[i];
  55. if(i < olen)s += b.bits[i];
  56. bits[i] = s;
  57. }
  58. maintain();
  59. }
  60. class FFTer
  61. {
  62. class Complex
  63. {
  64. public:
  65. double real, image;
  66. Complex(double a = 0, double b = 0)
  67. {
  68. real = a;
  69. image = b;
  70. }
  71. Complex operator + (const Complex &o){return Complex(real+o.real, image+o.image);}
  72. Complex operator - (const Complex &o){return Complex(real-o.real, image-o.image);}
  73. Complex operator * (const Complex &o){return Complex(real*o.real-image*o.image, real*o.image+o.real*image);}
  74. Complex operator * (double k){return Complex(real*k, image*k);}
  75. Complex operator / (double k){return Complex(real/k, image/k);}
  76. };
  77. public:
  78. vector<Complex> a; //系数向量
  79. int n; //多项式次数上界
  80. FFTer(vector<int> &vec)
  81. {
  82. a.resize(vec.size());
  83. for(int i = 0; i < vec.size(); i++)
  84. a[i].real = vec[i];
  85. n = vec.size();
  86. }
  87. void transform()
  88. {
  89. int j = 0;
  90. int k;
  91. for(int i = 0; i < n; i++)
  92. {
  93. if(j > i)swap(a[i], a[j]);
  94. k = n;
  95. while(j & (k >>= 1))j &= ~k;
  96. j |= k;
  97. }
  98. }
  99. void FFT(bool IDFT = false)
  100. {
  101. const double Pi = IDFT?-acos(-1.0):acos(-1.0);
  102. //IDFT与DFT选择方向相反(符号相反)
  103. transform(); //交换元素(翻转二进制位,具体看下面注释,再具体看算导
  104. for(int s = 1; s < n; s <<= 1)
  105. { //算法导论上是fot s = 1 to lgn,考虑到精度问题改为上面那个...
  106. for(int t = 0; t < n; t += s<<1)
  107. {
  108. //合并[t, t+s-1]与 [t+s, t+2*s-1] (算导上以指数形式给出,注意到他的s....)
  109. //合并为[t, t+2*s-1] (看起来像是废话) (有示例图在算导上,画得很形象的)
  110. /* 一个更简单的示例图
  111. (翻转过程) 翻转 合并
  112. 00 -> 00 0-|--|---------------------------
  113. | |
  114. 01 -> 10 1-|--|---\ /---------------------
  115. | | X
  116. 10 -> 01 2-|--|---/ \---------------------
  117. | |
  118. 11 -> 11 3-|--|---------------------------
  119. */
  120. double x = Pi/s;
  121. Complex omgn(cos(x), sin(x));
  122. Complex omg(1.0, 0.0); //单位向量
  123. for(int m = 0; m < s; m++)
  124. { //旋转
  125. int a1 = m+t;
  126. int a2 = m+t+s; //取两边系数向量的系数
  127. //算导上管这个叫公共子表达式消除
  128. //(其实就是一个变量计算一次然后保存下来用多次...嗯算导总是这么有逼格)
  129. Complex comm = omg * a[a2];
  130. a[a2] = a[a1] - comm;
  131. a[a1] = a[a1] + comm; //这两个顺序不要反了
  132. omg = omg * omgn;
  133. }
  134. }
  135. }
  136. if(IDFT)
  137. for(int i = 0; i < n; i++)
  138. a[i] = a[i] / n;
  139. }
  140. void mul(FFTer &o)
  141. {
  142. int s = 1;
  143. while(s < n + o.n)s <<= 1;
  144. n = o.n = s;
  145. a.resize(s);
  146. o.a.resize(s);
  147.  
  148. FFT(false);
  149. o.FFT(false);
  150. for(int i = 0; i < n; i++)
  151. a[i] = a[i] * o.a[i];
  152. FFT(true);
  153. }
  154. };
  155. void us_multo(Bignum &b)
  156. {
  157. FFTer x(bits);
  158. FFTer y(b.bits);
  159. x.mul(y);
  160. bits.clear();
  161. bits.resize(x.a.size());
  162. for(int i = 0; i < x.n; i++)
  163. bits[i] = (int)(x.a[i].real+0.5);
  164. maintain();
  165. }
  166. void us_multo_simu(Bignum &b)
  167. {
  168. vector<int> r;
  169. r.resize(max(length(),b.length())<<1);
  170. for(int i = 0; i < length(); i++)
  171. for(int j = 0; j < b.length(); j++)
  172. r[i+j] += bits[i] * b.bits[j];
  173. *(&(this -> bits)) = r;
  174. maintain();
  175. }
  176. void us_subto(Bignum &b) // abs(self) >= abs(other)
  177. {
  178. int mlen = length();
  179. int olen = b.length();
  180. for(int i = 0; i < mlen; i++)
  181. {
  182. int s = bits[i];
  183. if(i < olen)s -= b.bits[i];
  184. bits[i] = s;
  185. if(bits[i] < 0)
  186. {
  187. bits[i] += 10;
  188. bits[i+1] -= 1;
  189. }
  190. }
  191. for(int i = bits.size() - 1; !bits[i] && i >= 1; i--)bits.pop_back();
  192. if(bits.size() == 1 && bits[0] == 0)sign = 0;
  193. }
  194. void us_divto(Bignum &b)
  195. {
  196. if(length() == 1 && bits[0] == 0)return;
  197. Bignum L("0");
  198. L.sign = 0;
  199. Bignum R(*this);
  200. R.sign = 0;
  201. Bignum two("2");
  202. R *= two;
  203. Bignum one("1");
  204. one.sign = 0;
  205. while(L + one != R)
  206. {
  207. Bignum M = L+R;
  208. M.divto2();
  209. Bignum t = M*b;
  210. if(t > *this)
  211. {
  212. R = M;
  213. }else if(t < *this)
  214. {
  215. L = M;
  216. }else
  217. {
  218. *this = M;
  219. L = M;
  220. break;
  221. }
  222. }
  223. *this = L;
  224. }
  225. public:
  226. int sign;
  227. vector<int> bits;
  228. int length()
  229. {
  230. return bits.size();
  231. }
  232. void maintain()
  233. {
  234. for(int i = 0; i < bits.size(); i++)
  235. {
  236. if(i + 1 < bits.size())
  237. bits[i+1] += bits[i]/10;
  238. else if(bits[i] > 9)
  239. bits.push_back(bits[i]/10);
  240. bits[i] %= 10;
  241. }
  242. if(bits.size() == 0)
  243. {
  244. bits.push_back(0);
  245. sign = 0;
  246. }
  247. for(int i = bits.size() - 1; !bits[i] && i >= 1; i--)bits.pop_back();
  248. }
  249.  
  250. Bignum(string &s)
  251. {
  252. Bignum();
  253. mknum(s);
  254. }
  255. Bignum(const char *s)
  256. {
  257. Bignum();
  258. mknum(s);
  259. }
  260. Bignum(int n)
  261. {
  262. Bignum();
  263. char buf[15];
  264. sprintf(buf, "%d", n);
  265. mknum(buf);
  266. }
  267. Bignum()
  268. {
  269. sign = 0;
  270. bits.push_back(0);
  271. }
  272. Bignum(const Bignum& b)
  273. {
  274. copy(b);
  275. }
  276. void copy(const Bignum& b)
  277. {
  278. sign = b.sign;
  279. bits = b.bits;
  280. }
  281.  
  282. // ------------------------------------------
  283. bool us_cmp(Bignum &b) //无符号的比较
  284. {
  285. if(length() != b.length())return false;
  286. int l = length();
  287. for(int i = 0; i < l; i++)
  288. if(bits[i] != b.bits[i])
  289. return false;
  290. return true;
  291. }
  292. bool us_larger(Bignum &b)
  293. {
  294. if(length() > b.length())return true;
  295. else if(length() < b.length())return false;
  296. int l = length();
  297. for(int i = l-1; i >= 0; i--)
  298. if(bits[i] > b.bits[i])
  299. return true;
  300. else if(bits[i] < b.bits[i])
  301. return false;
  302. return false;
  303. }
  304. bool operator== (Bignum &o)
  305. {
  306. if(sign != o.sign)
  307. return false;
  308. return us_cmp(o);
  309. }
  310. bool operator!= (Bignum &o)
  311. {
  312. return !(*this == o);
  313. }
  314. bool operator> (Bignum &o)
  315. {
  316. if(sign == 0 && o.sign == 1)return true;
  317. if(sign == 1 && o.sign == 0)return false;
  318. if(sign == o.sign && sign)return !us_larger(o);
  319. return us_larger(o);
  320. }
  321. bool operator< (Bignum &o)
  322. {
  323. return !(*this == o || *this > o); //小于就是不等于也不大于
  324. }
  325. bool operator<= (Bignum &o)
  326. {
  327. return *this < o || *this == o;
  328. }
  329. bool operator>= (Bignum &o)
  330. {
  331. return *this > o || *this == o;
  332. }
  333.  
  334. // -------------------------------
  335. Bignum& operator+= (Bignum &o)
  336. {
  337. if(!sign && !o.sign)
  338. {
  339. us_addto(o);
  340. sign = 0;
  341. }
  342. else if(sign && o.sign)
  343. {
  344. us_addto(o);
  345. sign = 1;
  346. }
  347. else if(sign && !o.sign)
  348. {
  349. if(o.us_larger(*this))
  350. {
  351. Bignum t(o);
  352. t.us_subto(*this);
  353. *this = t;
  354. sign = 0;
  355. }else
  356. {
  357. us_subto(o);
  358. sign = 1;
  359. if(bits.size() == 1 && bits[0] == 0)sign = 0;
  360. }
  361. }else if(!sign && o.sign)
  362. {
  363. if(us_larger(o))
  364. {
  365. us_subto(o);
  366. sign = 0;
  367. }else
  368. {
  369. Bignum t(o);
  370. t.us_subto(*this);
  371. *this = t;
  372. sign = 1;
  373. if(bits.size() == 1 && bits[0] == 0)sign = 0;
  374. }
  375. }
  376. return *this;
  377. }
  378. Bignum operator+ (Bignum &o)
  379. {
  380. Bignum t(*this);
  381. t += o;
  382. return t;
  383. }
  384.  
  385. // ------------------------------
  386. Bignum& operator*= (Bignum &o)
  387. {
  388. if(length() + o.length() > 800)
  389. us_multo(o); //FFT
  390. else
  391. us_multo_simu(o); //simulate
  392. if(sign == o.sign)sign = 0;
  393. else sign = 1;
  394. return *this;
  395. }
  396. Bignum operator* (Bignum &o)
  397. {
  398. Bignum t(*this);
  399. t *= o;
  400. return t;
  401. }
  402. // -------------------------------
  403. Bignum& operator-= (Bignum &o)
  404. {
  405. if(!sign && !o.sign)
  406. {
  407. if(us_larger(o))
  408. {
  409. us_subto(o);
  410. sign = 0;
  411. }
  412. else
  413. {
  414. Bignum t(o);
  415. t.us_subto(*this);
  416. *this = t;
  417. sign = 1;
  418. if(bits.size() == 1 && bits[0] == 0)sign = 0;
  419. }
  420. }else if(sign && o.sign)
  421. {
  422. if(us_larger(o))
  423. {
  424. us_subto(o);
  425. sign = 1;
  426. if(bits.size() == 1 && bits[0] == 0)sign = 0;
  427. }else
  428. {
  429. Bignum t(o);
  430. t.us_subto(*this);
  431. *this = t;
  432. sign = 0;
  433. }
  434. }else if(!sign && o.sign)
  435. {
  436. us_addto(o);
  437. sign = 0;
  438. }else if(sign && !o.sign)
  439. {
  440. us_addto(o);
  441. sign = 1;
  442. }
  443. return *this;
  444. }
  445. Bignum operator- (Bignum &o)
  446. {
  447. Bignum t(*this);
  448. t -= o;
  449. return t;
  450. }
  451. // ---------------------------------
  452. Bignum& divto2()
  453. {
  454. if(!bits.size())return *this;
  455. bits[0] >>= 1;
  456. int i;
  457. for(i = 1; i < bits.size(); i++)
  458. {
  459. if(bits[i] & 1)bits[i-1] += 5;
  460. bits[i] >>= 1;
  461. }
  462. if(bits[i-1] == 0)bits.pop_back();
  463. return *this;
  464. }
  465. Bignum& operator/= (Bignum &o)
  466. {
  467. us_divto(o);
  468. if(sign == o.sign)sign = 0;
  469. else sign = 1;
  470. return *this;
  471. }
  472. Bignum operator/ (Bignum &o)
  473. {
  474. Bignum t(*this);
  475. t /= o;
  476. return t;
  477. }
  478. // ---------------------------------
  479. Bignum abs()
  480. {
  481. Bignum t(*this);
  482. t.sign = 0;
  483. return t;
  484. }
  485.  
  486.  
  487. Bignum sqrt()
  488. {
  489. Bignum L("0"), R(*this);
  490. Bignum one("1");
  491. Bignum m, t;
  492. while(L + one != R)
  493. {
  494. m = L+R;
  495. m.divto2();
  496. Bignum t = m*m;
  497. if(t == *this)return m;
  498. else if(t > *this)R = m;
  499. else L = m;
  500. }
  501. return L;
  502. }
  503.  
  504. //若e <= 0则会返回1
  505. //底数(也就是this)是负数的话会根据次数决定符号
  506. Bignum pow(Bignum &e)
  507. {
  508. if(e.sign)return 1;
  509. Bignum ans("1");
  510. Bignum base(*this);
  511. Bignum zero("0");
  512. Bignum exp(e);
  513. while(exp > zero)
  514. {
  515. if(exp.bits[0] & 1)
  516. {
  517. ans *= base;
  518. }
  519. base *= base;
  520. exp.divto2();
  521. }
  522. if(sign && e.bits[0] & 1)ans.sign = 1;
  523. return ans;
  524. }
  525.  
  526. //注意,如果本数小于底数返回1...
  527. Bignum log(Bignum &base)
  528. {
  529. if(sign)return 0;
  530. if(length() == 1 && bits[0] == 1)return 0;
  531. if(*this <= base)return 1;
  532. Bignum one("1");
  533.  
  534. Bignum r("1");
  535. Bignum c("0");
  536. while(r < *this)
  537. {
  538. r *= base;
  539. c += one;
  540. }
  541. if(r != *this)c -= one;
  542. return c;
  543. }
  544. Bignum lg()
  545. {
  546. Bignum ten("10");
  547. return log(ten);
  548. }
  549.  
  550. Bignum factorial()
  551. {
  552. Bignum r("1");
  553. Bignum zero("0");
  554. Bignum one("1");
  555. Bignum t(*this);
  556. while(t > zero)
  557. {
  558. r *= t;
  559. t -= one;
  560. }
  561. return r;
  562. }
  563.  
  564. // -----------------------------------
  565. friend istream& operator>>(istream &is, Bignum &b)
  566. {
  567. string s;
  568. is >> s;
  569. b.mknum(s);
  570. return is;
  571. }
  572. friend ostream& operator<<(ostream &os, Bignum b)
  573. {
  574. if(b.sign)os << '-';
  575. for(int i = b.bits.size()-1; i >= 0; i--)os << b.bits[i];
  576. return os;
  577. }
  578.  
  579. string to_string()
  580. {
  581. int sz = length();
  582. string s;
  583. if(sign)
  584. s.resize(sz+1);
  585. else
  586. s.resize(sz);
  587. int i = 0;
  588. if(sign)s[i++] = '-';
  589. for(int j = sz-1; i < sz+sign; i++, j--)
  590. s[i] = bits[j] + '0';
  591. return s;
  592. }
  593.  
  594. };
  595. // --
  596. #ifdef BIGNUM_DEBUG
  597. #ifdef __GNUC__
  598. __attribute__((noinline)) //禁止内联
  599. #endif
  600. #ifdef __MINGW32__
  601. __attribute__((noinline))
  602. #endif
  603. char* printB(Bignum &b)
  604. {
  605. //仅仅是用于能在gdb中使用它来输出自己
  606. string s = b.to_string();
  607. char *buf = (char *)malloc(sizeof(char) * s.length());
  608. //这个函数仅用于调试,这里申请的内存不会释放
  609. //因此非调试时不要使用这个函数
  610. strcpy(buf, s.c_str());
  611. return buf; //然后gdb中就可以 print printB(一个Bignum )
  612. }
  613. #endif
  614. Bignum data[202];
  615. int main()
  616. {
  617. /*
  618. //freopen("bzoj_1002.in", "r", stdin);
  619. //freopen("bzoj_1002.out", "w", stdout);
  620. data[1] = Bignum("1");
  621. data[2] = Bignum("5");
  622. Bignum two("2");
  623. Bignum three("3");
  624. int n;
  625. scanf("%d", &n);
  626. for(int i = 3; i <= n; i++)
  627. data[i] = (data[i-1]*three-data[i-2]+two);
  628. cout << data[n];
  629. */
  630. freopen("chess_2016.in", "r", stdin);
  631. freopen("chess_2016.out", "w", stdout);
  632. int n;
  633. cin >> n;
  634. data[1] = Bignum("0");
  635. data[2] = Bignum("1");
  636. for(int i = 3; i <= n; i++)
  637. {
  638. data[i] = Bignum(i-1);
  639. Bignum t = data[i-1]+data[i-2];
  640. data[i] *= t;
  641. }
  642. cout << data[n];
  643. return 0;
  644. }