用 SHGetFileInfo
获取文件图标句柄最可靠
Windows 原生图标(包括不同大小、状态)必须通过 Shell API 提取,.NET 自带的
Icon.ExtractAssociatedIcon只能拿到小尺寸(16×16)且常缓存过期,对快捷方式、注册表关联缺失的文件经常返回默认图标。
SHGetFileInfo是唯一能按需获取 16/32/48/256px 图标、叠加层(如快捷方式箭头、加密锁)、以及真实 Shell 渲染效果的途径。
关键点:
SHGetFileInfo返回的是 GDI
HICON,需用
Icon.FromHandle转为托管
Icon,但注意:该句柄**不能直接释放**,否则图标立刻失效;应调用
DestroyIcon手动清理(尤其在循环加载大量图标时) 必须传入
SHGFI_ICON | SHGFI_LARGEICON或
SHGFI_SMALLICON控制尺寸,仅传
SHGFI_ICON默认返回小图标 路径必须是**绝对路径**,相对路径或 UNC 路径(如
\servershareile.txt)需先用
Path.GetFullPath规范化,否则返回空图标
SHGetFileInfo
的典型调用参数组合
以下是最常用且稳定的参数组合(C# P/Invoke):
[DllImport("shell32.dll")]
public static extern IntPtr SHGetFileInfo(
string pszPath,
uint dwFileAttributes,
out SHFILEINFO psfi,
uint cbSizeFileInfo,
uint uFlags);
<p>[StructLayout(LayoutKind.Sequential)]
public struct SHFILEINFO {
public IntPtr hIcon;
public int iIcon;
public uint dwAttributes;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 260)]
public string szDisplayName;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 80)]
public string szTypeName;
}</p><p>// 示例:获取 32×32 图标(含叠加层)
var shinfo = new SHFILEINFO();
var flags = (uint)(0x000000100 | // SHGFI_ICON
0x000000040 | // SHGFI_LARGEICON (32×32)
0x000000002 | // SHGFI_USEFILEATTRIBUTES (无文件时可用属性模拟)
0x000000010); // SHGFI_OVERLAYINDEX (需要叠加层索引)</p><p>SHGetFileInfo(filePath, 0, out shinfo, (uint)Marshal.SizeOf(shinfo), flags);
if (shinfo.hIcon != IntPtr.Zero) {
var icon = Icon.FromHandle(shinfo.hIcon);
// 使用 icon...
}
注意:
SHGFI_USEFILEATTRIBUTES允许传入不存在的路径(比如只给扩展名),此时需手动指定
dwFileAttributes(如
FileAttributes.Directory),否则图标为空。
处理图标缩放与 DPI 缩放问题
直接从
SHGetFileInfo拿到的图标是位图格式,不随系统 DPI 缩放自动适配。若目标控件(如
ListView、
TreeView)启用了高 DPI 模式,图标会模糊或错位: 不要对原始图标调用
icon.ToBitmap()后再缩放——会失真;应优先用
SHGFI_LARGEICON+
SHGFI_JUMBOICON(Windows 10+)获取原生大尺寸图标 若必须缩放,用
Graphics.DrawImage并设置
InterpolationMode.HighQualityBicubic和
SmoothingMode.AntiAlias对
ImageList,务必在创建时指定
ImageList.ColorDepth = ColorDepth.Depth32Bit,否则透明通道丢失,叠加层变黑块
快捷方式(.lnk)和特殊文件类型图标容易出错
.lnk 文件默认显示目标文件图标,但
SHGetFileInfo对它行为不稳定——有时返回 lnk 自身图标(带箭头),有时返回目标图标,取决于系统缓存和是否已解析过该链接: 安全做法:先用
ShellLinkObject(COM)解析 .lnk,拿到真实目标路径后再调用
SHGetFileInfo对虚拟文件(如 OneDrive 同步状态图标、WslFs 路径),
SHGFI_USEFILEATTRIBUTES常失效,必须确保路径可被 Shell 正确识别(例如 WSL 路径要转成
\wsl$…格式) 注册表中未关联扩展名的文件(如 .xyz),即使存在,也会回退到空白文档图标;此时可尝试用
SHGFI_USEFILEATTRIBUTES+
FILE_ATTRIBUTE_NORMAL强制走通用图标逻辑
真正麻烦的不是怎么拿图标,而是怎么让图标在各种路径、DPI、文件状态组合下保持一致——多数问题出在路径规范化和标志位漏设,而不是代码本身。
