题面

题解

毒瘤题浪费我大好青春

容易知道, 如果\(l\)和\(r\)位数不一样, 直接选形似\(99..99\)的数, 输出答案即可

\(l\)和\(r\)位数一样的话, 当位数确定的时候, 由于已知\(p\), 所以每一位要跟\(k\)中哪些位相乘已经确定了

设\(f[i]\)为\(k\)中与\(x\)中第\(i\)位相乘的数的和

首先, 每\(lcm(lenx, lenk)\)一个周期, 这个直接暴力算即可

//考虑到x中mod d(lenx与lenk的gcd)同余的数在一个周期内f相等(自己举几个特例, 我不会证)
//故只用枚举mod d的余数即可, 然后模拟选的过程
//假设这个数这一次被选是t时间, p位置, 那么它下一次被选就是(t + lenx)时间, (p + lenx)%lenk位置, 由于我的数组是从1开始, 所以是(p + lenx - 1)%lenk + 1
//然后由于1~lenk中每个数只会被一个i跳到(自己画图, 或者感性理解一下:如果跳到了说明这个周期会更小, 矛盾)
//给每个数标记他被i所跳到的时间(这个等会会用到) 也就是pos, 再给每个位置标记上他被i跳到时累加的值, 也就是sum
for(int i = 0; i < d; i++)
{
int num = 0, poss = 1, j = i + 1;
do
{
num += k[j];
pos[j] = poss;
sum[j] = num;
++poss;
j = ((j + l[0] - 1) % k[0] + k[0]) % k[0] + 1;
}
while(j != i + 1);
g[i] = num;
}

这样就可以计算出整个周期的\(f[]\)的值, 然后我们看怎么计算不是整周期的\(f[]\)的值

由于每个位置跳了多少次是可以算出来的, 所以我们可以算出来它最后在哪一个位置

然后用终点的\(sum\)减去起点的\(sum\)即可

但是有一个细节, 就是当终点的\(pos\)小于等于起点的\(pos\)时, 它要加上一个\(g[i \% d]\), 根据图理解一下吧, 我讲不清

这个图太小了吧...

算了, 蓝色点是终点, 绿色点是起点, 可以知道起点到终点是那条红色路径, 然后你如果用终点的\(sum\)减去起点的\(sum\)就是负的绿色的那条, 加上一个\(g[i \% d]\)就是那条红色的了

最后讲一下统计答案

考虑贪心

对于一个数x, 它的前\(i\)位跟\(r\)一样, 第\(i + 1\)位小于\(r[i + 1]\), 后面的都是9, 这样必定是最优的, 从后往前不断更改这个\(i + 1\)的位置即可

Code

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdlib>
#include <cstdio>
#include <vector>
#define itn int
#define reaD read
#define N 100005
#define int long long
using namespace std; int l[N], r[N], k[N], pos[N];
long long p, res1, res2, ans, g[N], sum[N], f[N]; template < typename T >
inline T read()
{
T x = 0, w = 1; char c = getchar();
while(c < '0' || c > '9') { if (c == '-') w = -1; c = getchar(); }
while(c >= '0' && c <= '9') { x = x * 10 + c - '0'; c = getchar(); }
return x * w;
} int gcd(int m, int n) { return !n ? m : gcd(n, m % n); } signed main()
{
char c = getchar();
while(c < '0' || c > '9') c = getchar();
while(c >= '0' && c <= '9') l[++l[0]] = c - '0', c = getchar();
while(c < '0' || c > '9') c = getchar();
while(c >= '0' && c <= '9') r[++r[0]] = c - '0', c = getchar();
while(c < '0' || c > '9') c = getchar();
while(c >= '0' && c <= '9') k[++k[0]] = c - '0', c = getchar();
p = read <long long> ();
if(l[0] < r[0])
{
int r = p % k[0], q = p / k[0];
for(int i = 1; i <= k[0]; i++)
ans += k[i] * (q + (i <= r)) * 9;
printf("%lld\n", ans); return 0;
}
int d = gcd(l[0], k[0]);
for(int i = 0; i < d; i++)
{
int num = 0, poss = 1, j = i + 1;
do
{
num += k[j];
pos[j] = poss;
sum[j] = num;
++poss;
j = ((j + l[0] - 1) % k[0] + k[0]) % k[0] + 1;
}
while(j != i + 1);
g[i] = num;
}
int q = p / (l[0] * k[0] / d), x = p % (l[0] * k[0] / d), w = x / l[0], y = x % l[0];
for(int i = 1; i <= l[0]; i++)
{
f[i] = g[((i - 1) % d + d) % d] * q;
int cnt = w + (i <= y);
if(!cnt) continue;
int lst = ((i - l[0] - 1) % k[0] + k[0]) % k[0] + 1, nxt = ((i + (cnt - 1) * l[0] - 1) % k[0] + k[0]) % k[0] + 1;
f[i] += sum[nxt] - sum[lst] + (pos[nxt] <= pos[lst] ? g[((i - 1) % d + d) % d] : 0);
}
int top = 1;
while(top <= l[0] && l[top] == r[top]) top++;
for(int i = 1; i <= l[0]; i++)
res1 += r[i] * f[i];
ans = res1; res1 -= f[l[0]];
for(int i = l[0]; i >= top + 1; i--)
{
res1 += (10 - r[i]) * f[i];
res1 -= f[i - 1];
ans = max(ans, res1);
}
printf("%lld\n", ans);
return 0;
}

