Aaronson
Accepts: 607
Submissions: 1869
Time Limit: 4000/2000 MS (Java/Others)
Memory Limit: 131072/131072 K (Java/Others)
问题描述
给出一个不定方程x0+2x1+4x2+...+2mxm=nx_{0}+2x_{1}+4x_{2}+...+2^{m}x_{m}=nx0+2x1+4x2+...+2mxm=n, 找出一组解(x0,x1,x2,...,xm)(x_0,x_1,x_2,...,x_m)(x0,x1,x2,...,xm), 使得∑i=0mxidisplaystylesum_{i=0}^{m} x_ii=0∑mxi最小, 并且每个xix_ixi (0≤i≤m0 le i le m0≤i≤m)都是非负的.
输入描述
输入包含多组数据, 第一行包含一个整数TTT (1≤T≤105)(1 le T le 10^5)(1≤T≤105)表示测试数据组数. 对于每组数据: 第一行包含两个整数nnn和mmm (0≤n,m≤109)(0 le n,m le 10^9)(0≤n,m≤109).
输出描述
对于每组数据, 输出∑i=0mxidisplaystylesum_{i=0}^{m} x_ii=0∑mxi的最小值.
输入样例
10 1 2 3 2 5 2 10 2 10 3 10 4 13 5 20 4 11 11 12 3
输出样例
1 2 2 3 2 2 3 2 3 2
分析:
就是把n二进制分解,只不过有m这个限制。
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<string>
#include<map>
#include<queue>
#include<vector>
using namespace std;
typedef long long ll;
const int N=1e5+9;
int a[N];
int num(int x,int m)
{
int t=0;
while(x){
if(x&1)a[t++]=1;
else a[t++]=0;
x>>=1;
}
int i=0,sum=0;
for(i=0;i<t&&i<=m;i++){
sum+=a[i];
}
int w=2;
for(;i<t;i++){
sum+=a[i]*w;
w<<=1;
}
return sum;
}
int main()
{
//freopen("f.txt","r",stdin);
int T;scanf("%d",&T);
int n,m;
while(T--){
scanf("%d%d",&n,&m);
printf("%d
",num(n,m));
}
return 0;
}
Bellovin
Accepts: 428
Submissions: 1685
Time Limit: 6000/3000 MS (Java/Others)
Memory Limit: 131072/131072 K (Java/Others)
问题描述
Peter有一个序列a1,a2,...,ana_1,a_2,...,a_na1,a2,...,an. 定义F(a1,a2,...,an)=(f1,f2,...,fn)F(a_1,a_2,...,a_n)=(f_1,f_2,...,f_n)F(a1,a2,...,an)=(f1,f2,...,fn), 其中fif_ifi是以aia_iai结尾的最长上升子序列的长度. Peter想要找到另一个序列b1,b2,...,bnb_1,b_2,...,b_nb1,b2,...,bn使得F(a1,a2,...,an)F(a_1,a_2,...,a_n)F(a1,a2,...,an)和F(b1,b2,...,bn)F(b_1,b_2,...,b_n)F(b1,b2,...,bn)相同. 对于所有可行的正整数序列, Peter想要那个字典序最小的序列. 序列a1,a2,...,ana_1, a_2, ..., a_na1,a2,...,an比b1,b2,...,bnb_1, b_2, ..., b_nb1,b2,...,bn字典序小, 当且仅当存在一个正整数iii (1≤i≤n)(1 le i le n)(1≤i≤n)满足对于所有的kkk (1≤k<i)(1 le k < i)(1≤k<i)都有ak=bka_k = b_kak=bk并且ai<bia_i < b_iai<bi.
输入描述
输入包含多组数据, 第一行包含一个整数TTT表示测试数据组数. 对于每组数据: 第一行包含一个整数nnn (1≤n≤100000)(1 le n le 100000)(1≤n≤100000)表示序列的长度. 第二行包含nnn个整数a1,a2,...,ana_1,a_2,...,a_na1,a2,...,an (1≤ai≤109)(1 le a_i le 10^9)(1≤ai≤109).
输出描述
对于每组数据, 输出nnn个整数b1,b2,...,bnb_1,b_2,...,b_nb1,b2,...,bn (1≤bi≤109)(1 le b_i le 10^9)(1≤bi≤109)表示那个字典序最小的序列.
输入样例
3 1 10 5 5 4 3 2 1 3 1 3 5
输出样例
1 1 1 1 1 1 1 2 3分析:
试几组数据就能发现要求的就是LIS
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<string>
#include<map>
#include<queue>
#include<vector>
using namespace std;
const int N=100000+9;
int a[N],d[N],g[N],n;
void LIS() //Nlog(N)
{
for(int i=1;i<=n;i++)g[i]=INF;
for(int i=0;i<n;i++){
int k=lower_bound(g+1,g+1+n,a[i])-g;
d[i]=k;
g[k]=a[i];
}
}
int main()
{
//freopen("f.txt","r",stdin);
int T;scanf("%d",&T);
while(T--){
scanf("%d",&n);
for(int i=0;i<n;i++)scanf("%d",&a[i]);
LIS();
for(int i=0;i<n-1;i++)printf("%d ",d[i]);
printf("%d
",d[n-1]);
}
return 0;
}
1003 留坑待补
Dertouzos
Accepts: 76
Submissions: 1357
Time Limit: 7000/3500 MS (Java/Others)
Memory Limit: 131072/131072 K (Java/Others)
问题描述
正整数xxx称为nnn的positive proper divisor, 当且仅当x∣nx | nx∣n并且1≤x<n1 le x < n1≤x<n. 例如, 1, 2, 和3是6的positive proper divisor, 但是6不是. Peter给你两个正整数nnn和ddd. 他想要知道有多少小于nnn的整数, 满足他们的最大positive proper divisor恰好是ddd.
输入描述
输入包含多组数据, 第一行包含一个整数TTT (1≤T≤106)(1 le T le 10^6)(1≤T≤106)表示测试数据组数. 对于每组数据: 第一行包含两个整数nnn和ddd (2≤n,d≤109)(2 le n, d le 10^9)(2≤n,d≤109).
输出描述
对于每组数据, 输出一个整数.
输入样例
9 10 2 10 3 10 4 10 5 10 6 10 7 10 8 10 9 100 13
输出样例
1 2 1 0 0 0 0 0 4题解:
随便推导下, 令y=xd, 如果ddd是yyy的maximum positive proper divisor, 显然要求x是y的最小质因子. 令mp(n)表示nnn的最小质因子, 那么就有x le mp(d)x≤mp(d), 同时有y < ny<n, 那么x≤⌊n−1d⌋x le lfloor frac{n-1}{d} floor于是就是计算有多少个素数xxx满足x≤min{mp(d),⌊n−1d⌋}x le min{mp(d), lfloor frac{n-1}{d} floor}
当d比较大的时候, ⌊n−1d⌋lfloor frac{n-1}{d} floor比较小, 暴力枚举xxx即可. 当ddd比较小的时候, 可以直接预处理出答案. 阈值设置到106∼107都可以过.
分析:自己写的时候超时,不知道怎么搞的,后来改了改过了,郁闷!
还可以像题解一样预处理出答案。
#include<iostream>
#include<algorithm>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<string>
#include<map>
#include<queue>
#include<vector>
using namespace std;
const int N=1000000+9;
int n,d;
int p[N],cnt,g[N];
bool np[N];
void init()
{
memset(np,0,sizeof(np));
cnt=0;
for(int i=2;i<N;i++){
if(!np[i]){
p[cnt++]=i;
g[i]=i;
}
for(int j=0;j<cnt&&i*p[j]<N;j++){
np[i*p[j]]=1;
g[i*p[j]]=p[j];
if(i%p[j]==0)break;
}
}
}
int cal(int x)
{
int t=upper_bound(p,p+cnt,x)-p;
return t;
}
int main()
{
//freopen("f.txt","r",stdin);
init();
int T;scanf("%d",&T);
while(T--){
scanf("%d%d",&n,&d);
n--;
if(n<d)printf("0
");
else if(d>=N){
int num=n/d,t=-1;
for(int i=0;p[i]<=num;i++)if(d%p[i]==0){
t=p[i];break;
}
if(t==-1)printf("%d
",cal(num));
else printf("%d
",cal(t));
}
else{
int x=min(n/d,g[d]);
printf("%d
",cal(x));
}
}
return 0;
}