文件日期元数据未正确显示

时间:2014-08-24 16:51:00

标签: datetime powershell com metadata powershell-v3.0

我一直在尝试编写基于某些code online的Powershell脚本,该脚本将从图片,视频和其他文件中读取元数据信息,然后根据其中一个日期对其进行排序(目前所采用的日期似乎是最好的选择,如果它可用,但修改日期适用于尚未更改的文件。)

但是,当我运行脚本并提取信息时,我无法将字符串转换为日期。这里大致是我如何通过COM对象获取信息:

PS C:/> $objShell = New-Object -ComObject Shell.Application
PS C:/> $objFolder = $objShell.namespace("C:\MyFolder")
PS C:/> $date = $objFolder.GetDetailsOf($objFolder.Items().Item(0), 12)
PS C:/> $date

7/‎10/‎2014 ‏‎7:09 PM

问题是我应该能够将其转换为日期时间对象。例如,如果我在其中手动编写它的工作原理:

PS C:/> [datetime]::ParseExact("7/10/2014 7:09 PM","g",$null)

Thursday, July 10, 2014 7:09:00 PM

但如果我替换变量,它就不起作用了:

PS C:/> [datetime]::ParseExact($date,"g",$null)
Exception calling "ParseExact" with "3" argument(s): "String was not recognized as a valid DateTime."
At line:1 char:1
+ [datetime]::ParseExact($date,"g",$null)
+ ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
    + CategoryInfo          : NotSpecified: (:) [], MethodInvocationException
    + FullyQualifiedErrorId : FormatException

这很可能是因为变量实际上并不是我所看到的。事实上它更长。更不用说如果你遍历所有角色,你可以看到额外长度的来源:

PS C:\> $date.Length #should be about 16, you'd think

22

PS C:\> $datearray = @()
PS C:\> for ($i = 0; $i -lt $date.Length; $i++) {$datearray += $date[$i]}
PS C:\> $datearray #i'm printing on one line and in quotes for ease of viewing

" 7/ 10/ 2014   7:09 PM"

如果您尝试使用连接或类似的东西打印数组,结果(对我来说,不知道发生了什么)是不可预测的。它对待它就像它有22个字符,但打印忽略空格。

我确信我可以花一点时间做一些字符串格式化,但我宁愿能够解析给定的日期。发生了什么事?

编辑:我可以轻松访问文件信息,但我不愿意。我主要关注的是为什么我看到的结果是不稳定的,并且显示的长度与打印的结果不符,以及我如何处理它们。如果不出意外,我很好奇发生了什么。

2 个答案:

答案 0 :(得分:4)

我想知道多余的角色是什么(你不会提到已经看过这个。)。我将您的数组代码$datearray += $date[$i]更新为$datearray += [int][char]$date[$i]截断输出显示两个奇怪的82078206,它们转换为从左到右的标记和从右到左的标记。它们通常与html相关联。不幸的是,我无法提供他们的存在的见解。好消息是它们很容易删除。

$date = ($date -replace [char]8206) -replace [char]8207
[datetime]::ParseExact($date,"g",$null)

哪个网络输出

Thursday, March 26, 2009 1:43:00 PM

希望这比您想要的更接近。我试着寻找那些ascii代码存在的原因,但我没有找到任何有用的东西。

其他读者的额外信息

我这样做是因为我不知道索引12是什么。所以我创建了一个数组,其中包含所有可能的文件元数据的友好名称(288个条目!)。为简洁起见,我将here-string设置为here

有了这个我正在测试以下代码对我的照片。

$objShell = New-Object -ComObject Shell.Application
$objFolder = $objShell.namespace("C:\Temp\")
0..$Meta.GetUpperBound(0)| %{
    $metaValue = $objFolder.GetDetailsOf($objFolder.Items().Item(2), $_)
    If ($metaValue) {Write-Host "$_ - $($meta[$_]) - $metaValue"}
    } 

$Meta是我之前谈到过的数组。 代码将循环显示我的文件的所有详细信息,由Item(2)表示,写入屏幕显示包含值的所有文件详细信息。最后有一行将字符串转换为日期值。脚本输出

0 - Name - IMG_0571.JPG
1 - Size - 3.12 MB
2 - Item type - JPEG image
3 - Date modified - 3/26/2009 3:34 PM
4 - Date created - 8/24/2014 5:19 PM
5 - Date accessed - 8/24/2014 5:19 PM
6 - Attributes - A
9 - Perceived type - Image
10 - Owner - TE_ST\Cameron
11 - Kind - Picture
12 - Date taken - ‎3/‎26/‎2009 ‏‎1:43 PM
19 - Rating - Unrated
30 - Camera model - Canon EOS DIGITAL REBEL XS
31 - Dimensions - ‪3888 x 2592‬
32 - Camera maker - Canon
53 - Computer - TE_ST (this computer)
155 - Filename - IMG_0571.JPG
160 - Bit depth - 24
161 - Horizontal resolution - ‎72 dpi
162 - Width - ‎3888 pixels
....output truncated....
247 - Program mode - Normal program
250 - White balance - Auto
269 - Sharing status - Not shared

答案 1 :(得分:1)

当我运行你的代码时,$objFolder.GetDetailsOf($objFolder.Items().Item(0), 12)我得到一个空字符串。我甚至更改了项目编号和我正在查找的文件夹,以确保我收到文件对象。

但是,如果我这样做:

$objShell = New-Object -ComObject Shell.Application
$objFolder = $objShell.namespace("C:\Temp")
$date = $objFolder.Items().Item(3).ModifyDate

我得到一个已经是DateTime对象的值。

(上面的代码使用我的文件夹和项目索引)