zoukankan      html  css  js  c++  java
  • 通过快捷方式lnk获得文件真实路径

    通过快捷方式.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)
    

      

  • 相关阅读:
    IE浏览器和谷歌浏览器相互跳转
    centos7安装docker
    centos7安装groovy
    centos7安装NodeJs
    mongodb数据库的备份还原
    centos7最小版配置
    centos7中python2.7升级到python3.7
    typedef struct用法详解与小结
    MinGW的gdb调试
    MinGW-w64安装教程——著名C/C++编译器GCC的Windows版本
  • 原文地址:https://www.cnblogs.com/navysummer/p/12055419.html
Copyright © 2011-2022 走看看