前段时间看到群友讨论压缩包能不能运行,想起了N年前用自解压文件SFX实现的一个“需求”:在没有安装任何应用软件的Windows(当时还要支持XP)上能双击打开自定义格式的文件。当时第一反应是这“需求”太奇葩了,简直是不可能。但思考后认为这个“需求”存在一定的合理性,因为当时的目标用户群体并不能熟练使用电脑,可能不知道打开一个文件需要安装对应的软件。

这里“需求”之所以打上引号,是因为我觉得这不是真正的需求,而是一个解决方案,真正的需求是如何让不熟悉电脑操作的用户方便的使用我们自定义格式的文件。

实施方案

Windows系统的注册表中会记录文件格式与对应软件的关联关系,双击文件时会找到关联的软件运行并加载文件。而这个文件与软件的关联信息是在安装应用程序时写进注册表的。正因如此,一开始觉得“需求”简直是不可能的。经过一番思考,尝试把自定义格式的文件和应用程序打包到一起形成一个exe,双击exe时释放应用程序和文件,并运行应用程序加载文件。具体到实施上有两种选择:用NSIS或者MSI打包工具生成安装包,用压缩软件制作SFX自解压文件。考虑到第一种方式依赖项较多且代码调用不方便,最后选择了自解压文件的方式。

自解压文件

自解压SFX(self extracting)文件是压缩文件的一种,其后缀名是exe。它可以不借助任何压缩工具,只需双击该文件就可以自动执行解压,并根据配置执行解压后的可执行程序(EXE)。WinRAR和7-Zip都可以制作自解压文件。这里以7-Zip为例介绍如何制作自解压文件。自解压安装包必须包含三个文件:7z_Archive,SFX_Module, Installer_Config。

  • 7z_Archive:用7z打包归档好的文件(希望放到自解压文件中的程序和文件)
  • SFX_Module:自解压文件的核心模块,必须与7z.exe放在同一目录,主要包含四种类型:
SFX_Module 说明
7z.sfx 带有GUI的自解压模块
7zCon.sfx 提供Console窗口交互的自解压模块
7zS.sfx 允许创建安装程序的带GUI的自解压模块
7zSD.sfx 允许创建安装程序的带GUI的自解压模块(使用MSVCRT.dll)

7z.sfx7zCon.sfx在7-Zip的安装目录中能找到,这两个基本上没什么用,仅仅是完成双击自解压功能,解压完成之后没有任何操作,即便是在Installer_Config配置文件中指定了需要执行的程序也没作用。以下两个图分别是7z.sfx7zCon.sfx制作的自解压文件解压过程。



7zS.sfx7zSD.sfx则需要从官网下载 LZMA SDK包获取。这两个制作的自解压文件会把打包的文件释放到用户的临时目录,然后根据Installer_Config配置文件执行指定程序,程序结束后会删除临时文件。制作自解压文件的命令如下:

copy /b 7zS.sfx + config.txt + archive.7z archive.exe
  • Installer_Config:这个配置文件包含了自解压文件对话框的标题,信息,解压完成后执行的文件和参数等信息。文件以;!@Install@!UTF-8!开始,并以;!@InstallEnd@!结束,且必须是用UTF-8编码。具体包含的配置信息内容参见7-Zip的帮助文档,以下是配置文件的示例:
;!@Install@!UTF-8!
Title="7-Zip 4.00"
BeginPrompt="自解压文件测试,是否继续?"
RunProgram="7zFM.exe"
;!@InstallEnd@!

下图是7zS.sfx配置了制作的自解压文件解压过程。虽然实现了双击自解压后运行指定的7zFM.exe,但是自解压文件的图标、文件说明、公司信息等文件属性不是我们想要的,并且解压过程的对话框样式和内容也不符合预期。接下来需要美化自解压文件。

美化自解压文件

7z自带的sfx以及帮助文档提供的信息有限,可以通过7z SFX Builder实现更多的自定义操作。下载安装后可以用图形界面的方式自定义解压过程对话框的样式,并生成相应的Installer_Config信息。首先设置解压文件覆盖模式和对话框样式。

