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 枚 ...
随机推荐
- JS——scroll动画
固定导航栏 1.计算导航栏到顶部的距离值 2.当scrollTop值大于这个距离值就添加定位,当小于距离值后解除定位 注意事项:当导航栏添加定位之后,导航栏就脱离了文档流,也就是不占位了,下面的盒子就 ...
- Fast-RCNN论文翻译
http://www.dengfanxin.cn/?p=423 原文地址 本文实现了Fast-RCNN主要部分的翻译工作,在SPPnet出来之后,同在微软的R-CNN的作者Ross迅速怼了回去,抛出了 ...
- Random同时生成多个随机数
贴一个简单示例 public DataTable selectStuInfo() { DataTable dt = new DataTable(); dt.Columns.Add("姓名&q ...
- hint: not have locally. This is usually caused by another repository pushing
git 提交代码前先pull代码,否则会报如下错误 wangju@wangju-HP-348-G4:~/test/reponselogiccheck$ git statusOn branch mast ...
- CPU重要性能参数
内容来自http://www.360doc.com/content/18/1124/15/60810319_796935567.shtml CPU有几个重要的参数:主频.核心.线程.缓存.架构.那么他 ...
- 类 Fabric 主机管理程序开发
类 Fabric 主机管理程序开发:1. 运行程序列出主机组或者主机列表2. 选择指定主机或主机组3. 选择让主机或者主机组执行命令或者向其传输文件(上传/下载4. 充分使用多线程或多进程5. 不同主 ...
- 计蒜客 成绩统计 (Hash表)
链接 : Here! 思路 : 如果用 $STL$ 的 $map$ 或者是使用 $unordered\underline{}map$ 的话是会 $T$ 的, 所以得手写一个 $hash表$. 其实这个 ...
- 洛谷——P1063 能量项链
P1063 能量项链 题目描述 在MarsMars星球上,每个MarsMars人都随身佩带着一串能量项链.在项链上有NN颗能量珠.能量珠是一颗有头标记与尾标记的珠子,这些标记对应着某个正整数.并且,对 ...
- 4 pandas模块,Series类
对gtx图像进行操作,使用numpy知识 如果让gtx这张图片在竖直方向上进行颠倒. 如果让gtx这张图片左右颠倒呢? 如果水平和竖直方向都要颠倒呢? 如果需要将gtx的颜色改变一下呢 ...
- Hibernate 之QBC
转自:http://blog.csdn.net/agromach/article/details/1932290 一.Hibernate 中聚合函数的使用 Criteria接口的Projections ...