首页 > 其他分享 >P1045 麦森数 题解

P1045 麦森数 题解

时间:2023-10-02 17:45:57浏览次数:45  
标签:ch 59 进制 麦森数 题解 ll long P1045 define

传送门


前排提醒:本篇题解没有使用压位和快速幂,运用了一种预处理的思想,希望能提供一种新的思路。


首先将 \(2^{p}-1(d)\) 转换为 \(1111…111(b)\)。

关于第一问:

我们先考虑 \(2\) 进制转 \(8\) 进制,将每 \(3\) 位转为 \(1\) 位,即每 \(\log{8}\) 位转为 \(1\) 位。

\(2\) 进制转 \(10\) 进制同理,将每 \(\log{10}\) 位转为 \(1\) 位,位数就是 \(\lceil \frac{p}{\log{10}} \rceil\)。

关于第二问:

考虑用高精度存储后 \(500\) 位,暴力乘上 \(p\) 个 \(2\),时间复杂度 \(\mathcal{O}(500 \times P)\),无法通过此题。

考虑优化暴力乘的过程,将几个 \(2\) 合并起来,乘上若干次 \(2^{k}\),使得 \(\sum k=p\) ,用 long long 存储每一位,那么每一位最多可以放 \(2^{63}-1\)。由于每一位 \(\in [0,9] \approx[0,2^4)\),所以每次最多可以乘上 \(2^{59}\)。

预处理 \(2^{59}\) 的值,将乘上 \(p\) 个 \(2\) 转换为乘上 \(\lfloor \frac{p}{59} \rfloor\) 个 \(2^{59}\) 和 \(p\mod{59}\) 个 \(2\),时间复杂度 \(\mathcal{O}(500 \times \lfloor \frac{p}{59} \rfloor)\)。


\(\mathcal{Code}\):

#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define ull unsigned long long
#define db double
#define il inline
#define re register
#define INF 0x3f3f3f3f
#define LINF 0x3f3f3f3f3f3f3f3f
#define int ll
#define F(i,a,b) for(re int (i)=(a);(i)<=(b);(i)++)
#define DF(i,a,b) for(re int (i)=(a);(i)>=(b);(i)--)
#define G(i,u) for(re int (i)=head[u];(i);(i)=nxt[(i)])
inline ll read(){ll x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}while(ch>='0'&&ch<='9'){x=(x<<3)+(x<<1)+ch-48;ch=getchar();}return x*f;}
int p;
int str[510],mi[60];
signed main()
{
	p=read();
	mi[0]=1;
	F(i,1,59) mi[i]=mi[i-1]*2;
	str[500]=1;
	F(i,1,p/59)
	{
		DF(k,500,1)
		{
			str[k]*=mi[59];
			str[k]+=str[k+1]/10;
			str[k+1]%=10;
		}
		str[1]%=10;
	}
	F(i,1,p%59)
	{
		DF(k,500,1)
		{
			str[k]<<=1;
			str[k]+=str[k+1]/10;
			str[k+1]%=10;
		}
		str[1]%=10;
	}
	str[500]--;
	printf("%d\n",(int)(ceil(p/log2(10))));
	F(t,1,10)
	{
		F(k,1,50) printf("%d",str[(t-1)*50+k]);
		putchar('\n');
	}
	return 0;
}

标签:ch,59,进制,麦森数,题解,ll,long,P1045,define
From: https://www.cnblogs.com/MooSheng/p/17740261.html

