bupt summer training for 16 #4 ——数论
https://vjudge.net/contest/173277#overview
A.平方差公式后变为 n = (x + y)(x - y)
令 t = x - y ,变成
n = (t + 2x) * t,要 x 最小
O(sqrt(n)) 枚举 n 的因数即可
#include <cstdio>
typedef long long ll;
ll n, m, k, t;
int main() {
scanf("%lld", &n);
while(n --) {
k = ;
scanf("%lld", &m);
for(ll i = ;i * i < m;i ++) {
if((m % i == ) && ((m / i - i) % == ) ) {
t = (m / i - i) / ;
k = ;
}
}
if(!k) puts("-1");
else printf("%lld\n", t);
}
return ;
}
B.
C.如果GCD中间是加号会困难很多
乘号的话,我们枚举质数对ans的贡献即可
质数 pi 对结果的贡献就是
pow(pi,(n/pi)*(m/pi) + (n/pi^2)*(m/pi^2) + (n/pi^3)*(m/pi^3) +... )
最坏效率,1kw质数大约 n / ln(n) 个
pi = 2, 计算贡献需要logn次,快速幂约logn
复杂度约为 O(n / ln(n) * logn),考虑到后面质数变大
pi > 100时, 计算贡献就只有3次了
实际计算贡献和快速幂平均下来不到2次
所以 n = 1kw 时是O(n)级别的
#include <cstdio>
#include <algorithm> using std::swap; typedef long long ll; int p[], v[]; int tt; long long n, m, ans; const int Mod = 1e9 + ; ll calc(ll x, ll k) {
ll res = ;
for(;k > ;x = x * x % Mod, k >>= )
if(k & ) res = res * x % Mod;
return res;
} int main() {
for(int i = ;i <= ;i ++) {
if(!v[i]) p[++ p[]] = i;
for(int j = ;j <= p[] && 1ll * i * p[j] <= ;j ++) {
v[i * p[j]] = ;
if(i % p[j] == ) break;
}
}
long long t, cnt;
scanf("%d", &tt);
while(tt --) {
ans = ;
scanf("%lld %lld", &n, &m);
if(m > n) swap(n, m);
for(int i = ;p[i] <= m && i <= p[];i ++) {
t = p[i], cnt = ;
while() {
cnt += (n / t) * (m / t);
if(m / t >= p[i]) t *= p[i];
else break;
}
ans = ans * calc(p[i], cnt) % Mod;
}
printf("%lld\n", ans);
}
return ;
}
D.这个题今年3月刚写的又忘记了...
我们一个想法是分解质因数
得到 n! 里每个质数出现的最少次数
但我们平时分解质因数都是O(sqrt(n))的
即使使用前缀和,10^7也无法O(n^1.5)分解
所以这里用了一个巧妙又优秀的的O(nlogn)处理方法
for(int i = ;i * i <= m;i ++) {
if(v[i]) continue;
for(int j = i * i;j <= m;j += i)
v[j] = -i;
}
for(int i = ;i <= m;i ++) {
int j = i;
while() {
if(v[j] >= ) {
cout << j << endl;
break;
}
else cout << -v[j] << endl, j /= -v[j];
}
}
当 i 为非质数时, -vis[i] 存的是 i 的最大的不大于sqrt(i)的质因数
分解质因数后可以二分
也可以用每个质数出现的个数得到不等式 n >= ci
实际效率约均为O(n)级别,前者思考复杂度低
#include <bits/stdc++.h> #define rep(i, j, k) for(int i = j;i <= k;i ++)
#define rev(i, j, k) for(int i = j;i >= k;i --) using namespace std; typedef long long ll; const int maxn = ; int n, m, a[maxn], b[maxn], p[maxn]; int v[maxn], num[maxn]; ll cnt[maxn]; bool judge(ll x) {
ll y, c;
rep(i, , p[]) {
y = x, c = ;
while(y) y /= p[i], c += y;
if(c < cnt[i]) return ;
}
return ;
} int main() {
ios::sync_with_stdio(false);
cin >> n;
rep(i, , n) {
cin >> a[i];
m = max(m, a[i]);
b[a[i]] ++;
}
for(int i = ;i * i <= m;i ++) {
if(v[i]) continue;
for(int j = i * i;j <= m;j += i)
v[j] = -i;
}
rep(i, , m)
if(!v[i])
p[++ p[]] = i, num[i] = p[];
rev(i, m, )
b[i] += b[i + ];
rep(i, , m) {
int j = i;
while() {
if(v[j] >= ) {
cnt[num[j]] += b[i];
break;
}
else cnt[num[-v[j]]] += b[i], j /= -v[j];
}
}
ll l = , r = 10000000000000ll, mid, ans;
while(l <= r) {
mid = (l + r) >> ;
if(judge(mid)) ans = mid, r = mid - ;
else l = mid + ;
}
cout << ans;
return ;
}
E.利用欧拉函数的性质,对于给定的 x
满足 phi[n] >= x 的最小的 n 一定是个质数
范围不大预处理即可
#include <cstdio>
const int maxn = ;
int pr[maxn], ph[maxn], vis[maxn];
int Case, n, a, b[maxn];
long long ans;
int main() {
for(int i = ;i < maxn;i ++) {
if(!vis[i]) {
pr[++ pr[]] = i;
ph[i] = i - ;
}
for(int j = ;j <= pr[] && i * pr[j] < maxn;j ++) {
vis[i * pr[j]] = ;
if(i % pr[j] == ) break;
}
}
int last = ;
for(int i = ;i;i --) {
b[i] = last;
if(ph[i]) last = i;
}
scanf("%d", &Case);
for(int t = ;t <= Case;t ++) {
ans = ;
scanf("%d", &n);
for(int j = ;j <= n;j ++) {
scanf("%d", &a);
ans += b[a];
}
printf("Case %d: %lld Xukha\n", t, ans);
}
return ;
}
F.
G.求前几位的题目,取个对数就行
利用斐波那契数列的通项公式
我们发现后面减去的一项<1,n变大之后忽略即可
取完对数乘法变加法
最后输出前4位,必须舍去后面位数所以直接double转int
%.0f 会四舍五入并不满足要求
#include <cstdio>
#include <cmath>
#include <algorithm> using std::abs; int n, f[]; int main() {
f[] = ;
for(int i = ;i < ;i ++) f[i] = f[i - ] + f[i - ];
while(scanf("%d", &n) != EOF) {
if(n < ) printf("%d\n", f[n] % );
else {
double x = (log( / sqrt(5.0)) + log((sqrt(5.0) + ) / ) * n) / log(10.0);
x -= (int)x;
x = pow(, x);
while(x < ) x *= ;
printf("%d\n", (int)x);
}
}
return ;
}
H.
I.混进来的放水题目
可以先把每一层节点都挂到上一层的一个节点上
可以出现最多的叶子节点
如果发现太多了就开始把当前层的节点
往上一层的其他节点上挪动,每动一次减少一个叶子节点
#include <cstdio>
const int maxn = ;
int n, t, k, b[maxn], s[maxn], a[maxn];
int main() {
s[] = , s[] = ;
scanf("%d %d %d", &n, &t, &k);
if(k > n - t) {
puts("-1");
return ;
}
for(int j = ,i = ;i <= t;i ++) {
scanf("%d", &a[i]), s[i + ] = a[i] + s[i];
for(int p = ;p <= a[i];p ++, j ++)
b[j] = s[i - ];
}
k = n - t - k;
if(k > ) {
for(int p = ;p <= t;p ++) {
for(int j = s[p - ] + , i = s[p] + ;i < s[p + ] && j < s[p];i ++, j ++) {
b[i] = j, k --;
if(!k) break;
}
if(!k) break;
}
}
if(k) puts("-1");
else {
printf("%d\n", n);
for(int i = ;i <= n;i ++)
printf("%d %d\n", i, b[i]);
}
}
J.快速幂水题
另外因为2009非质数,所以不到50的时候ans已经变成0了
#include <cstdio>
typedef long long ll;
ll n, a[];
ll power(ll x, ll k) {
ll res = ;
for(;k;k >>= , x = x * x % )
if(k & ) res = res * x % ;
return res;
}
int main() {
ll m, ans;
a[] = , a[] = ;
for(int i = ;i < ;i ++)
a[i] = a[i - ] * i % ;
while(scanf("%lld", &n) != EOF) {
ans = ;
m = n / ;
ans *= power(a[], m);
printf("%lld\n", ans * a[n % ] % );
}
return ;
}
bupt summer training for 16 #4 ——数论的更多相关文章
- bupt summer training for 16 #8 ——字符串处理
https://vjudge.net/contest/175596#overview A.设第i次出现的位置左右端点分别为Li,Ri 初始化L0 = 0,则有ans = sum{ (L[i] - L[ ...
- bupt summer training for 16 #7 ——搜索与DP
https://vjudge.net/contest/174962#overview A.我们发现重点在于x,y只要累加就ok了 在每个x上只有上下两种状态,所以可以记忆化搜索 f[0/1][i]表示 ...
- bupt summer training for 16 #6 ——图论
https://vjudge.net/contest/174020 A.100条双向边,每个点最少连2个边 所以最多100个点,点的标号需要离散化 然后要求恰好经过n条路径 快速幂,乘法过程就是flo ...
- bupt summer training for 16 #5 ——数据结构
https://vjudge.net/contest/173780 A.假设 Pt = i,则由Ppi = i得 Ppt = t = Pi 所以就有 if Pt = i then Pi = t #in ...
- bupt summer training for 16 #3 ——构造
https://vjudge.net/contest/172464 后来补题发现这场做的可真他妈傻逼 A.签到傻逼题,自己分情况 #include <cstdio> #include &l ...
- bupt summer training for 16 #2 ——计算几何
https://vjudge.net/contest/171368#overview A.一个签到题,用叉积来判断一个点在一条线的哪个方向 可以二分,数据范围允许暴力 #include <cst ...
- bupt summer training for 16 #1 ——简单题目
D.What a Mess 给n个数,求其中能满足 a[i] % a[j] == 0 的数对之和 n = 1W,max_ai = 100W 不是很大,所以就直接筛就可以了 计算可得最高复杂度 < ...
- 【Android Developers Training】 16. 暂停和恢复一个Activity
注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...
- 【Android Developers Training】 14. 序言:管理Activity生命周期
注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...
随机推荐
- PCB MS SQL 存储过程(CLR) 实现Json转DataTable表的方法
一.准备需转为DataTable的json字符串 原json字符串数据 [{"TechName":"ECN","TechNo":" ...
- PCB 使用第3方网站做为外链图片资源
计划将Web版SI9000阻抗计算工具放外网了,由于个人网站带宽太小原因, 准备将静态的图片资源放在第三方网站,今天找了好几个图床网站,把阻抗模型图上传到图床网站,这样一来就能解决带宽的问题了,今天折 ...
- P4407 [JSOI2009]电子字典
传送门 我的哈希打挂了--然而大佬似乎用哈希可以过还跑得很快-- 删除,枚举删哪个字符,记删之后的哈希值存map 插入,相当于在单词里删字符,去对应的map里查找 更改,相当于两个都删掉同一个位置的字 ...
- Django day30 自定义配置settings文件,分页器,版本控制
一:自定义配置settings文件 1.有两套配置文件,默认配置,用户的配置 2.如果某个字段,用户配置了,就用用户的,如果没配置,就用默认的 二:分页器 1.三种分页: # 普通分页 from re ...
- Akka源码分析-Remote-网络链接生命周期
remote模式下,网络链接的生命周期往往影响着对应Actor的生命周期,那么网络链接的生命周期是怎么样的呢? 每一个与远程系统的链路都是四个状态之一:空闲.活跃.被守护.被隔离.远程系统的某个地址没 ...
- 怎么在windows上安装 ansible How to install ansible to my python at Windows
答案是不能再window上安装,答案如下: It's back! Take the 2018 Developer Survey today » Join Stack Overflow to learn ...
- Offer收割_5
训练 投入 欲望. ---贾森博尔特 第一题:二分枚举答案,check时候模拟一下即可. 时间复杂度: O(n*logn). 第二题: 描述 小Hi在虚拟世界中有一只小宠物小P.小P有K种属性,每种 ...
- docker容器如何安装vim
mv /etc/apt/sources.list /etc/apt/sources.list.bak && \ echo "deb http://mirrors.16 ...
- C#学习-图片的处理
1.在图片上加防伪标记 private void btnAddString_Click(object sender, EventArgs e) { //以流的方式,获取一张图片 using (File ...
- Android常用的Dialog对话框用法
Android的版本有很多通常开发的时候对话框大多数使用自定义或是 Google提供的V4, V7 兼容包来开发保持各个版本的对话框样式统一,所以这里使用的是V7 包里的AlertDialog. im ...