[51nod1666] 最大值的更多相关文章

  1. C语言 · 最大值与最小值计算

    输入11个整数,计算它们的最大值和最小值. 样例输入 0 1 2 3 4 5 6 7 8 9 10 样例输出 10 0   #include<stdio.h> int main(){ ]; ...

  2. [LeetCode] Split Array Largest Sum 分割数组的最大值

    Given an array which consists of non-negative integers and an integer m, you can split the array int ...

  3. [LeetCode] Sliding Window Maximum 滑动窗口最大值

    Given an array nums, there is a sliding window of size k which is moving from the very left of the a ...

  4. 整理:Javascript获取数组中的最大值和最小值的方法汇总

    方法一: ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 //最小值 Array.prototype.min = function ...

  5. HDU 1754 I Hate It 线段树单点更新求最大值

    题目链接 线段树入门题,线段树单点更新求最大值问题. #include <iostream> #include <cstdio> #include <cmath> ...

  6. Poj The xor-longest Path 经典题 Trie求n个数中任意两个异或最大值

    Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 5646   Accepted: 1226 Description In an ...

  7. 不需要sql进行计算数据的平均值、最大值、最小值、和

    介绍下SqlServer.前端js.后台C#三个阶段对均值.最大值.最小值.和计算int[] jisuan = {0, 1, 3, 5, 7,8 }; List<int> jisuan2 ...

  8. 在一个SQL Server表中的多个列找出最大值

    在一个SQL Server表中一行的多个列找出最大值 有时候我们需要从多个相同的列里(这些列的数据类型相同)找出最大的那个值,并显示 这里给出一个例子 IF (OBJECT_ID('tempdb..# ...

  9. C语言 · 寻找数组中的最大值

    问题描述 对于给定整数数组a[],寻找其中最大值,并返回下标. 输入格式 整数数组a[],数组元素个数小于1等于100.输出数据分作两行:第一行只有一个数,表示数组元素个数:第二行为数组的各个元素. ...

随机推荐

  1. Dual 表

    我们先从名称来说,dual不是缩写词,本身就是完整的单词.dual名词意思是对数,做形容词时是指二重的,二元的. Oracle中的dual表是一个单行单列的虚拟表. Dual表是oracle与数据字典 ...

  2. IOS 改变UISearchBar的背景色

    之前网上提供的方法试了很多种  都不能很好的去掉背景色  ,修改背景色方法如下: searchbar.barStyle = UIBarStyleBlackTranslucent; searchbar. ...

  3. CentOS系统 Amoeba+MySql主从读写分离配置 适合新手傻瓜式教程!-----仅供参考!

    废话不说,直接开始: 一.安装mysql的三种方式,这里采用第2种(安装方式不再详解,请参照) http://www.cnblogs.com/babywaa/articles/4837946.html ...

  4. C和指针--动态内存分配

    1.为什么需要使用动态内存分配 数组的元素存储于内存中连续的位置上,当一个数组被声明时,它所需要的内存在编译时就被分配.当你声明数组时,必须用一个编译时常量指定数组的长度.但是,数组的长度常常在运行时 ...

  5. Visual Studio中使用GitHub

    一.简介 GitHub被微软收购后免费开放了私人仓库,这对个人或小型团队来说简直是福音.GitHub高效免费,无论是团队代码管理还是DevOps都不再话下,特别是对于想要开放研究成果提高知名度的团队或 ...

  6. sql 连接数查询

    SELECT * FROM master.dbo.sysprocesses WHERE dbid IN ( SELECT dbid FROM master.dbo.sysdatabases WHERE ...

  7. 算法9-----输出全排列(递归)---移除K个数,剩下最小数。

    1.题目:给定一个字符串,输出所有的字典序. 如: 输入字符串:'ac',输出:['ac','ca'] 输入字符串:‘abc' ,输出:['abc','acb','bac','bca','cab',' ...

  8. iview 如何在表格中给操作图标添加Tooltip文字提示?

    项目需要用到的iview 表格中操作项目有各种各样的图标,而各种各样的图标代表不同的操作,面对新用户可能很懵,那如何给这些图标添加Tooltip文字提示? 废话不多讲,直接看代码: <templ ...

  9. Navicat Premium 12 安装 与 激活

    官方简体中文下载网址:https://www.navicat.com.cn/download/navicat-premium 安装的过程....(都是中文)所以略过. 开始破解....( ...... ...

  10. java Timer和TimerTask(简单的使用)

    Timer 是一个定时工具 TimerTask 是一个实现了Runnable接口抽象类,代表可以被Timer执行的任务 (1)Timer.schedule(TimerTask task,Date ti ...