1

为什么 SQLCMD 的存储输出只有Length属性而不是column names?。sqlcmd不能用它的属性来存储输出吗?

Invoke-sqlcmd正确存储它但Invoke-SQLcmd需要更长的时间来处理,所以我试图让它与 SQLcmd 一起工作,因为这个方法将是不同脚本的一部分,这些脚本计划每分钟运行一次,每小时运行一次等,

知道这是否可能或问题是什么?

存储输出并回显 $var:

     PS C:> $var=(SQLCMD -S 'x.x.x.x' -U 'user' -P 'password' -i "C:\query.sql" -W -m 1) 

     PS C:> $var
     job_id name
     ------ ----
     12345-aaaa-1234-5678-000000000000000 Clear DB entries
     12345-bbbb-1234-5678-000000000000000 TempLog DB

Echo $var[0,1,2] 不显示属性名称。

     PS C:> $var[0]
     job_id name
     PS C:> $var[1]
     ------ ----
     PS C:> $var[2]
     12345-aaaa-1234-5678-000000000000000 Clear DB entries

显示 $var 属性

     PS C:> $var | select *
     Length
     ------
     11
     53

显示 $var 类型

     PS C:> $var.GetType()

     IsPublic IsSerial Name                                     BaseType
     -------- -------- ----                                     --------
     True     True     Object[]                                 System.Array
4

2 回答 2

4
$var=(SQLCMD -S 'x.x.x.x' -U 'user' -P 'password' -i "C:\query.sql" -W -m 1) 

您正在调用sqlcmd.exe,它不知道 .Net 对象是什么,更不用说如何将它们传递给 PowerShell。就 PowerShell 而言,该命令输出字符串。您需要自己将字符串转换为对象。

如果你必须使用sqlcmd.exe,我会建议这样的事情:

$Delimiter = "`t"
$var = SQLCMD -S 'x.x.x.x' -U 'user' -P 'password' -i "C:\query.sql" -W -m 1 -s $Delimiter |
    ConvertFrom-Csv -Delimiter $Delimiter |
    Select-Object -Skip 1

我使用制表符作为字段分隔符。如果您的数据包含制表符,您将需要不同的分隔符。如果您的数据包含双引号,您也可能会遇到问题。这Select-Object -Skip 1是跳过sqlcmd始终在标题下方创建的下划线行。

另请注意,您应该使用-w参数 onsqlcmd以防止任何不正确的包装。另请注意,空值始终作为文字字符串输出NULL

也就是说,我仍然可能会坚持使用Invoke-Sqlcmd. 它更不容易出错并且更可预测。如果我真的需要性能,我可能会使用直接的 .Net 方法或 SSIS。

于 2018-03-14T18:35:01.823 回答
0

我为此目的编写了一个函数......没有完全充实......希望它有所帮助

    function Invoke-MSSqlCommand
    {
      [CmdletBinding()]
      param
      (
        [Parameter(Position=0, Mandatory=$true)]
        [ValidateNotNullOrEmpty()]
        [string]
        $Query,

        [Parameter(Position=1, Mandatory=$true)]
        [ValidateNotNullOrEmpty()]
        [string]
        $ConnectionString,

        [Switch]
        $NoOutput
      )
      try {
        $connection = New-Object -TypeName System.Data.SqlClient.SqlConnection
        $connection.ConnectionString = $ConnectionString
        $null = $connection.Open()
      }
      catch {
        Throw "$connectionstring could not be contacted"
      }
      $command = New-Object -TypeName System.Data.SqlClient.SqlCommand
      $command.CommandText = $query
      $command.Connection = $connection


      if ($NoOutput) {
        $null = $command.ExecuteNonQuery()
      }
      else {

        if ($dataset.Tables[0].Rows[0] -eq $null) {
          write-verbose -Message 'no record'
          $connection.Close()
          return $null
        }

        $dataset.Tables[0].Rows
        $connection.close()
      }
    }
于 2018-03-14T18:43:18.620 回答