CF912E Prime Gift 数学
Posted zxyqzy
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了CF912E Prime Gift 数学相关的知识,希望对你有一定的参考价值。
Opposite to Grisha‘s nice behavior, Oleg, though he has an entire year at his disposal, didn‘t manage to learn how to solve number theory problems in the past year. That‘s why instead of Ded Moroz he was visited by his teammate Andrew, who solemnly presented him with a set of n distinct prime numbers alongside with a simple task: Oleg is to find the k-th smallest integer, such that all its prime divisors are in this set.
The first line contains a single integer n (1?≤?n?≤?16).
The next line lists n distinct prime numbers p1,?p2,?...,?pn (2?≤?pi?≤?100) in ascending order.
The last line gives a single integer k (1?≤?k). It is guaranteed that the k-th smallest integer such that all its prime divisors are in this set does not exceed 1018.
Print a single line featuring the k-th smallest integer. It‘s guaranteed that the answer doesn‘t exceed 1018.
3
2 3 5
7
8
5
3 7 11 13 31
17
93
The list of numbers with all prime divisors inside {2,?3,?5} begins as follows:
(1,?2,?3,?4,?5,?6,?8,?...)
The seventh number in this list (1-indexed) is eight.
首先在1e18的范围内,那么质数的个数最多只有16个,你可以将这16个列出来然后相乘,看看多大;
那么我们分开算;
不妨将奇数位置的dfs和偶数位置的分别dfs;
分别算出包含这些因子的所有可能值,如果对数组开多大不确定,用 vector.push_back就好;
那么我们要求第k大,考虑二分答案;
那么对于我们之前求出的那两个集合,我们用 two-pointer 来扫一遍确定当前答案是第几个;
然后调整上下界即可;
#include<iostream> #include<cstdio> #include<algorithm> #include<cstdlib> #include<cstring> #include<string> #include<cmath> #include<map> #include<set> #include<vector> #include<queue> #include<bitset> #include<ctime> #include<deque> #include<stack> #include<functional> #include<sstream> //#include<cctype> //#pragma GCC optimize(2) using namespace std; #define maxn 300005 #define inf 0x3f3f3f3f //#define INF 1e18 #define rdint(x) scanf("%d",&x) #define rdllt(x) scanf("%lld",&x) #define rdult(x) scanf("%lu",&x) #define rdlf(x) scanf("%lf",&x) #define rdstr(x) scanf("%s",x) typedef long long ll; typedef unsigned long long ull; typedef unsigned int U; #define ms(x) memset((x),0,sizeof(x)) const long long int mod = 1e9 + 7; #define Mod 1000000000 #define sq(x) (x)*(x) #define eps 1e-3 typedef pair<int, int> pii; #define pi acos(-1.0) const int N = 1005; #define REP(i,n) for(int i=0;i<(n);i++) typedef pair<int, int> pii; inline ll rd() { ll x = 0; char c = getchar(); bool f = false; while (!isdigit(c)) { if (c == ‘-‘) f = true; c = getchar(); } while (isdigit(c)) { x = (x << 1) + (x << 3) + (c ^ 48); c = getchar(); } return f ? -x : x; } ll gcd(ll a, ll b) { return b == 0 ? a : gcd(b, a%b); } ll sqr(ll x) { return x * x; } /*ll ans; ll exgcd(ll a, ll b, ll &x, ll &y) { if (!b) { x = 1; y = 0; return a; } ans = exgcd(b, a%b, x, y); ll t = x; x = y; y = t - a / b * y; return ans; } */ int n; int p[30]; ll k; vector<ll>vc[2]; int tmp[30]; int cnt; void dfs(int belong, ll val, int cur) { vc[belong].push_back(val); for (int i = cur; i <= cnt; i++) { if (1e18 / tmp[i] >= val) { dfs(belong, val*tmp[i], i); } } } ll sol(ll x) { ll res = 0; int j = 0; for (int i = vc[0].size() - 1; i >= 0; i--) { while (j < vc[1].size() && vc[1][j] <= x / vc[0][i])j++;// 双指针扫一遍 res += j; } return res; } int main() { //ios::sync_with_stdio(0); rdint(n); for (int i = 1; i <= n; i++)rdint(p[i]); rdllt(k); for (int i = 1; i <= n; i++)if (i & 1)tmp[++cnt] = p[i]; dfs(0, 1, 1); cnt = 0; for (int i = 1; i <= n; i++)if (i % 2 == 0)tmp[++cnt] = p[i]; dfs(1, 1, 1); sort(vc[0].begin(), vc[0].end()); sort(vc[1].begin(), vc[1].end()); ll l = 1, r = 1e18; while (l <= r) { ll mid = (l + r) / 2; if (sol(mid) >= k)r = mid - 1; else l = mid + 1; } cout << l << endl; return 0; }
以上是关于CF912E Prime Gift 数学的主要内容,如果未能解决你的问题,请参考以下文章
CF912E Prime Gift(Meeting In The Middle+Two pointers)
Prime Gift CodeForces - 912E (中途相遇)