关于同余最短路的部分 【同余最短路】P3403跳楼机/P2371墨墨的等式

【P2662牛场围栏】

题目背景

小L通过泥萌的帮助,成功解决了二叉树的修改问题,并因此写了一篇论文,

成功报送了叉院(羡慕不?)。勤奋又勤思的他在研究生时期成功转系,考入了北京大学光华管理学院!毕业后,凭着自己积累下的浓厚经济学与计算机学的基础,成功建设了一个现代化奶牛场!

题目描述

奶牛们十分聪明,于是在牛场建围栏时打算和小L斗智斗勇!小L有N种可以建造围栏的木料,长度分别是l1,l2 … lN,每种长度的木料无限。

修建时,他将把所有选中的木料拼接在一起,因此围栏的长度就是他使用的木料长度之和。但是聪明的小L很快发现很多长度都是不能由这些木料长度相加得到的,于是决定在必要的时候把这些木料砍掉一部分以后再使用。

不过由于小L比较节约,他给自己规定:任何一根木料最多只能削短M米。当然,每根木料削去的木料长度不需要都一样。不过由于测量工具太原始,小L只能准确的削去整数米的木料,因此,如果他有两种长度分别是7和11的木料,每根最多只能砍掉1米,那么实际上就有4种可以使用的木料长度,分别是6, 7,10, 11。

因为小L相信自己的奶牛举世无双,于是让他们自己设计围栏。奶牛们不愿意自己和同伴在游戏时受到围栏的限制,于是想刁难一下小L,希望小L的木料无论经过怎样的加工,长度之和都不可能得到他们设计的围栏总长度。不过小L知道,如果围栏的长度太小,小L很快就能发现它是不能修建好的。因此她希望得到你的帮助,找出无法修建的最大围栏长度。

这一定难不倒聪明的你吧!如果你能帮小L解决这个问题,也许他会把最后的资产分给你1/8哦!

输入输出格式

输入格式:

输入的第一行包含两个整数N,  M,分别表示木料的种类和每根木料削去的最大值。以下各行每行一个整数li(1< li< 3000),表示第i根木料的原始长度。

输出格式:

输出仅一行,包含一个整数,表示不能修建的最大围栏长度。如果任何长度的围栏都可以修建或者这个最大值不存在,输出-1。

输入输出样例

输入样例#1:

  1. 2 1
  2. 7 11
输出样例#1:

  1. 15

说明

40 % :1< N< 10,  0< M< 300

100 % :1< N< 100,  0< M< 3000

思路

关于同余最短路的东西已经写过了,这里直接本题相关。

依然是由一些数字去凑一个数字的剩余系,这道题由于多了M的条件,需要先暴力把所有能用的数字求出来,并让其中最小的那个成为提供剩余系的x。

然后跑一遍所有能用的数字,和x的剩余系建边,最后跑最短路。

求不能凑出的最大数的时候,我们要先考虑d数组的意义。d[i]即为其他数字能凑出来的%x=i的最小数字,那么d[i]+x,d[i]+2x,d[i]+3x... d[i]以上跳所有个x都能达到。

那么%x=i的数字,最大凑不出来的就是d[i]-x。

那么很显然了,把所有的d[i]-x求出来,取其中最大值。

但是还有要注意的地方,本题存在输出-1的要求。其中一种输出-1的情况是没有凑不出来的数,那么当我们能用的木料中存在长为1的,自然就能达到所有的长度。

另一种情况是不存在这个最大值。这里有两种考虑方向,一种是求出所有数字的gcd,若其不等于1,自然有一系列没法凑出来的数字。因为能凑出来的数字一定是这个gcd的倍数,其不为一的时候必然存在凑不出来的空缺。还有一种方法是最后找ans的时候顺便看一下是否有d[i]>max(max是给d数组跑最短路前设的最大值),如果有,那么一定存在一串%x=i的数字都凑不出来。(其实这种做法大概相当于猜测利用了数据比较小)

代码:

  1. #include<iostream>
  2. #include<cstdio>
  3. #include<cstring>
  4. #include<queue>
  5. #include<algorithm>
  6. using namespace std;
  7. int n,m;
  8. int x,a[];
  9. int ver[],Next[],head[],edge[],tot,d[],vis[],ans=;
  10. queue<int>q;
  11. void add(int x,int y,int z){
  12. ver[++tot]=y;
  13. Next[tot]=head[x];
  14. edge[tot]=z;
  15. head[x]=tot;
  16. }
  17. int main()
  18. {
  19. scanf("%d%d",&n,&m);
  20. for(int i=;i<=n;i++){
  21. scanf("%d",&a[i]);
  22. }
  23. sort(a+,a+n+);
  24. x=max(,a[]-m);
  25. if(x==){
  26. printf("-1");
  27. return ;
  28. }
  29. for(int i=;i<=n;i++){
  30. for(int j=max(a[i-]+,a[i]-m);j<=a[i];j++){
  31. if(j!=x){
  32. for(int k=;k<x;k++){
  33. add(k,(k+j)%x,j);
  34. }
  35. }
  36. }
  37. }
  38. memset(d,0x3f,sizeof(d));
  39. d[]=;
  40. q.push();
  41. while(!q.empty()){
  42. int u=q.front();
  43. q.pop();
  44. vis[u]=;
  45. for(int i=head[u];i;i=Next[i]){
  46. int v=ver[i],z=edge[i];
  47. if(d[v]>d[u]+z){
  48. d[v]=d[u]+z;
  49. if(!vis[v]){
  50. vis[v]=;
  51. q.push(v);
  52. }
  53. }
  54. }
  55. }
  56. d[x]=;
  57. for(int i=;i<x;i++){
  58. // printf("%d %d\n",i,d[i]);
  59. if(d[i]>){
  60. printf("-1");
  61. return ;
  62. }
  63. ans=max(ans,d[i]-x);
  64. }
  65. printf("%d",ans);
  66. return ;
  67. }