相关文章

  • UVA1471 防线 Defense Lines 题解
    传送门首先可以将题意大概可以简化为:取两端不重复的连续子序列,组成一个最长的连续递增子序列。我们先dp预处理出以\(i\)为结尾的连续递增子序列长度\(dpr_{i}\)。同样预处理出以\(i\)为开头的连续递增子序列长度\(dpl_{i}\)。考虑对于每个\(dpr_{i}\),找到满足\(a_{......
  • P5503 灯塔 题解
    决策单调性二分传送门数据加强版:P3515前置知识:二分,决策单调性首先很容易写出答案式子:\[ans_{i}=\max_{j=i}^{n}{(a_{j}-a_{i}+\lceil\sqrt{\left|i-j\right|}\rceil)}\]先将向上取整符号拆掉,只要在输出时处理就行。再将绝对值符号拆掉,分\(j<i\)和\(j>i\)的情况......
  • 题解 Codeforces Round 901 (Div. 1) / CF1874A~E
    题解CodeforcesRound901(Div.1)/CF1874A~E比赛情况:过了AB。赛后发现B是假复杂度。https://codeforc.es/contest/1874A.JellyfishandGameProblemAlice&Bob又在博弈,Alice手上有\(n\)个苹果,第\(i\)个苹果的价值是\(a_i\);Bob手上有\(m\)个苹果,第\(i\)......
  • [题解]AT_abc240_f [ABC240F] Sum Sum Max
    思路题目要求的是\(\max_{a=1}^{n}\{\sum_{i=1}^{a}\sum_{j=1}^{a}{A_j}\}\),所以我们将\(\sum_{i=1}^{a}\sum_{j=1}^{a}{A_j}\)化简一下,得:\[i\timesA_1+(i-1)\timesA_2+\dots+1\timesA_x\]在\(a\)每增加\(1\)时,这个和\(s\)将会变为\(s+......
  • [题解]AT_abc245_f [ABC245F] Endless Walk
    思路首先我们可以发现,在任意一个节点数量大于\(1\)的强连通分量中的点都满足条件。所以,我们可以对这张图跑一边TarJan。但是这样是错的,因为我们还需要考虑节点数量为\(1\)的强连通分量。如果这种连通分量能够到达任意一个节点数量大于\(1\)的强连通分量,那么,这个连通分......
  • CSES.1141 C++题解
    题意传送门有一个长度为\(n\)的歌单,问最长多少首歌互不相同?每首歌用一个\(1-10^9\)的整数表示。样例输入812132742样例输出5算法双指针算法。桶思想。对于歌单中重复出现的数,可以用桶来存储。定义两个指针i,j,i指向大数,j指向小数。当出现某个桶的数大于1时,则......
  • CF1878C Vasilije in Cacak 题解
    题目传送门简化题意有\(t\)组询问,每次询问是否能从\(1\simn\)中选择\(k\)个数使得它们的和为\(x\)。解法考虑临界情况,从\(1\simn\)中选择最小的\(k\)个数时和为\(\sum\limits_{i=1}^ki=\dfrac{(k+1)k}{2}\),从\(1\simn\)中选择最大的\(k\)个数时和为\(......
  • Codeforces 1765H 题解
    题目大意题目大意给定一个\(n\)个点和\(m\)条边的有向图,并给定\(p_1,p_2,\cdots,p_n\)表示第\(i\)个点的拓扑序必须小于等于\(p_i\),求出每个点的最小拓扑序。题解题解题目要求拓扑序尽量小,转换一下就是在反图上拓扑序尽量大。考虑拓扑排序,当一个点不得不入队......
  • UVA10054 The Necklace 题解
    好可恶一道题,怎么没人告诉我输出之间有空行(思路是先抽象成图,然后跑一边dfs记录边的前后顺序。对于不能成环的情况,只需要再开个数组记录度数判断奇点即可。若存在奇点则break掉,剩下的跑dfs、//producedbymiya555//stupidmistakes:1.多测要清空2.输出之间有空行//ideas:d......
  • 题解 hdu 1269 迷宫城堡
    找点图论练习题写,发现hdu又寄了,那就发到blog里吧。思路:tarjan缩点判断DAG中点数是否为1。若是,则该图为强连通图。 //producedbymiya555//stupidmistakes:多测记得清空//ideas:tarjan模板#include<bits/stdc++.h>usingnamespacestd;constintN=10010;intn,m,low[......