3

我正在尝试使用 .net API 来查找大型数据文件。由于某种原因,我无法使其工作。这是我的代码:

function check_logs{
  $pos = 8192
  $count = 1
  $path = 'C:\Program Files\Microsoft SQL Server\MSSQL10.MSSQLSERVER\MSSQL\Log\ERRORLOG.2'
  $br = 0
  $reader = [System.IO.File]::OpenText($path)
  $reader.DiscardBufferedData()
  $reader.BaseStream.Seek(0, [System.IO.SeekOrigin]::Begin)
    for(;;){
    $line = $reader.ReadLine()
    if($line -ne $null){$br = $br + [System.Text.Encoding]::UTF8.GetByteCount($line)}
    if($line -eq $null -and $count -eq 0){break}
    if($line -eq $null){$count = 0}
    elseif($line.Contains('  Error:')){
        Write-Host "$line  $br"
    }
}

}

如果我使用 0 作为 seek 方法的参数,它会按预期从头开始搜索,但它也会在写入读取的行之前将 0 写入控制台。例子:

 0
 2011-08-31 09:26:36.31 Logon       Error: 17187, Severity: 16, State: 1.  4101
 2011-08-31 09:26:36.32 Logon       Error: 17187, Severity: 16, State: 1.  4489
 2011-08-31 09:26:38.25 Logon       Error: 17187, Severity: 16, State: 1.  4929
 2011-08-31 09:26:38.25 Logon       Error: 17187, Severity: 16, State: 1.  5304
 2011-08-31 09:26:43.75 Logon       Error: 17187, Severity: 16, State: 1.  6120

如果我尝试使用 4096 而不是 0 它只会写出:

4096

我原以为除了前两行之外,它会写出与第一行相同的行。

有人可以看到问题吗?我有另一个问题让我想到了这个。有关更多背景,请参阅

编辑:仍在试图解决这个问题。有谁知道我还能在哪里找到有关此问题的信息?是否可以向 Microsoft 脚本专家发送问题?

此致

吉斯利

4

3 回答 3

5

Seek 方法返回流中的新位置,这就是打印数字的原因。

至于为什么你没有得到输出:

  1. 确认文件大小大于 4K。
  2. 尝试打印出所有行,而不仅仅是其中包含“错误”一词的行。这可能会给你一个线索
  3. StreamReader 是基本流周围的缓冲包装器,因此 Seek 和 Position 可能不会像您期望的那样工作。考虑http://geekninja.blogspot.com/2007/07/streamreader-annoying-design-decisions.html。尝试在搜索$reader.DiscardBufferedData() 之前添加调用。
于 2011-09-01T12:05:27.193 回答
0

所以我终于找到了答案。由于某种我不知道的原因,我必须使用二进制阅读器。下面是我的完整功能:

 function check_logs{
 Write-Host "New test `n`n"
 $pos = 19192
 $path = 'C:\Program Files\Microsoft SQL Server\MSSQL10.MSSQLSERVER\MSSQL\Log\ERRORLOG.2'
 $br = 0
 $b = new-object System.IO.BinaryReader([System.IO.File]::Open($path,[System.IO.FileMode]::Open));
 $required = $b.BaseStream.Length - $pos
 $b.BaseStream.Seek($pos, [System.IO.SeekOrigin]::Begin)
 $bytes = $b.ReadBytes($required)
 $log = [System.Text.Encoding]::Unicode.GetString($bytes)
 $split = $log.Split("`n")
 foreach($s in $split)
 {
     if($s.contains("  Error:"))
     {
         Write-Host $s  "`n"
     }
 }
 $b.close
 }

谢谢您的帮助

吉斯利

于 2011-09-12T11:46:22.633 回答
0

我有一个类似的问题。寻找的位置正在控制台上打印。我只需要将返回值分配给某个变量,就解决了问题。

所以而不是:

$reader.BaseStream.Seek(0, [System.IO.SeekOrigin]::Begin)

我不得不写这样的东西:

$pos = $reader.BaseStream.Seek(0, [System.IO.SeekOrigin]::Begin)

问候, Thejasvi V

于 2014-01-30T05:38:51.773 回答