bzoj3957
数学+模拟
细节很多
首先我们发现,如果两个区间已经包含,那么可以输出empty,一个数能通过变换得到另一个区间的数,这个区间的大小必须小于等于终点区间的大小。加法不会改变区间大小,只有乘法会改变,而且每次乘法会使区间大小扩大m倍。其实我们发现,最终一个数会变成p*x+y,x是m的几次幂,y是一个a乘上一些m的幂再加上一些a.x=m^l+a(A0*m^l+A1*m^n-1+...+An)所以我们就是要把后面的东西展开。
但是题目要求操作数最少且字典序最小。所以我们要枚举最大的次数,枚举次数时还要保证次数最少。保证次数最少应该更多用乘法保证,所以就是把一些加法换成乘法,也应该尽量用高次代替低次,所以我们每次枚举次数,然后枚举每位,把每位用尽量大的数替代,后面的数清零,然后判断。字典序的判断比较鬼畜。
当m=0或m=1时把m赋成极大值,这样就可以保证m不会乘,impossible就是先把ans变成一个极大值,然后判断。
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<ll, char> PII;
vector<PII> ans;
ll a, m, p, q, r, s;
int kase;
bool cp(vector<PII> &A, vector<PII> &B)
{
ll s1 = , s2 = ;
for(int i = ; i < A.size(); ++i)
s1 += A[i].first;
for(int i = ; i < B.size(); ++i)
s2 += B[i].first;
// printf("%d %d\n", A.size(), B.size());
if(s1 != s2)
return s1 < s2;
int lim = min(A.size(), B.size());
for(int i = ; i < lim; ++i)
{
if(A[i].second != B[i].second)
return A[i].second < B[i].second;
if(A[i].first != B[i].first)
{
if(A[i].second == 'A')
return A[i].first > B[i].first;
else
return A[i].first < B[i].first;
}
}
return A.size() < B.size();
}
void update(int x, ll target)
{
// printf("x=%d target=%d\n", x, target);
vector<PII> v; v.clear();
for(int i = ; i < x; ++i)
{
if(target % m)
v.push_back(make_pair(target % m, 'A'));
target /= m;
if(v.empty() || v.back().second == 'A')
v.push_back(make_pair(1ll, 'M'));
else
++v.back().first;
}
if(target)
v.push_back(make_pair(target, 'A'));
reverse(v.begin(), v.end());
// for(int i = 0; i < v.size(); ++i)
// printf(" %d%c", v[i].first, v[i].second);
// printf("\n");
if(cp(v, ans))
swap(ans, v);
}
void process(int low, int high, int m, int t)
{
for(int i = , j = ; i <= t; ++i, j *= m)
{
int x = (low + j - ) / j * j;
if(x > high)
break;
update(t, x);
}
}
int main()
{
// freopen("output.txt", "w", stdout);
while(scanf("%lld%lld%lld%lld%lld%lld", &a, &m, &p, &q, &r, &s))
{
if(!a && !m && !p && !q && !s && !r)
break;
ans.clear();
if(m == || m == )
m = 1000000010ll;
printf("Case %d:", ++kase);
if(p >= r && q <= s)
{
printf(" empty\n");
continue;
}
ans.push_back(make_pair(2000000000ll, 'A'));
for(int mul = ; q <= s && q - p <= s - r; ++mul, p *= m, q *= m)
{
ll minadd = max(0ll, (r - p + a - ) / a), maxadd = (s - q) / a;
if(minadd > maxadd)
continue;
process(minadd, maxadd, m, mul); //mul:最多乘mul次
}
if(ans[].first == 2000000000ll)
{
printf(" impossible\n");
continue;
}
for(int i = ; i < ans.size(); ++i)
printf(" %lld%c", ans[i].first, ans[i].second);
printf("\n");
}
// fclose(stdout);
return ;
}
bzoj3957的更多相关文章
- bzoj3957: [WF2011]To Add or to Multiply
gay队牛逼! 我们可以强行拆一下柿子,最终得到的值会是m^k*x+m^k*u(k)*a+m^k-1*u(k-1)*a……m^0*u(0)*a 其中u表示后面有i个m的a有多少个 答案就是k+∑u 枚 ...
随机推荐
- log4net 局部代码 看不懂....
public interface ILogger {} public interface ILoggerWrapper { ILogger Logger {get;} } public interfa ...
- css知识框架
- CAD在网页中打印的图纸里面添加页眉及页脚
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 3 ...
- c++ map迭代器
#include <stdio.h> #include <map> using namespace std; int main(){ map<int, int> m ...
- [Luogu] P1233 木棍加工
题目描述 一堆木头棍子共有n根,每根棍子的长度和宽度都是已知的.棍子可以被一台机器一个接一个地加工.机器处理一根棍子之前需要准备时间.准备时间是这样定义的: 第一根棍子的准备时间为1分钟: 如果刚处理 ...
- JAVA正则表达式matcher.find()和 matcher.matches()的区别
1.find()方法是部分匹配,是查找输入串中与模式匹配的子串,如果该匹配的串有组还可以使用group()函数.matches()是全部匹配,是将整个输入串与模式匹配,如果要验证一个输入的数据是否为数 ...
- Git 基础教程 之 别名
配置别名, 例如: git config --global alias.st status git config ...
- 第五节:web爬虫之urllib(一)
一.urllib.request.urlopen(url, data=None, [timeout, ]*, cafile=None, capath=None, cadefault=False, ...
- 《hello-world》第八次团队作业:Alpha冲刺-Scrum Meeting 5
项目 内容 这个作业属于哪个课程 2016级计算机科学与工程学院软件工程(西北师范大学) 这个作业的要求在哪里 实验十二 团队作业8:软件测试与Alpha冲刺 团队名称 <hello--worl ...
- MINSUB - Largest Submatrix
MINSUB - Largest Submatrix no tags You are given an matrix M (consisting of nonnegative integers) a ...