热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

LuoguP1948[USACO08JAN]电话线TelephoneLines(最短路+dp)

P1948[USACO08JAN]电话线TelephoneLines题意题目描述FarmerJohnwantstosetupatelephonelineathis

P1948 [USACO08JAN]电话线Telephone Lines

题意

题目描述

Farmer John wants to set up a telephone line at his farm. Unfortunately, the phone company is uncooperative, so he needs to pay for some of the cables required to connect his farm to the phone system.

There are \(N(1 \leq N \leq 1,000)\) forlorn telephone poles conveniently numbered \(1 \cdots N\) that are scattered around Farmer John's property; no cables connect any them. A total of \(P(1 \leq P \leq 10,000)\) pairs of poles can be connected by a cable; the rest are too far apart.

The i-th cable can connect the two distinct poles \(A_i\) and \(B_i\), with length \(L_i(1 \leq L_i \leq 1,000,000)\) units if used. The input data set never names any \(\{ Ai, Bi \}\) pair more than once. Pole \(1\) is already connected to the phone system, and pole \(N\) is at the farm. Poles \(1\) and \(N\) need to be connected by a path of cables; the rest of the poles might be used or might not be used.

As it turns out, the phone company is willing to provide Farmer John with \(K(0 \leq K lengths of cable for free. Beyond that he will have to pay a price equal to the length of the longest remaining cable he requires (each pair of poles is connected with a separate cable), or \(0\) if he does not need any additional cables.

Determine the minimum amount that Farmer John must pay.

多年以后,笨笨长大了,成为了电话线布置师。由于地震使得某市的电话线全部损坏,笨笨是负责接到震中市的负责人。该市周围分布着\(N(1 \leq N \leq 1000)\)根据\(1 \cdots n\)顺序编号的废弃的电话线杆,任意两根线杆之间没有电话线连接,一共有\(p(1 \leq p \leq 10000)\)对电话杆可以拉电话线。其他的由于地震使得无法连接。

\(i\)对电线杆的两个端点分别是\(a_i,b_i\),它们的距离为\(l_i(1 \leq l_i \leq 1000000)\)。数据中每对\((a_i,b_i)\)只出现一次。编号为\(1\)的电话杆已经接入了全国的电话网络,整个市的电话线全都连到了编号\(N\)的电话线杆上。也就是说,笨笨的任务仅仅是找一条将\(1\)号和\(N\)号电线杆连起来的路径,其余的电话杆并不一定要连入电话网络。

电信公司决定支援灾区免费为此市连接\(k\)对由笨笨指定的电话线杆,对于此外的那些电话线,需要为它们付费,总费用决定于其中最长的电话线的长度(每根电话线仅连接一对电话线杆)。如果需要连接的电话线杆不超过\(k\)对,那么支出为\(0\)

请你计算一下,将电话线引导震中市最少需要在电话线上花多少钱?

输入输出格式

输入格式:

输入文件的第一行包含三个数字\(n,p,k\);

第二行到第\(p+1\)行,每行分别都为三个整数\(a_i,b_i,l_i\)

输出格式:

一个整数,表示该项工程的最小支出,如果不可能完成则输出\(-1\)

输入输出样例

输入样例:

5 7 1
1 2 5
3 1 4
2 4 8
3 2 3
5 2 9
3 4 7
4 5 6

输出样例:

4

思路

\(solo\)! --Mercury

\(solo\)又赢了水星 祭。

这道题水星打的二分,直接二分答案,确实很简单(然而他现在还在调试)。在这里给一个动态规划的思路。

在一般的最短路中,我们用数组\(dis[v]\)表示起点到点\(v\)的最短路。我们把这个数组再开一维,用\(dis[v][i]\)表示已经消除了\(i\)条边的价格之后到达\(v\)点的最短路。那么在松弛一条边\((u,v)\)的时候我们就有以下转移策略:

  • 直接转移到下一个点,也就是\(dis[v][i]=max( \text{那条边的长度},dis[u][i])\)
  • 这条边的价格消除为零,也就是\(dis[v][i+1]=max( \text{那条边的长度},dis[u][i])\)

然后我们再套上一个\(SPFA\)或者\(Dijkstra\)就能过了。

AC代码
#include
using namespace std;
typedef pair PII;
const int MAXN=1005;
const int MAXM=20005;
int n,m,k,f[MAXN][MAXN];
int cnt,top[MAXN],to[MAXM],len[MAXM],nex[MAXM];
bool vis[MAXN][MAXN];
int read()
{
    int re=0;
    char ch=getchar();
    while(!isdigit(ch)) ch=getchar();
    while(isdigit(ch)) re=(re<<3)+(re<<1)+ch-'0',ch=getchar();
    return re;
}
void SPFA()
{
    memset(f,0x3f,sizeof f);
    f[1][0]=0;
    queueQ;
    Q.push(make_pair(1,0));
    while(!Q.empty())
    {
        int now=Q.front().first,dep=Q.front().second;Q.pop();
        vis[now][dep]=false;
        for(int i=top[now];i;i=nex[i])
        {
            if(f[to[i]][dep]>max(f[now][dep],len[i]))
            {
                f[to[i]][dep]=max(f[now][dep],len[i]);
                if(!vis[to[i]][dep])
                {
                    vis[to[i]][dep]=true;
                    Q.push(make_pair(to[i],dep));
                }
            }
            if(depf[now][dep])
            {
                f[to[i]][dep+1]=f[now][dep];
                Q.push(make_pair(to[i],dep+1));
            }
        }
    }
}
int main()
{
    n=read(),m=read(),k=read();
    while(m--)
    {
        int x=read(),y=read(),z=read();
        to[++cnt]=y,len[cnt]=z,nex[cnt]=top[x],top[x]=cnt;
        to[++cnt]=x,len[cnt]=z,nex[cnt]=top[y],top[y]=cnt;
    }
    SPFA();
    printf("%d",f[n][k]==0x3f3f3f3f?-1:f[n][k]);
    return 0;
}

推荐阅读
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 本文介绍了如何使用JQuery实现省市二级联动和表单验证。首先,通过change事件监听用户选择的省份,并动态加载对应的城市列表。其次,详细讲解了使用Validation插件进行表单验证的方法,包括内置规则、自定义规则及实时验证功能。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 本文详细介绍了 GWT 中 PopupPanel 类的 onKeyDownPreview 方法,提供了多个代码示例及应用场景,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 数据管理权威指南:《DAMA-DMBOK2 数据管理知识体系》
    本书提供了全面的数据管理职能、术语和最佳实践方法的标准行业解释,构建了数据管理的总体框架,为数据管理的发展奠定了坚实的理论基础。适合各类数据管理专业人士和相关领域的从业人员。 ... [详细]
  • 本文详细介绍了 Dockerfile 的编写方法及其在网络配置中的应用,涵盖基础指令、镜像构建与发布流程,并深入探讨了 Docker 的默认网络、容器互联及自定义网络的实现。 ... [详细]
  • 本文详细介绍了如何使用 Yii2 的 GridView 组件在列表页面实现数据的直接编辑功能。通过具体的代码示例和步骤,帮助开发者快速掌握这一实用技巧。 ... [详细]
  • 在金融和会计领域,准确无误地填写票据和结算凭证至关重要。这些文件不仅是支付结算和现金收付的重要依据,还直接关系到交易的安全性和准确性。本文介绍了一种使用C语言实现小写金额转换为大写金额的方法,确保数据的标准化和规范化。 ... [详细]
  • 本文介绍了如何使用 Spring Boot DevTools 实现应用程序在开发过程中自动重启。这一特性显著提高了开发效率,特别是在集成开发环境(IDE)中工作时,能够提供快速的反馈循环。默认情况下,DevTools 会监控类路径上的文件变化,并根据需要触发应用重启。 ... [详细]
author-avatar
黑糖老姜茶
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有