洛谷 P2832 行路难

本文深入探讨了SPFA算法在解决特殊最短路径问题中的应用,特别是在考虑到疲劳度增加的复杂场景下。通过对比Dijkstra算法,文章阐述了SPFA如何通过优化路径选择过程,有效地找到最优解。此外,还详细解释了如何在复杂DAG中追踪并打印最短路径。

题面

    这个最短路有点special,会有疲劳度的加成效应,这个时候应该怎么办呢?

    难就难在,如果走一条路比另一条路长,但是用的边少,那么这条路并不一定就更差。

    我们要是能解决这个问题,就可以做出本题。

    

    想一想两种常用的单源最短路的实现过程,dij是优先队列每次弹出非标记点中最近的那个,而spfa则可以看成bfs的延伸,用的边少的一定是会比用的边多的先被扩展到的。

    如果这么一想,那么选择就很显然了:我们如果用spfa的话,顺带记录一下疲劳值,是一定能找出答案的。我们考虑spfa的过程,如果 d[x](目前要扩展的点)+ val[i] (边权)+pl(队列里这个点的疲劳值)>= d[to](边的出点),那么这条路径一定是无用的,因为这样走到to的路径长度不仅没变短,反而疲劳值更大(注意spfa的队列里的疲劳值肯定是单调不减的),所以肯定不会是答案,而spfa的过程也是不允许这种情况再入队列的;相反的,如果d[x] + val[i] + pl< d[to],那么说明这条路径还有可能是答案,我们把它加到队列里去,但这并不会影响之前已经入队列的疲劳值更小的状态的寻找答案的过程。

 

    最后就是打印最短路的过程了,显然不能像以前一样开个p[i]记录最短路树的上一个节点,因为这种特殊的spfa求出的根本就不是一颗树(其实普通最短路也不一定是一棵树,不过很多边是等价的于是我们可以把等价的边删的只剩一条就可以构建最短路树)!而是一个复杂的DAG!

    但如果我们记录队列里的每个状态是由哪个状态转移过来的,答案就可以轻松找到了,具体操作可以看代码、、、

 

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int N=10005,M=200005,inf=2e9;

inline int read(){
    int x=0; char ch=getchar();
    for(;!isdigit(ch);ch=getchar());
    for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';
    return x;
}

int to[M*2],ne[M*2],hd[N],val[M*2],D[N];
int ans=inf,P,num,n,m,l,r;
struct node{
	int x,pl,pr,d;
}q[10000005];

inline void add(int x,int y,int z){
	to[++num]=y,ne[num]=hd[x],hd[x]=num,val[num]=z;
}

inline void solve(){
	q[l=r=1]=(node){1,0,0,0},D[1]=0;
	
	for(node now;l<=r;l++){
		now=q[l];
		if(now.x==n&&now.d<ans) ans=now.d,P=l;
		
		for(int i=hd[now.x],dd;i;i=ne[i]) if((dd=now.d+now.pl+val[i])<D[to[i]])
			D[to[i]]=dd,q[++r]=(node){to[i],now.pl+1,l,dd};
	}
}

void Print(int x){
	if(!x) return;
	Print(q[x].pr);
	printf("%d ",q[x].x);
}

int main(){
	n=read(),m=read(),fill(D+1,D+n+1,inf);
	for(int u,v,w;m;m--) u=read(),v=read(),w=read(),add(u,v,w);
	solve(),printf("%d\n",ans);
	Print(P);
	return 0;
}

  

 

转载于:https://www.cnblogs.com/JYYHH/p/11295390.html

