我一直在尝试编写基于某些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个字符,但打印忽略空格。
我确信我可以花一点时间做一些字符串格式化,但我宁愿能够解析给定的日期。发生了什么事?
编辑:我可以轻松访问文件信息,但我不愿意。我主要关注的是为什么我看到的结果是不稳定的,并且显示的长度与打印的结果不符,以及我如何处理它们。如果不出意外,我很好奇发生了什么。
答案 0 :(得分:4)
我想知道多余的角色是什么(你不会提到已经看过这个。)。我将您的数组代码$datearray += $date[$i]
更新为$datearray += [int][char]$date[$i]
。 截断输出显示两个奇怪的8207
和8206
,它们转换为从左到右的标记和从右到左的标记。它们通常与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
对象的值。
(上面的代码使用我的文件夹和项目索引)