P1948 [USACO08JAN]电话线Telephone Lines

本文详细介绍了如何使用二分查找与最短路径快速算法(SPFA)结合解决特定类型的最短路径问题,特别是在边权包含免费和付费两种情况下的路径寻优策略。通过实例代码展示了算法的具体实现,包括如何设定最小支出阈值,如何处理分层图,并提供了两种不同的解决方案:一种基于二分查找优化SPFA,另一种采用分层图加SPFA的暴力求解方法。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

传送门

最短路

二分+SPFA

二分最小支出

如果边权<=最小支出,那么就相当于0

如果大于最小支出,值设为1

跑SPFA

如果dis[n] > k 说明到不了

否则说明可以到

模板套进去就好了,没什么好注释的...

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
using namespace std;
const int INF=2e9+7;
const int N=1e6+7;
int n,m,k;
int fir[N],from[N],to[N],val[N],cnt;
inline void add(int a,int b,int c)
{
    from[++cnt]=fir[a];
    fir[a]=cnt; to[cnt]=b; val[cnt]=c;
}
int dis[1007];
bool vis[1007];
queue <int> q;
bool SPFA(int mx)
{
    memset(dis,0x7f,sizeof(dis));
    memset(vis,0,sizeof(vis));
    q.push(1); vis[1]=1; dis[1]=0;
    while(!q.empty())
    {
        int u=q.front(); q.pop();
        vis[u]=0;
        for(int i=fir[u];i;i=from[i])
        {
            int v=to[i],cost= val[i]>mx ? 1 : 0;
            if(dis[v]>dis[u]+cost)
            {
                dis[v]=dis[u]+cost;
                if(!vis[v]) q.push(v),vis[v]=1;
            }
        }
    }
    return dis[n]>k ? 0 : 1;
}
int main()
{
    int a,b,c,mx=-INF,mi=INF;
    cin>>n>>m>>k;
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d%d",&a,&b,&c);
        add(a,b,c); add(b,a,c);
        mx=max(mx,c); mi=min(mi,c);
    }
    bool flag=0;
    while(mi<=mx)
    {
        int mid=(mi+mx)>>1;
        if(SPFA(mid)) mx=mid-1,flag=1;
        else mi=mid+1;
    }
    if(!flag) cout<<"-1";
    else cout<<mi;
    return 0;
}
二分+SPFA

 

还有一种很暴力的方法

分层图+SPFA

设dis [ i ] [ j ] 表示到了第 i 个点,用了 j 次免费次数时的最小支出

然后每次分为用免费次数和不用免费次数两种情况讨论,更新后面的dis

速度竟然差不多...

直接看代码就好了

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
using namespace std;
const int N=1e6+7;
int n,m,k;
int fir[N],from[N],to[N],val[N],cnt;
inline void add(int a,int b,int c)
{
    from[++cnt]=fir[a];
    fir[a]=cnt; to[cnt]=b; val[cnt]=c;
}
int dis[1007][1007];
bool vis[1007][1007];
struct node
{
    int pos,num;//当前位置为pos,用了num次免费次数
};
queue <node> q;
void SPFA()
{
    memset(dis,0x7f,sizeof(dis));
    q.push((node){1,0}); vis[0][0]=1; dis[1][0]=0;
    while(!q.empty())
    {
        int pos=q.front().pos,num=q.front().num; q.pop();
        vis[pos][num]=0;
        for(int i=fir[pos];i;i=from[i])
        {
            int v=to[i],mx=max(dis[pos][num],val[i]);
            if( dis[v][num]>mx )//不用免费次数
            {
                dis[v][num]=mx;
                if(!vis[v][num])
                {
                    q.push((node){v,num});
                    vis[v][num]=1;
                }
            }
            if(dis[v][num+1]>dis[pos][num]&&num<=k)//用免费次数
            {
                dis[v][num+1]=dis[pos][num];
                if(!vis[v][num+1])
                {
                    q.push((node){v,num+1});
                    vis[v][num+1]=1;
                }
            }
        }
    }
}
int main()
{
    int a,b,c;
    cin>>n>>m>>k;
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d%d",&a,&b,&c);
        add(a,b,c); add(b,a,c);
    }
    SPFA();
    int ans=1e9+7;
    ans=min(ans,dis[n][k]);
    if(ans==1e9+7) cout<<"-1";
    else cout<<ans;
    return 0;
}
分层图+SPFA

转载于:https://www.cnblogs.com/LLTYYC/p/9692191.html

资源下载链接为: https://pan.quark.cn/s/22ca96b7bd39 在 IT 领域,文档格式转换是常见需求,尤其在处理多种文件类型时。本文将聚焦于利用 Java 技术栈,尤其是 Apache POI 和 iTextPDF 库,实现 doc、xls(涵盖 Excel 2003 及 Excel 2007+)以及 txt、图片等格式文件向 PDF 的转换,并实现在线浏览功能。 先从 Apache POI 说起,它是一个强大的 Java 库,专注于处理 Microsoft Office 格式文件,比如 doc 和 xls。Apache POI 提供了 HSSF 和 XSSF 两个 API,其中 HSSF 用于读写老版本的 BIFF8 格式(Excel 97-2003),XSSF 则针对新的 XML 格式(Excel 2007+)。这两个 API 均具备读取和写入工作表、单元格、公式、样式等功能。读取 Excel 文件时,可通过创建 HSSFWorkbook 或 XSSFWorkbook 对象来打开相应格式的文件,进而遍历工作簿中的每个 Sheet,获取行和列数据。写入 Excel 文件时,创建新的 Workbook 对象,添加 Sheet、Row 和 Cell,即可构建新 Excel 文件。 再看 iTextPDF,它是一个用于生成和修改 PDF 文档的 Java 库,拥有丰富的 API。创建 PDF 文档时,借助 Document 对象,可定义页面尺寸、边距等属性来定制 PDF 外观。添加内容方面,可使用 Paragraph、List、Table 等元素将文本、列表和表格加入 PDF,图片可通过 Image 类加载插入。iTextPDF 支持多种字体和样式,可设置文本颜色、大小、样式等。此外,iTextPDF 的 TextRenderer 类能将 HTML、
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值