然后根据实际情况选择对话框具体的样式信息。

接下来就是设置对话框的标题、解压进度窗体、错误窗体、警告窗体的标题信息。

设置完标题信息后就是设置对话框内的各种信息内容。

再然后就是配置解压后执行文件以及参数信息。

配置完这些信息之后,可以在“output”下看到Installer_Config信息,我们可以复制保存下来以备后用。需要注意的是,Installer_Config配置文件以;!@InstallEnd@!结束,从“output”中复制信息时,;!@InstallEnd@!后边还有内容就不要复制了。

7z SFX Builder也提供了更多的sfx模块供选择,这些模块保存在C:\Program Files (x86)\7z SFX Builder\3rdParty\Modules目录下,

选择了sfx模块后可以设置文件说明、公司信息等属性。这也决定了最终制作的自解压文件的文件属性。

到了这里,我们离大功告成只剩下修改图标这一步了。这时需要借助Resource Hacker来修改图标信息,如果上一步中没有编辑sfx模块的信息,也可以在这里通过Resource Hacker修改。我们只需准备好ico文件,然后在Resource Hacker打开sfx模块,并替换图标文件或编辑版本信息,完成后保存sfx模块文件。至此就完成了自解压文件的美化工作。

小结

文中只介绍了7z SFX Builder的基本用法,它的帮助文档提供了更多的操作说明以及示例。

自解压文件在某些特定的场景给我们提供了便利,但也容易被杀毒软件当成病毒处理。我在实现开篇提到的“需求”时,也饱受杀毒软件困扰。最后分享几点经验:

  1. 对自解压文件中的可执行程序及依赖文件进行code review,删除所有未使用过的变量和属性;
  2. 对可执行程序以及所有依赖的类库进行强签名;
  3. 提交常用杀毒软件白名单;
  4. 最后一点,也是最重要的一点。前边几点只能缓解杀毒软件误报,要想彻底解决,就不要在生产环境使用自解压文件。

SFX的妙用——如何在不安装软件的情况下打开自定义格式文件?的更多相关文章

  1. 在安装ISE的情况下,充分利用ISE的安装目录,查找资料

    2013-06-22 11:03:02 在找资料时,通过官网输入关键字的方法找资料,有事会给出很多版本的链接.或者找不到,下面给出一种简便的方法,可以快速找到想要的资料. 如果要找ISE各个工具如pl ...

  2. 在本地没有安装Oracle的情况下,使用plsql远程连接数据库

    远程连接数据库的方式不止一种.这里主要写在本地没有安装Oracle的情况下,使用plsql可视化工具远程连接数据库的方式. 一.前提 首先你本地得安装有plsql工具.具体的安装步骤就不多说了. 二. ...

  3. 本地在不安装Oracle的情况下安装PLSQL客户端

    本文解决问题:   通常在本地安装PLSQL后,如果本地没有安装Oracle数据库的话,PLSQL是不能使用的,输入远程数据库登录信息会提示:"Oracle Client没有正确安装&quo ...

  4. linux下python3的安装(已安装python2的情况下)

    前段时间想自学一下python,就在虚拟机里已安装python2.7的情况下又安装了最新版python3.6.4.于是问题来了..只要一打开终端就出现一大段错误代码(忘记截图了),当时看到是ros和p ...

  5. 在注册表中无Python3.5安装路径的情况下安装pywin32-

    当安装pywin32出现Python Version 3.5 required which was not found in the registry的时候表面注册表中没有Python3.5的安装路径 ...

  6. 在不安装mysql-connector-net的情况下使用FluentData框架

    最近在开发项目中使用了FluentData框架,通过使用这个框架减少了很多开发的工作量,FluentData是一个轻量级的框架操作起来的自由度很大也少了很多负责的配置.但是在开发的时候发现一个问题就是 ...

  7. 不安装Oracle客户端情况下使用PL/SQL 远程连接数据库

    附送PL/SQL Developer11中文版下载地址 1.先到Oracle网站下载Instant Client : http://www.oracle.com/technetwork/databas ...

  8. C# 不安装Oracle客户端情况下,如何连接到Oracle数据库

    简介: 在我们开发应用场景经常碰到需要连接Oracle数据库,这也是相当常见的一种情况.一般.Net环境连接Oracle数据库,可以通过TNS/SQL.NET 配置文件,而 TNS 必须要 Oracl ...

  9. PLSQL Developer在未安装Oracle Client情况下连接Oracle

    常用的Oracle开发的工具有SQL Developer和PL/SQL Developer, 用PL/SQL连接oracle数据库,不管是本地的还是远程的,一般都需要安装oracle客户端 如何达到不 ...

  10. iOS 手机没有安装支付宝的情况下,不调支付宝网页的解决的方法

    NSArray *array = [[UIApplication sharedApplication] windows]; UIWindow* win=[array objectAtIndex:0]; ...