<think>我们参考引用[2]、[3]、[4]中关于Java基础语法的内容,以及引用[1]中关于二进制文件读写的思路(但我们要操作的是文本文件)。任务:使用Java完成以下两个功能:1.将指定文本保存到桌面的file.txt文件2.从该文件逐行读取内容,并存储到字符串数组中(忽略以#开头的注释行)步骤:1.保存文本到文件:使用FileWriter或BufferedWriter等。2.读取文件:使用BufferedReader逐行读取,并过滤注释行。注意:Java中操作文件需要处理异常(参考引用[3]),我们使用try-with-resources确保资源正确关闭。具体实现:保存文本到文件(桌面上的file.txt):获取桌面路径:可以使用System.getProperty("user.home")+"/Desktop/file.txt"写入文本:将字符串写入文件,注意关闭流。读取文件并过滤注释行:逐行读取,如果行不为空且不以#开头,则加入数组(或列表)。我们按照要求编写代码,并添加必要的注释。</think> 以下是用 Java 实现文件操作的完整解决方案,包含文件写入和注释过滤读取功能: ### Java 文件操作实现 ```java import java.io.*; import java.util.ArrayList; import java.util.List; public class FileOperations { public static void main(String[] args) { // 桌面文件路径 (跨平台兼容) String desktopPath = System.getProperty("user.home") + "/Desktop/file.txt"; // 1. 写入文本到文件 String content = "# 这是注释行\nHello Java!\n# 另一条注释\n文件操作示例\nWelcome!"; writeToFile(desktopPath, content); System.out.println("文件写入成功!保存路径: " + desktopPath); // 2. 读取文件并过滤注释行 System.out.println("\n读取文件内容(忽略注释行):"); String[] lines = readFileWithoutComments(desktopPath); for (String line : lines) { System.out.println(line); } } // 写入文本到文件 public static void writeToFile(String filePath, String content) { try (BufferedWriter writer = new BufferedWriter(new FileWriter(filePath))) { writer.write(content); } catch (IOException e) { System.err.println("文件写入错误: " + e.getMessage()); e.printStackTrace(); } } // 读取文件并过滤注释行 public static String[] readFileWithoutComments(String filePath) { List<String> validLines = new ArrayList<>(); try (BufferedReader reader = new BufferedReader(new FileReader(filePath))) { String line; while ((line = reader.readLine()) != null) { // 忽略空行和以#开头的注释行 if (!line.trim().isEmpty() && !line.trim().startsWith("#")) { validLines.add(line); } } } catch (IOException e) { System.err.println("文件读取错误: " + e.getMessage()); e.printStackTrace(); } return validLines.toArray(new String[0]); } } ``` ### 功能说明 1. **文件写入**: - 使用 `FileWriter` 和 `BufferedWriter` 写入内容 - 自动处理文件路径(兼容 Windows/macOS/Linux) - 示例内容包含普通文本和注释行(以 # 开头) 2. **文件读取**: - 使用 `FileReader` 和 `BufferedReader` 逐行读取 - 自动忽略以下内容: - 空行(包含空白字符的行) - 以 `#` 开头的注释行 - 返回过滤后的字符串数组 3. **异常处理**: - 使用 try-with-resources 确保资源自动关闭[^2] - 捕获并处理 `IOException` - 输出详细的错误信息 ### 使用说明 1. 文件默认保存在当前用户的桌面 (`~/Desktop/file.txt`) 2. 程序执行流程: ```mermaid graph LR A[开始] --> B[构建文件路径] B --> C[写入示例内容] C --> D[读取并过滤文件] D --> E[输出结果] E --> F[结束] ``` 3. 示例输出: ``` 文件写入成功!保存路径: /Users/username/Desktop/file.txt 读取文件内容(忽略注释行): Hello Java! 文件操作示例 Welcome! ``` ### 注意事项 1. Java 文件操作必须处理 `IOException`[^3] 2. 使用 `System.getProperty("user.home")` 保证跨平台兼容性 3. 字符串数组转换使用 `toArray(new String[0])` 最佳实践 4. 类名必须与文件名一致(`FileOperations.java`)[^4] 此解决方案满足您所有需求:文本保存、注释过滤读取、桌面路径支持、健壮的异常处理。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值