通过快捷方式.lnk获得文件真实路径
前提
最近开发资源管理,需要预先上传大量资源,负责整理资源的同学因为空间不足,直接用快捷键方式整理视频资源OTZ,所以只能想办法通过.lnk文件获得文件的真实地址。
以下所有内容都来自网络,博主仅做了参考与总结。
.lnk文件格式解析
此处对lnk文件组成做一个大概介绍主旨是帮助了解如何从link文件中提取需要的信息
一个lnk文件包括一下几个模块:
模块 备注
注意:
不是所有的模块都必须包含在内,但如果存在就要按上述的顺序排列。
以下我们详细了解需要用到的两个模块:
1. 文件头(lnk file header)
偏移 长度 类型 备注
0x14处16进制数的含义:
0x18处16进制数的含义:
2. 文件位置信息段(File location info)
0x08偏移flags 具体含义:
如果目标文件是本地文件,那么文件名称 = 本地路径信息+剩余偏移路径
如果目标文件是网络文件,那么文件名称 = 网络卷中共享名称+剩余偏移路径
所以,File location info节之后的数据是,本地卷信息表,及网络卷信息表。
1. 本地卷信息表结构
2. 网络卷信息表结构
注意:
八个比特(Bit)称为一个字节(Byte),两个字节称为一个字(Word),两个字称为一个双字(Dword),两个双字称为一个四字(Qword)。
代码
private void parseLink(File f) throws FileNotFoundException, IOException { FileInputStream fin = new FileInputStream(f); byte[] link = new byte[(int)f.length()]; //读取文件中的内容到link[]数组 fin.read(link); fin.close(); // 判断当前文件是否为快捷方式 if(!isLnkFile(link)){ return; } // 获得flags信息 byte flags = link[0x14]; int shell_len = 0; // 0000 0000 xxxx xxxx & 0000 0000 0000 0001(判断是否包含shell item id list段) if((flags & 0x1) > 0) { // 如果存在,则获取shell item id list段的总长度,加2是为了将link[0x4c]本身的长度计算在内 shell_len = bytes2short(link,0x4c) + 2; } // 获得文件位置信息段的开始位置=shell item id list段的开始位置+shell item id list段的总长度 int file_start = 0x4c + shell_len; // 获取本地路径信息的偏移 int local_sys_off = link[file_start + 0x10] + file_start; String real_file = getNullDelimitedString(link, local_sys_off); System.out.println(real_file); } private boolean isLnkFile(byte[] link) { if (link[0x00]== 0x4c) {// 76,L,0x4c代表lnk文件格式 return true; } return false; } /** * 将两个字节转换为short<br> * 注意,因为仅限英特尔操作系统,所以这是小端字节<br> */ private int bytes2short(byte[] bytes, int off) { return bytes[off] | (bytes[off + 1] << 8); } /** * 获得从偏移位置off到以‘0’为结尾分割字符串 * @param bytes 源数组 * @param off 偏移位置 * @return 字符串 */ private String getNullDelimitedString(byte[] bytes, int off) { int len = 0; // 计算字符串占用数组的真实长度 while (true) { if (bytes[off + len] == 0) { break; } len++; } byte[] results = new byte[len]; for (int i = off, j = 0; i < off + len; i++, j++) { results[j] = bytes[i]; } try { // 因为我是中文系统,所以设置了字符集GBK,否则中文路径会出现乱码 return new String(bytes, off, len, "GBK"); } catch (UnsupportedEncodingException e) { e.printStackTrace(); } return null; }
import sys import win32com.client shell = win32com.client.Dispatch("WScript.Shell") shortcut = shell.CreateShortCut("t:\test.lnk") print(shortcut.Targetpath)