读取/写入“扩展”文件属性(C#)


104

我试图找出如何在C#中读取/写入扩展文件属性,例如注释,比特率,访问日期,类别等,您可以在Windows资源管理器中看到它们。任何想法如何做到这一点?编辑:我主要是读/写视频文件(AVI / DIVX / ...)


3
由于已接受的答案仅显示了如何获取扩展属性,而不显示如何设置扩展属性,因此显然回答该问题。
Dmitri Nesteruk 2013年

1

Answers:


83

对于那些不喜欢VB的人,这里是c#:

注意,必须从“引用”对话框的“ COM”选项卡添加对Microsoft Shell控件和自动化的引用。

public static void Main(string[] args)
{
    List<string> arrHeaders = new List<string>();

    Shell32.Shell shell = new Shell32.Shell();
    Shell32.Folder objFolder;

    objFolder = shell.NameSpace(@"C:\temp\testprop");

    for( int i = 0; i < short.MaxValue; i++ )
    {
        string header = objFolder.GetDetailsOf(null, i);
        if (String.IsNullOrEmpty(header))
            break;
        arrHeaders.Add(header);
    }

    foreach(Shell32.FolderItem2 item in objFolder.Items())
    {
        for (int i = 0; i < arrHeaders.Count; i++)
        {
            Console.WriteLine(
              $"{i}\t{arrHeaders[i]}: {objFolder.GetDetailsOf(item, i)}");
        }
    }
}

3
如何设置这些值之一?例如,.txt文件的作者或发布者。我赢了7,并使用了它,它确实显示了空白的作者和出版者以及282个其他属性
Vaibhav Garg,2010年

1
@Vainbhav-您无法设置这些。
csharptest.net 2010年

26
您必须从“引用”对话框的“ COM”选项卡添加对Microsoft Shell控件和自动化的引用。
csharptest.net 2011年

2
如何设置它们更好地覆盖在这个问题:stackoverflow.com/questions/5337683/...
尼古拉斯·拉乌尔

1
在Windows 10中,它仅返回52个字段,不包括帧频,帧高,宽度,...。
Minh Nguyen


26

解决方案2016

将以下NuGet软件包添加到您的项目中:

  • Microsoft.WindowsAPICodePack-Shell 由Microsoft
  • Microsoft.WindowsAPICodePack-Core 由Microsoft

读写属性

using Microsoft.WindowsAPICodePack.Shell;
using Microsoft.WindowsAPICodePack.Shell.PropertySystem;

string filePath = @"C:\temp\example.docx";
var file = ShellFile.FromFilePath(filePath);

// Read and Write:

string[] oldAuthors = file.Properties.System.Author.Value;
string oldTitle = file.Properties.System.Title.Value;

file.Properties.System.Author.Value = new string[] { "Author #1", "Author #2" };
file.Properties.System.Title.Value = "Example Title";

// Alternate way to Write:

ShellPropertyWriter propertyWriter =  file.Properties.GetPropertyWriter();
propertyWriter.WriteProperty(SystemProperties.System.Author, new string[] { "Author" });
propertyWriter.Close();

重要:

该文件必须是有效的,由特定的分配软件创建。每种文件类型都有特定的扩展文件属性,但并非所有文件都是可写的。

如果右键单击桌面上的文件而无法编辑属性,则也将无法在代码中对其进行编辑。

例:

  • 在桌面上创建txt文件,将其扩展名重命名为docx。您无法编辑其AuthorTitle属性。
  • 用Word打开它,编辑并保存。现在你可以。

所以只要确保使用一些 try catch

进一步的主题: MSDN:实现属性处理程序


我没有看到任何包由微软这些名字
雅各布布鲁尔

1
@JacobBrewer实际上是“ acdvorak”上载的一个文件:nuget.org/packages/Microsoft.WindowsAPICodePack-Shell。在Visual Studio NuGet-Package-Manager中,它显示为“由Microsoft”。其他名称相似的软件包就是它的分支,也可能正常工作,甚至更好。我不知道...
Martin Schneider

25

VB.NET中的此示例读取所有扩展属性:

Sub Main()
        Dim arrHeaders(35)

        Dim shell As New Shell32.Shell
        Dim objFolder As Shell32.Folder

        objFolder = shell.NameSpace("C:\tmp")

        For i = 0 To 34
            arrHeaders(i) = objFolder.GetDetailsOf(objFolder.Items, i)
        Next
        For Each strFileName In objfolder.Items
            For i = 0 To 34
                Console.WriteLine(i & vbTab & arrHeaders(i) & ": " & objfolder.GetDetailsOf(strFileName, i))
            Next
        Next

    End Sub

您必须从“ 引用”对话框的“ COM”选项卡中添加对Microsoft Shell控件和自动化引用


2
值得一提的是,至少在Windows 7上,您可以将arrHeaders的大小增加到280以上,并获取大量其他元数据。我在寻找从WTV文件获取元数据的方法时发现了这一点(Windows 7 Media Center录制了电视节目)。
理查德

1
i = 0到34循环的第一个应该是i = 0到Integer.MaxValue的循环。然后测试objFolder.GetDetailsOf(objFolder.Items,i)的返回值。如果它返回null或空格,那么您将拥有所有标题。
蒂姆·墨菲

@TimMurphy,不幸的是,这是不正确的(至少在Windows 10或Windows 7上)。一些标题是空的,因此您需要遍历所有索引。(当然,由于没有数百万个,因此可以将循环数限制为1,000。)
skst

8

谢谢你们这个线程!当我想知道一个exe的文件版本时,它对我有所帮助。但是,我需要自己弄清楚什么是扩展属性。

如果在Windows资源管理器中打开exe(或dll)文件的属性,则将获得“版本”选项卡以及该文件的“扩展属性”视图。我想访问这些值之一。

解决方案是使用属性索引器FolderItem.ExtendedProperty,如果在属性名称中删除所有空格,则将获得该值。例如,文件版本为FileVersion,就在那里。

希望这对其他人有帮助,只是以为我会将此信息添加到此线程中。干杯!


2
这还可以确保您的代码(至少大部分情况下)仍可以在非英语机器上运行。
Ed Norris '02

1
这里的一项小改进,FolderItem不包含ExtendedProperty()。FolderItem2但是。
Mixxiphoid 2014年

这个答案比其他答案要简单。我提供了在此处可以工作的示例代码:stackoverflow.com/a/46648086/661933
nawfal

8

GetDetailsOf()方法-检索有关文件夹中项目的详细信息。例如,其大小,类型或上次修改的时间。文件属性可能会因Windows-OS版本而异。

List<string> arrHeaders = new List<string>();

 Shell shell = new ShellClass();
 Folder rFolder = shell.NameSpace(_rootPath);
 FolderItem rFiles = rFolder.ParseName(filename);

 for (int i = 0; i < short.MaxValue; i++)
 {
      string value = rFolder.GetDetailsOf(rFiles, i).Trim();
      arrHeaders.Add(value);
 }

我已经复制并使用了与上面相同的代码。但是,我在runime遇到COM异常Folder rFolder = shell.NameSpace(_rootPath);。仅供参考,我正在使用Windows 8 OS。
DonMax 2014年

1
那是什么错误?确保您使用的是正确版本的Shell32.dll。检查它是否有用
RajeshKdev

1
+1以上链接。您建议的链接工作正常。我还需要你帮忙。即,如何传递单个文件以获取元数据?因为,它只接受传递文件夹。
DonMax 2014年

6
  • 在研究了该线程和其他地方的许多解决方案之后,以下代码组合在一起。这仅是读取属性。
  • 我无法使Shell32.FolderItem2.ExtendedProperty函数正常工作,它应该采用字符串值并返回该属性的正确值和类型...对于我来说,这始终为null,并且开发人员参考资源非常薄。
  • WindowsApiCodePack似乎已经被微软放弃这给我们带来了下面的代码。

用:

string propertyValue = GetExtendedFileProperty("c:\\temp\\FileNameYouWant.ext","PropertyYouWant");
  1. 将以给定文件和属性名称的字符串形式返回您想要的扩展属性的值。
  2. 仅循环直到找到指定的属性-直到发现所有属性(如某些示例代码)为止
  3. 如果仅尝试正常创建Shell32对象,则将在Windows版本(如Windows Server 2008)上工作,在该版本中,您将收到错误消息“无法将类型为'System .__ ComObject'的COM对象转换为接口类型为'Shell32.Shell'”

    public static string GetExtendedFileProperty(string filePath, string propertyName)
    {
        string value = string.Empty;
        string baseFolder = Path.GetDirectoryName(filePath);
        string fileName = Path.GetFileName(filePath);
    
        //Method to load and execute the Shell object for Windows server 8 environment otherwise you get "Unable to cast COM object of type 'System.__ComObject' to interface type 'Shell32.Shell'"
        Type shellAppType = Type.GetTypeFromProgID("Shell.Application");
        Object shell = Activator.CreateInstance(shellAppType);
        Shell32.Folder shellFolder = (Shell32.Folder)shellAppType.InvokeMember("NameSpace", System.Reflection.BindingFlags.InvokeMethod, null, shell, new object[] { baseFolder });
    
        //Parsename will find the specific file I'm looking for in the Shell32.Folder object
        Shell32.FolderItem folderitem = shellFolder.ParseName(fileName);
        if (folderitem != null)
        {
            for (int i = 0; i < short.MaxValue; i++)
            {
                //Get the property name for property index i
                string property = shellFolder.GetDetailsOf(null, i);
    
                //Will be empty when all possible properties has been looped through, break out of loop
                if (String.IsNullOrEmpty(property)) break;
    
                //Skip to next property if this is not the specified property
                if (property != propertyName) continue;    
    
                //Read value of property
                value = shellFolder.GetDetailsOf(folderitem, i);
            }
        }
        //returns string.Empty if no value was found for the specified property
        return value;
    }

1
请注意,您可以强制转换shell为任何IShellDispatch接口(1-6)并直接调用该成员,而不是使用InvokeMember()Shell32.IShellDispatch ishell = (Shell32.IShellDispatch)shell; Shell32.Folder shellFolder = ishell.NameSpace(baseFolder);
skst

5

杰克的答案要简单一些。这是在MS上工作的示例代码:

var folder = new Shell().NameSpace(folderPath);
foreach (FolderItem2 item in folder.Items())
{
    var company = item.ExtendedProperty("Company");
    var author = item.ExtendedProperty("Author");
    // Etc.
}

对于那些不能静态引用shell32的人,可以这样动态地调用它:

var shellAppType = Type.GetTypeFromProgID("Shell.Application");
dynamic shellApp = Activator.CreateInstance(shellAppType);
var folder = shellApp.NameSpace(folderPath);
foreach (var item in folder.Items())
{
    var company = item.ExtendedProperty("Company");
    var author = item.ExtendedProperty("Author");
    // Etc.
}

1

我不确定要为哪种类型的文件编写属性,但taglib-sharp是一个出色的开源标签库,可以很好地包装所有这些功能。它具有对大多数流行媒体文件类型的大量内置支持,但还允许您对几乎所有文件进行更高级的标记。

编辑: 我已经更新了链接到taglib锋利。旧的链接不再起作用。

编辑:根据kzu的评论再次更新了链接。


这看起来非常有趣,我将主要查看视频文件(AVI,DIVX等)。感谢您的指点
David Hayes

的taglib锋利的链接似乎是死了:-( -这是怪异的维基在... developer.novell.com/wiki/index.php/TagLib_Sharp:_Examples ...指向URL
SteveC

谢谢SteveC,在我发布此链接时,这两个链接都有效,而且我不确定哪个是官方网站,现在Novell似乎是该lib的正确网站。
模拟对象

感谢kzu的注意,我已经将原始答案中的链接更新为也指向github位置。
模拟对象2012年

1

这是一种基于我在本页上找到的内容以及在shell32对象帮助下读取(而不是写入)扩展属性的解决方案

要明确的是,这是一个hack。该代码看起来仍将在Windows 10上运行,但会击中某些空属性。Windows的早期版本应使用:

        var i = 0;
        while (true)
        {
            ...
            if (String.IsNullOrEmpty(header)) break;
            ...
            i++;

在Windows 10上,我们假设有大约320个要读取的属性,只是跳过了空条目:

    private Dictionary<string, string> GetExtendedProperties(string filePath)
    {
        var directory = Path.GetDirectoryName(filePath);
        var shell = new Shell32.Shell();
        var shellFolder = shell.NameSpace(directory);
        var fileName = Path.GetFileName(filePath);
        var folderitem = shellFolder.ParseName(fileName);
        var dictionary = new Dictionary<string, string>();
        var i = -1;
        while (++i < 320)
        {
            var header = shellFolder.GetDetailsOf(null, i);
            if (String.IsNullOrEmpty(header)) continue;
            var value = shellFolder.GetDetailsOf(folderitem, i);
            if (!dictionary.ContainsKey(header)) dictionary.Add(header, value);
            Console.WriteLine(header +": " + value);
        }
        Marshal.ReleaseComObject(shell);
        Marshal.ReleaseComObject(shellFolder);
        return dictionary;
    }

如前所述,您需要引用Com程序集Interop.Shell32。

如果收到与STA相关的异常,则可以在此处找到解决方案:

使用Shell32获取文件扩展属性时发生异常

我不知道这些属性名称在外部系统上是什么样的,并且找不到用于访问字典的可本地化常量的信息。我还发现返回的字典中并不存在“属性”对话框中的所有属性。

顺便说一句,这非常慢,而且-至少在Windows 10上-解析检索到的字符串中的日期将是一个挑战,因此使用它似乎是一个坏主意。

在Windows 10,你一定要使用Windows.Storage库,包含SystemPhotoProperties,SystemMusicProperties等 https://docs.microsoft.com/en-us/windows/uwp/files/quickstart-getting-file-properties

最后,我发布一个更好的解决方案,它使用WindowsAPICodePack

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.