以上。

【同余最短路】洛谷 P2662 牛场围栏的更多相关文章

  1. 洛谷 P2662 牛场围栏

    做法是这样的: 首先暴力把所有可能的边长搞出来..(当然<=0的不要) 排序边长+去重, 当且仅当可行边长里面有1时,任何长度都能取到,输出-1 当且仅当所有可行边长的gcd大于1时,不能取到的 ...

  2. luoguP3951 小凯的疑惑/P2662 牛场围栏

    其实就是当年sxy给我讲的墨墨的等式,只是当时比较菜听得似懂非懂. 小凯的疑惑 去年noipday1t1,当时随便猜了个结论结果猜对了,现在瞎证一下,答案是a*b-a-b. 设a为a,b中较小的一个, ...

  3. 最短路洛谷P2384

    题目背景 狗哥做烂了最短路,突然机智的考了Bosh一道,没想到把Bosh考住了...你能帮Bosh解决吗? 他会给你100000000000000000000000000000000000%10金币w ...

  4. AC日记——最短路 洛谷 P2384

    题目背景 狗哥做烂了最短路,突然机智的考了Bosh一道,没想到把Bosh考住了...你能帮Bosh解决吗? 他会给你100000000000000000000000000000000000%10金币w ...

  5. P2384 最短路 洛谷

    https://www.luogu.org/problem/show?pid=2384 题目背景 狗哥做烂了最短路,突然机智的考了Bosh一道,没想到把Bosh考住了...你能帮Bosh解决吗? 他会 ...

  6. luogu P2662 牛场围栏

    传送门 因为一个木板可以切掉最多\(m\),所以可以先预处理哪些长度的木板可用,开个桶,然后对\([l-m,l]\)打标记,再把打了标记的数取出来 假设可用长度\(a_1,a_2,,,a_n\)从小到 ...

  7. LG2662 牛场围栏 和 test20181107 数学题

    P2662 牛场围栏 题目背景 小L通过泥萌的帮助,成功解决了二叉树的修改问题,并因此写了一篇论文, 成功报送了叉院(羡慕不?).勤奋又勤思的他在研究生时期成功转系,考入了北京大学光华管理学院!毕业后 ...

  8. 【同余最短路】【例题集合】洛谷P3403 跳楼机/P2371 墨墨的等式

    接触到的新内容,[同余最短路]. 代码很好写,但思路不好理解. 同余最短路,并不是用同余来跑最短路,而是通过同余来构造某些状态,从而达到优化时间空间复杂度的目的.往往这些状态就是最短路中的点,可以类比 ...

  9. 洛谷P3403跳楼机(最短路构造/同余最短路)

    题目-> 解题思路: 最短路构造很神啊. 先用前两个值跑在第三个值模意义下的同余最短路(这步贪心可以证明,如果第三步长为z,那么如果n+z可以达到,n+2z同样可以达到) 最后计算与楼顶差多少个 ...

随机推荐

  1. Angular 监听滚动条事件

    一.引用fromEvent import { fromEvent } from 'rxjs'; 二.调用fromEvent this.subscribeScoll = fromEvent(window ...

  2. (转)第02节:在Canvas上画简单的图形

    我们现在已经可以在HTML中使用Fabric.js库了,那这节我们就详细的学习一下如何在canvas上画出简单的图形. 在画东西之前我们需要了解画任何东西的基本三个步骤: 声明画布(canvas),用 ...

  3. VS2010-MFC(VS2010应用程序工程中文件的组成结构)

    转自:http://www.jizhuomi.com/software/143.html 用应用程序向导生成框架程序后,我们可以在之前设置的Location下看到以解决方案名命名的文件夹,此文件夹中包 ...

  4. IoC深入理解

    1. IoC理论的背景 我们都知道,在采用面向对象方法设计的软件系统中,它的底层实现都是由N个对象组成的,所有的对象通过彼此的合作,最终实现系统的业务逻辑. 图1:软件系统中耦合的对象 如果我们打开机 ...

  5. Poj 2559 最大矩形面积 v单调栈 分类: Brush Mode 2014-11-13 20:48 81人阅读 评论(0) 收藏

    #include<iostream> #include<stack> #include<stdio.h> using namespace std; struct n ...

  6. java最大余数法(百分比算法Echarts)

    最近工作中使用Echarts开发报表的时候遇到了这样的一个问题,需求是一个div中左边是一个环形图表,右边是一个表格,表格中展示图表中每个类别占用的百分比.存在的问题:1.当存在四舍五入的时候,Ech ...

  7. <scrapy爬虫>爬取校花信息及图片

    1.创建scrapy项目 dos窗口输入: scrapy startproject xiaohuar cd xiaohuar 2.编写item.py文件(相当于编写模板,需要爬取的数据在这里定义) # ...

  8. sqlserver存储过程事务回滚

    set ANSI_NULLS ON set QUOTED_IDENTIFIER ON go ALTER PROCEDURE [dbo].[AddUserOnChannel] ), ), @Channe ...

  9. elasticsearch 中文API 记数(八)

    计数API 计数API允许开发者简单的执行一个查询,返回和查询条件相匹配的文档的总数.它可以跨多个索引以及跨多个类型执行. import static org.elasticsearch.index. ...

  10. Windows API 第四篇 文件操作

    创建或打开文件(也可用于打开管道,油槽,硬件设备等): HANDLE CreateFile( LPCTSTR lpFileName, // file name DWORD dwDesiredAcces ...