UVa-12333:Revenge of Fibonacci 高精度

之前自己仿照紫书上写了高精度库,完善了乘法、减法,并且通过了和C++高精度库GMP的对拍测试,并在一些OJ上过了一些高精度的模板题,代码仓库地址:https://github.com/Edward-Elric233/BigInt

求解思路

题目的意思是求前100000斐波那契数列中某个前缀(不超过40个字符)第一次出现的位置。刚开始我的想法很简单,先求出这十万个斐波那契数列的前缀,然后每次读入的时候查找一遍就可以了,结果超时了。每次查找的复杂度是O(1e5∗40)O(1e5 * 40)O(1e540),有5e4个样例,这样10s也会超时。

但是我发现这个数据结构只有查找操作,因此使用unordered_map再合适不过了。我将所有前缀保存在这个unordered_map中,每次查找近似都是O(1)O(1)O(1)的,因此肯定不会超时。

需要注意的是这里将前缀保存的时候需要将每个数字的所有前缀都保存,例如对于123,就要保存前缀112123

AC代码

// Copyright(C), Edward-Elric233
// Author: Edward-Elric233
// Version: 1.0
// Date: 2021/8/8
// Description:#include <string>
#include <vector>
#include <iostream>
#include <sstream>
#include <algorithm>
#include <iomanip>
#include <map>
#include <unordered_map>class BigInt {
public:using ll = long long;static constexpr int BASE = 1e8;    //基数是1e8,即满1e8才进一位static constexpr int WIDTH = 8;     //每一位的十进制长度是8位,主要用于字符串和转换std::vector<int> bit;               //保存BigInt的每一位,小端模式:低位低字节,高位高字节bool negative;                      //记录数字是否是负数static void trim(std::string &s);                           //trim函数:删除头部和尾部的空格static void num2big(std::vector<int> &bit, ll num);         //实际处理函数:将正数num放入bit数组中static void str2big(std::vector<int> &bit, std::string &s); //实际处理函数:将正数字符串放入bit数组中static bool less(const BigInt &lhs, const BigInt &rhs);     //比较两个数字的绝对值的大小BigInt(ll num = 0);BigInt(std::string s); BigInt & operator =(ll num);         //必须返回BigInt&,与内置类型一致BigInt & operator =(std::string s);BigInt operator -() const;friend std::ostream &operator << (std::ostream &os, const BigInt &bigInt);friend std::istream &operator >> (std::istream &is, BigInt &bigInt);friend BigInt operator + (const BigInt &lhs, const BigInt &rhs);friend BigInt operator - (const BigInt &lhs, const BigInt &rhs);friend BigInt operator * (const BigInt &lhs, const BigInt &rhs);friend BigInt operator / (const BigInt &lhs, const BigInt &rhs);friend bool operator < (const BigInt &lhs, const BigInt &rhs);friend bool operator > (const BigInt &lhs, const BigInt &rhs);friend bool operator == (const BigInt &lhs, const BigInt &rhs);friend bool operator != (const BigInt &lhs, const BigInt &rhs);friend bool operator <= (const BigInt &lhs, const BigInt &rhs);friend bool operator >= (const BigInt &lhs, const BigInt &rhs);BigInt & operator += (const BigInt &rhs);BigInt & operator -= (const BigInt &rhs);BigInt & operator *= (const BigInt &rhs);BigInt & operator /= (const BigInt &rhs);std::string toString() const;BigInt & setOppo();                 //取相反数bool isNegative() const;            //判断是否是一个负数
};std::ostream & operator << (std::ostream &os, const BigInt &bigInt);
std::istream & operator >> (std::istream &is, BigInt &bigInt);
BigInt operator + (const BigInt &lhs, const BigInt &rhs);
BigInt operator - (const BigInt &lhs, const BigInt &rhs);
BigInt operator * (const BigInt &lhs, const BigInt &rhs);
BigInt operator / (const BigInt &lhs, const BigInt &rhs);
bool operator < (const BigInt &lhs, const BigInt &rhs);
bool operator > (const BigInt &lhs, const BigInt &rhs);
bool operator == (const BigInt &lhs, const BigInt &rhs);
bool operator != (const BigInt &lhs, const BigInt &rhs);
bool operator <= (const BigInt &lhs, const BigInt &rhs);
bool operator >= (const BigInt &lhs, const BigInt &rhs);BigInt BigInt::operator-() const {BigInt ret = *this;ret.negative = !ret.negative;return ret;
}BigInt & BigInt::setOppo() {negative = !negative;return *this;
}bool BigInt::isNegative() const {return negative;
}void BigInt::num2big(std::vector<int> &bit, ll num) {ll x;//必须使用do while,循环至少应该执行一遍,处理num是0的情况do {x = num % BASE;bit.push_back(static_cast<int>(x));num /= BASE;} while (num);
}BigInt::BigInt(ll num):negative(false) {if (num < 0) {num = -num;negative = true;}num2big(this->bit, num);
}void BigInt::str2big(std::vector<int> &bit, std::string &s) {int len = (s.size() - 1) / WIDTH + 1;       //ceil得到lenint start, end;for (int i = 0; i < len; ++i) {end = s.size() - i * WIDTH;start = std::max(0, end - WIDTH);bit.push_back(std::stoi(s.substr(start, end - start)));}
}BigInt::BigInt(std::string s):negative(false) {trim(s);if (s[0] == '-') {negative = true;s.erase(s.begin());}str2big(this->bit, s);
}void BigInt::trim(std::string &s) {auto ltrim = [](std::string &s) {s.erase(s.begin(), std::find_if(s.begin(), s.end(), [](unsigned char c) -> bool {//找到第一个非空字符return !std::isspace(c);}));};auto rtrim = [](std::string &s) {s.erase(std::find_if(s.rbegin(), s.rend(), [](unsigned char c) -> bool {//找到最后一个非空字符return !std::isspace(c);}).base(), s.end());};ltrim(s);rtrim(s);
}BigInt &BigInt::operator=(ll num) {bit.clear();negative = false;if (num < 0) {num = -num;negative = true;}num2big(this->bit, num);return *this;
}BigInt &BigInt::operator=(std::string s) {bit.clear();negative = false;trim(s);if (s[0] == '-') {negative = true;s.erase(s.begin());}str2big(this->bit, s);return *this;
}std::ostream &operator << (std::ostream &os, const BigInt &bigInt) {if (bigInt.negative) {//输出负号os << "-";}auto &bit = bigInt.bit;//以大端字节序输出os << bit.back();//除了最后一位,其他的如果有前导0不能忽略os << std::setfill('0');for (int i = bit.size() - 2; i >= 0; --i) {os << std::setw(BigInt::WIDTH) << bit[i];}os << std::setfill(' ');return os;
}std::istream &operator >> (std::istream &is, BigInt &bigInt) {std::string s;if (is >> s) {//必须判断是否读入成功bigInt = s;}return is;
}BigInt & BigInt::operator+=(const BigInt &rhs) {if (!negative && rhs.negative) {BigInt tmp = rhs;return *this = (tmp -= this->setOppo());} else if (negative && !rhs.negative) {return *this -= -rhs;}auto &rbit = rhs.bit;int max_len = std::max(bit.size(), rbit.size());int min_len = std::min(bit.size(), rbit.size());int g = 0;      //进位for (int i = 0; i < min_len; ++i) {bit[i] += rbit[i] + g;g = bit[i] / BASE;bit[i] %= BASE;}if (bit.size() < rbit.size()) {for (int i = min_len; i < max_len; ++i) {bit.push_back(g + rbit[i]);g = bit.back() / BASE;bit.back() %= BASE;}} else {for (int i = min_len; i < max_len; ++i) {bit[i] += g;g = bit[i] / BASE;bit[i] %= BASE;if (!g) break;}}if (g) {//加法的进位最多为1bit.push_back(g);}return *this;
}BigInt operator + (const BigInt &lhs, const BigInt &rhs) {BigInt ret = lhs;ret += rhs;return std::move(ret);
}BigInt & BigInt::operator-=(const BigInt &rhs) {if (!negative && !rhs.negative) {if (*this >= rhs) {} else {BigInt tmp = rhs;tmp -= *this;return *this = tmp.setOppo();}} else if (!negative && rhs.negative) {this->setOppo();*this += rhs;this->setOppo();return *this;} else if (negative && !rhs.negative) {return *this += -rhs;} else {BigInt tmp = -rhs;this->setOppo();return *this = (tmp -= *this);}auto &rbit = rhs.bit;for (int i = 0; i < rbit.size(); ++i) {if (bit[i] < rbit[i]) {bit[i] += BASE;bit[i + 1] -= 1;}bit[i] -= rbit[i];}for (int i = rbit.size(); i < bit.size(); ++i) {if (bit[i] >= 0) {break;}bit[i] += BASE;bit[i + 1] -= 1;}//删除前导0for (int i = bit.size() - 1; i > 0; --i) {if (!bit[i]) bit.pop_back();}return *this;
}BigInt operator - (const BigInt &lhs, const BigInt &rhs) {BigInt res = lhs;res -= rhs;return std::move(res);
}BigInt & BigInt::operator*=(const BigInt &rhs) {//负负得正if (negative && rhs.negative || !negative && !rhs.negative) {negative = false;} else {negative = true;}auto &rbit = rhs.bit;constexpr ll LBASE = BASE;std::vector<ll> c(bit.size() + rbit.size(), 0);for (int i = 0; i < bit.size(); ++i) {for (int j = 0; j < rbit.size(); ++j) {c[i + j] += static_cast<ll>(bit[i]) * static_cast<ll>(rbit[j]);//在这里处理进位防止溢出if (c[i + j] >= LBASE) {//有必要再进行除法,毕竟除法比较慢c[i + j + 1] += c[i + j] / LBASE;c[i + j] %= LBASE;}}}//处理进位for (int i = 0; i < c.size(); ++i) {if (c[i] >= LBASE) {//有必要再进行除法,毕竟除法比较慢c[i + 1] += c[i] / LBASE;c[i] %= LBASE;}}//删除前导0for (int i = c.size() - 1; i > 0; --i) {    //至少留一位if (!c[i]) c.pop_back();else break;}bit.resize(c.size());for (int i = 0; i < c.size(); ++i) {bit[i] = static_cast<int>(c[i]);}return *this;
}BigInt operator * (const BigInt &lhs, const BigInt &rhs) {BigInt res = lhs;res *= rhs;return std::move(res);
}std::string BigInt::toString() const {std::ostringstream os;os << *this;return os.str();
}bool BigInt::less(const BigInt &lhs, const BigInt &rhs) {if (lhs.bit.size() != rhs.bit.size()) return lhs.bit.size() < rhs.bit.size();for (int i = lhs.bit.size() - 1; i >= 0; --i) {if (lhs.bit[i] != rhs.bit[i]) return lhs.bit[i] < rhs.bit[i];}//相等return false;
}bool operator < (const BigInt &lhs, const BigInt &rhs) {if (!lhs.negative && !rhs.negative) {return BigInt::less(lhs, rhs);} else if (lhs.negative && !rhs.negative) {return true;} else if (!lhs.negative && rhs.negative) {return false;} else if (lhs.negative && rhs.negative) {//都是负数if (BigInt::less(lhs, rhs)) {return false;} else if (BigInt::less(rhs, lhs)) {return true;} else {return false;}}
}bool operator > (const BigInt &lhs, const BigInt &rhs) {return rhs < lhs;
}bool operator == (const BigInt &lhs, const BigInt &rhs) {return !(lhs < rhs) && !(rhs < lhs);
}
bool operator != (const BigInt &lhs, const BigInt &rhs) {return (lhs < rhs) || (rhs < lhs);
}bool operator <= (const BigInt &lhs, const BigInt &rhs) {return !(rhs < lhs);
}bool operator >= (const BigInt &lhs, const BigInt &rhs) {return !(lhs < rhs);
}using namespace std;constexpr int MAXN = 1e5;
vector<BigInt> fib;
unordered_map<string, int> str_hash;int main() {ios::sync_with_stdio(false);fib.push_back(1);fib.push_back(1);for (int i = 2; i < MAXN; ++i) {fib.push_back(fib[i - 1] + fib[i - 2]);}for (int i = 0; i < MAXN; ++i) {string line;auto &bit = fib[i].bit;line.append(to_string(bit.back()));for (int i = bit.size() - 2, j = std::max(0, int(bit.size()) - 6); i >= j; --i) {const string &number = to_string(bit[i]);for (int i = 0, j = 8 - number.size(); i < j; ++i) line.append("0");line.append(number);}
//        cout << line << endl;for (int ii = 1, jj = std::min(int(line.size()), 40); ii <= jj; ++ii) {const string &w = line.substr(0, ii);
//            cout << w << endl;if (str_hash.count(w)) {if (i < str_hash[w]) {str_hash[w] = i;}} else {str_hash[w] = i;}}}
//    for (int i = 0; i < 10; ++i) cout << fibHead[i] << " ";
//    cout << "\n";int T;cin >> T;string line;for (int caseI = 1; caseI <= T; ++caseI) {cin >> line;cout << "Case #" << caseI << ": ";if (str_hash.count(line)) {cout << str_hash[line] << "\n";} else {cout << "-1\n";}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/383528.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vim命令笔记

vim折叠函数&#xff1a;https://www.cnblogs.com/zlcxbb/p/6442092.html Vim录制宏及使用&#xff1a;https://www.jianshu.com/p/9d999c72a9f3 将vim与系统剪贴板的交互使用&#xff1a;https://zhuanlan.zhihu.com/p/73984381

Educational Codeforces Round 114总结

绪论 https://codeforces.com/contest/1574/ 以前想要打CF&#xff0c;总是觉得没有时间&#xff0c;要做这个&#xff0c;要做那个&#xff0c;现在时间充裕了一些&#xff0c;想要多打一些CF&#xff0c;但是光打比赛不总结是没有什么帮助的&#xff0c;这是我从以前的ACM训…

UVA - 210:Concurrency Simulator

题目链接&#xff1a;https://vjudge.net/problem/UVA-210 题目分析 就是一道模拟题&#xff0c;但是细节有点多。 写代码两个小时&#xff0c;调试代码用了两天。。。很长时间不刷题了&#xff0c;这道虽然算法简单但是细节满满的题目对我来说是一个很好的热身。 尽量不要去…

UVA - 514:Rails

题目链接&#xff1a;https://vjudge.net/problem/UVA-514 题目分析 题目的意思是给一个栈输入一系列数据&#xff0c;在这个过程中可以出栈&#xff0c;看能否达到某个结果。 刚开始我觉得这个情况好多&#xff0c;因此不是用模拟&#xff0c;而应该观察结果本身。对于结果中…

UVA - 442:Matrix Chain Multiplication

题目链接&#xff1a;https://vjudge.net/problem/UVA-442 题目分析 题目的意思非常简单&#xff0c;就是给定一个矩阵乘法的表达式然后计算就可以了。随便写写 AC代码 #include <iostream> #include <deque> #include <vector> #include <string>…

leetcode869. 重新排序得到 2 的幂

题目连接&#xff1a;https://leetcode-cn.com/problems/reordered-power-of-2/ 题目分析 如果直接顺着题目的思路&#xff0c;得到数字n的全排列&#xff0c;然后再去判断其是不是2的幂是比较复杂的。 我们应该注意到&#xff0c;因为数字是可以随意排列的&#xff0c;因此所…

使用wireshark+ssh+tcpdump远程抓包

因为需要抓取远程服务器上的数据包&#xff0c;又不想使用tcpdump这种命令行工具进行&#xff08;用了wireshark后谁还愿意去看密密麻麻的命令行呢&#xff09;&#xff0c;所以在网上查找了一下使用wireshark远程抓包的方法&#xff0c;在这里记录一下。 原生支持 wireshark…

C++ Variadic Templates(可变参数模板)

本文参考侯捷老师的视频&#xff1a;https://www.youtube.com/watch?vTJIb9TGfDIw&listPL-X74YXt4LVYo_bk-jHMV5T3LHRYRbZoH 以及C primer第五版 相关内容。 可变参数模板函数 //递归的终止条件 void print() {} //Variadic Templates //一般用于递归处理 template <…

Ubuntu修复Fix Busybox Initramfs错误

今天早上我打开电脑&#xff0c;进入Ubuntu系统&#xff0c;结果黑屏了&#xff0c;屏幕显示&#xff1a; BusyBox v1.30.1 (Ubuntu 1:1.30.1-4ubuntu6.1) built-in shell (ash) Enter help for a list of built-in commands.(initramfs)然而我并不知道这个是什么意思&#x…

Leetcode第284场周赛

绪论 最近发现Leetcode每周的周赛难度挺适合我的&#xff0c;而且时间也比较友好&#xff08;不像Codeforces每次都是半夜&#xff09;。所以连续参加了三周的周赛。这次才想起来应该记录一下自己的参赛历程。一方面是总结经验&#xff0c;另一方面有了记录就更有动力去提升&a…

Leetcode第286场周赛

绪论 上周因为有事没有参加周赛&#xff0c;这周没有错过。这次周赛拿到了人生第一个AK&#xff0c;参加大大小小的比赛这么多次&#xff0c;从来没有AK过&#xff0c;泪目了。 感觉这次比赛的思维难度对我来讲稍高一些&#xff0c;前三道题就花了一个小时&#xff0c;而以往…

第287场周赛

绪论 虽然是上周日参加的比赛&#xff0c;但是这周没有怎么学习&#xff0c;每天就是玩耍。也导致对周赛的总结迟迟没有进行。想着再拖下去下次周赛都要开始了&#xff0c;在这里补一下。 这场比赛总体比上场简单一些&#xff0c;但是最后一道题因为忘记初始化类内变量导致调试…

第288场周赛

绪论 虽然没有AK&#xff0c;但是不知道为什么排名比以前AK了都靠前。可能是因为最后一道题有些难度&#xff0c;缩小了我和大佬之间的差距。最后一个小时写最后一道题&#xff0c;累死累活想了一个贪心遍历的算法&#xff0c;当时是一直RE&#xff0c;后来下来调了调又WA了。 …

Clion远程部署和运行

绪论 作为Clion的忠实粉丝&#xff0c;现在的我的几乎所有的coding都是通过Clion完成。因为需要在服务器上进行开发&#xff0c;又离不开Clion&#xff0c;就了解了如何通过Clion远程部署和开发。 主要是借鉴了博客&#xff1a;使用Clion优雅的完全远程自动同步和远程调试c。如…

C++ 单例模式 call_once : terminate called after throwing an instance of ‘std::system_error‘

在学习了C中可以使用call_once进行初始化资源后&#xff0c;我就想着写一个单例模板供以后使用。 template<typename T> class SingleTon {using Ptr std::shared_ptr<T>;static Ptr p;static std::once_flag flag;template<typename ...Args>static void …

C++读写锁造成死锁

C14支持std::shared_timed_mutex C17支持std::shared_mutex 前者相比后者支持的操作更多&#xff0c;但是后者相对性能更好。 使用std::lock_guard<std::shared_mutex>和std::unique_lock<std::shared_mutex>互斥访问使用std::shared_lock<std::shared_mutex…

每日一题:449. 序列化和反序列化二叉搜索树

题目分析 题目链接&#xff1a;449. 序列化和反序列化二叉搜索树 觉得序列化很简单&#xff0c;前序遍历、后序遍历、中序遍历、层序遍历等等。其中得到前序遍历和后序遍历是可以通过递归解法反序列化的&#xff0c;觉得这样子做有点复杂。就想着可不可以一次遍历。一次遍历的…

C++高效集合数据结构设计

绪论 在复杂算法实现过程中我们经常会需要一个高效的集合数据结构&#xff0c;支持常数级别的增、删、查&#xff0c;以及随机返回、遍历&#xff0c;最好还能够支持交集、并集、子集操作 哈希集合实现 大家可能很快想到unordered_set&#xff0c;unordered_set由于底层是哈…

C++ 工具函数库

在写一些大型项目的过程中经常需要一些工具函数&#xff0c;例如获取随机数、计时器、打印函数、重要常量&#xff08;如最大值&#xff09;、信号与槽等&#xff0c;由于每一个工程都自己手动实现一个实在是太傻&#xff0c;我将其总结放入一个文件中。 utils.h // Copyright…

muduo网络库使用入门

muduo网络库介绍 muduo网络库是陈硕大神开发的基于主从Reactor模式的&#xff0c;事件驱动的高性能网络库。 网络编程中有很多是事务性的工作&#xff0c;使用muduo网络库&#xff0c;用户只需要填上关键的业务逻辑代码&#xff0c;并将回调注册到框架中&#xff0c;就可以实…