C#获取文件图标 C#如何提取文件在Windows中的显示图标

来源:这里教程网 时间:2026-02-21 17:41:26 作者:

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 路径(如
\servershareile.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、文件状态组合下保持一致——多数问题出在路径规范化和标志位漏设,而不是代码本身。

相关推荐