随机推荐

  1. WorkManager的用法

    一.WorkManager的作用 绝大部分应用程序都有后台执行任务的需求,根据需求的不同,Android为后台任务提供了多种解决方案,如JobShedule,Loader,Service等.如果这些a ...

  2. 【Hexo】插件推荐以及使用小技巧

    目录 插件推荐 hexo-deployer-git hexo-word-counter hexo-abbrlink hexo-generator-sitemap 小技巧 自定义提交信息 参考资料 He ...

  3. 从壹开始前后端开发【.Net6+Vue3】(二)前端框架

    项目名称:KeepGoing(继续前进) 介绍 工作后,学习的脚步一直停停走走,希望可以以此项目为基础,可以不断的迫使自己不断的学习以及成长 将以Girvs框架为基础,从壹开始二次开发一个前后端管理框 ...

  4. 《Python魔法大冒险》005 魔法挑战:自我介绍机器人

    魔法师和小鱼坐在图书馆的一扇窗户旁,窗外的星空闪烁着神秘的光芒.魔法师轻轻地拍了拍小鱼的肩膀. 魔法师: 小鱼,你已经学会了编写简单的魔法程序,现在我要教你如何创造一个有自己思想的机器人,让它能够和我 ...

  5. js获取当前月的天数

    //取得本月天数(实际代码:) var now=new Date(); var d = new Date(now.getFullYear(),now.getMonth()+1,0); var days ...

  6. 10月TIOBE榜Java跌出前三!要不我转回C#吧

    前言 Java又要完了,又要没了,你没看错,10月编程语言榜单出炉,Java跌出前三,并且即将被C#超越,很多资深人士预测只需两个月,Java就会跌出前五. 看到这样的文章,作为一名Java工程师我感 ...

  7. [GXYCTF 2019]Ping Ping Ping

    题目就是ping,而且这还有一个查询窗口,就随便查询试试 ping了一下本地,发现没有什么很大的作用,给出了提示是php可以执行系统函数这就感到神奇了,就还是上网搜了搜 发现可以在查询IP后面进行拼接 ...

  8. 2020/5/8—cf,我裂开来

    呜呜呜我爆零了呜呜呜ljll 嗯T1T2防爆零的没了呜呜呜在此纪念可怜的yjz大佬21发AC 太惨了(逃 先来说说我们都有些啥题目吧... T1 嗯,裂开了,当场裂开我一看!桶排!然后实现,嗯?嗯!嗯 ...

  9. 【Unity3D】动态路径特效

    1 前言 ​ 本文通过导航系统(NavMeshAgent)和线段渲染器(LineRenderer)实现了角色走迷宫和绘制路径功能,同时实现动态路径特效. ​ 导航系统的介绍详见博客:导航系统.分离路面 ...

  10. 【Unity3D】消融特效

    1 前言 ​ 选中物体消融特效中基于 Shader 实现了消融特效,本文将基于 Shader Graph 实现消融特效,两者原理一样,只是表达方式不同,另外,选中物体消融特效中通过 discard 丢 ...