简短的回答是:标准的 TFTP 协议完全没有提供"列出目录"或"查看文件列表"的功能。 这也就是为什么它被称为 Trivial FTP(简单/琐碎的 FTP)。它被设计得极其精简,以便能固化在网络设备的单片机、ROM 或网卡启动芯片(如 PXE 启动)中。
💡 为什么 TFTP 不能列出目录?
我们可以对比一下它与普通 FTP 的区别:
-
普通 FTP (File Transfer Protocol) :有一套复杂的命令集。当你输入
ls或dir时,客户端会向服务器发送LIST命令,服务器会把目录下的文件名当作文本数据传回来。 -
TFTP (Trivial FTP) :总共只有 5 种 基本的数据包类型(读请求、写请求、数据包、确认包、错误包)。它没有"控制通道"和"数据通道"的分离,只认准一件事:要么你给我一个精确的文件名让我读(RRQ),要么你给我一个精确的文件名让我写(WRQ)。
所以,如果客户端不知道文件名,TFTP 服务器是绝对不会主动告诉你有什么文件的。
🛠️ 现实中如何解决这个问题?
如果你在开发 VB.NET 程序或运维时,确实需要知道服务器有哪些文件,通常有以下几种替代或折中方案:
方案一:在服务器端(如 Tftpd64)查看
如果你能控制服务器,最直接的方法是看服务器软件的界面:
- Tftpd64/Tftpd32 的主界面上有一个 "Show Dir" 按钮,点击它会弹出一个本地窗口,列出当前根目录下的所有文件。但这只能在服务器本地看,客户端无法通过网络获取这个列表。
方案二:通过"约定文件"实现(推拉模式)
如果你正在写客户端和服务器的配对软件,可以通过生成一个"索引文件"来绕过这个限制:
-
在服务器端写一个定时脚本,每隔几分钟执行一次
dir /b > filelist.txt,将当前目录下的文件名导出到一个固定的文本文件中。 -
你的 VB.NET 客户端在需要获取列表时,先固定下载
filelist.txt这个文件:DoTftp("download", "filelist.txt", Application.StartupPath & "\list.txt") -
下载下来后,用代码读取这个文本文件,解析出里面有哪些文件名。
方案三:更换网络协议
如果项目还允许调整,且"获取文件列表"是刚需,建议放弃 TFTP,改用以下协议:
-
FTP / SFTP :支持标准的目录浏览(C#/.NET4.5.2 下可以用
FtpWebRequest或第三方库SSH.NET)。 -
HTTP / WebAPI :在服务器部署一个简单的 Web 服务,客户端访问
http://127.0.0.1/api/files返回 JSON 列表,然后再通过 HTTP 下载,这在现在的网络环境下更稳定高效。