将字符串变量转换为 GUID

Posted

技术标签:

【中文标题】将字符串变量转换为 GUID【英文标题】:Convert string variable to GUID 【发布时间】:2018-04-25 17:29:12 【问题描述】:

我想使用Execute SQL Task 查询从数据库表中返回varbinary 数据。该查询需要UniqueIdentifier 作为参数,该参数以string 形式存储在Package 变量中。此查询的结果(varbinary 数据)将存储在 Byte 类型的第二个变量中。

下表显示了我的局部变量列表,注意 DocumentGuid 是由控制流的另一部分预先填充的

| Variable Name | Qualified Name     | Scope   | Data Type | Comments                                 |
|---------------|--------------------|---------|-----------|------------------------------------------|
| DocumentGuid  | User::DocumentGuid | Package | String    | Used to store a GUID value               |
| DocumentData  | User::DocumentData | Package | Byte      | Used to hold varbinary data from a query |

当我尝试在这样的Execute SQL Task 查询中使用它时:

SELECT DocData 
FROM  docsRepo
WHERE DocumentGuid = ?

传入参数为

| Variable name      | Direction | Data Type | Parameter Name | Parameter Size |
|--------------------|-----------|-----------|----------------|----------------|
| User::DocumentGuid | Input     | GUID      | 0              | 100            |

和结果集

| Result Name | Variable Name      |
|-------------|--------------------|
| DocData    | User::DocumentData |

我收到以下错误:

[执行 SQL 任务] 错误:执行查询“SELECT DocData FROM dbo.docsRepo...”失败,出现以下错误:“转换失败 从字符串转换为唯一标识符时。”。 可能的失败原因:查询有问题,“ResultSet” 属性设置不正确,参数设置不正确,或 连接未正确建立。

我在这里遗漏了某种基本逻辑吗?以下脚本在 SQL Server 中运行良好:

SELECT DocData 
FROM  docsRepo
WHERE DocumentGuid = '53A394A7-5D2B-40C0-A04D-90553E4535C3'

【问题讨论】:

将参数类型改为字符串。我知道,它是一个 guid,但由于它是一个字符串变量,所以保持数据类型对齐并让后端 rdbms 处理转换 - 这行得通吗? @billinkc 知道了。错误消息不是来自查询执行,而是来自 Execute SQL Task 试图获取字符串参数并将其转换为 GUID。 SSIS 无法做到这一点,因此在您进入框内的查询之前它会令人窒息。 【参考方案1】:

我对您的示例进行了两次更改,并且能够使其正常工作。

首先是在执行 SQL 任务中将参数类型更改为字符串。我知道,它是一个 guid,但由于它是 SSIS 中的字符串变量,因此保持数据类型对齐并让后端 rdbms/驱动程序处理转换。

我更改的另一件事是 DocData 的数据类型。您定义为字节但从未使用过,我担心它是一个完整的字节,而不是字节。无论如何,对于我创建的表,使用 SSIS 的 Object 数据类型使其工作。

来源查询

这是我的表格,里面有一个值

IF NOT EXISTS
(
    SELECT
        *
    FROM
        sys.tables
    WHERE
        name = 'docsRepo'
)
BEGIN
    CREATE TABLE dbo.docsRepo
    (
        DocumentGuid uniqueidentifier
    ,   DocumentData varbinary(MAX)
    );

    INSERT INTO
        dbo.docsRepo
    SELECT
        '53A394A7-5D2B-40C0-A04D-90553E4535C3'
    ,   CAST('Hello world' AS varbinary(MAX));
END;

检索查询

SELECT D.DocumentData FROM dbo.docsRepo AS D WHERE D.DocumentGuid = ?;

配置了完整的结果集。使用了 OLE DB 驱动程序。参数名称 0,数据类型 varchar,变量 User::DocumentGuid。结果选项卡,我的结果集名称为 0,变量 User::DocumentData

Biml 将创建一个 SSIS 包来演示所有这些

<Biml xmlns="http://schemas.varigence.com/biml.xsd">
    <Connections>
        <OleDbConnection Name="localhost" ConnectionString="Provider=SQLNCLI11;Integrated Security=SSPI;Data Source=.\dev2014;Initial Catalog=tempdb" />
    </Connections>
    <Packages>
        <Package Name="SO_50028154" ConstraintMode="Linear">
            <Variables>
                <Variable Name="DocumentGuid" DataType="String">53A394A7-5D2B-40C0-A04D-90553E4535C3</Variable>
                <Variable Name="DocumentData" DataType="Object" />
            </Variables>
            <Tasks>
                <ExecuteSQL Name="SQL GenerateTable" ConnectionName="localhost">
                    <DirectInput>IF NOT EXISTS(SELECT * FROM sys.tables WHERE name='docsRepo')BEGIN CREATE TABLE dbo.docsRepo(DocumentGuid uniqueidentifier, DocumentData varbinary(max)); INSERT INTO dbo.docsRepo SELECT '53A394A7-5D2B-40C0-A04D-90553E4535C3', CAST('Hello world' AS varbinary(max)); END</DirectInput>
                </ExecuteSQL>
                <ExecuteSQL Name="SQL Retrieve data" ConnectionName="localhost" ResultSet="Full">
                    <DirectInput>SELECT D.DocumentData FROM dbo.docsRepo AS D WHERE D.DocumentGuid = ?;</DirectInput>
                    <Parameters>
                        <Parameter DataType="AnsiString" Name="0" VariableName="User.DocumentGuid" />
                    </Parameters>
                    <Results>
                        <Result Name="0" VariableName="User.DocumentData" />
                    </Results>
                </ExecuteSQL>
            </Tasks>
        </Package>
    </Packages>
</Biml>

【讨论】:

感谢@billinkc 的详尽解释。事实证明,我在变量数据类型/参数和结果映射方面犯了一些错误。非常感谢,您提供的内容绝对正确!【参考方案2】:

你可以试试明确的CAST:

SELECT DocData 
FROM  docsRepo
WHERE DocumentGuid = CAST(? AS UNIQUEIDENTIFIER);

【讨论】:

你能发布你的整个查询吗? 谢谢@lad2025 我试试看 感谢您的解释。事实证明,使用 Billnkc 的解释,SQL 实际上可以自己处理转换,而不是尝试将其显式转换为特定类型。然而,这是一种非常有用的技术,供将来使用。非常感谢您的帮助【参考方案3】:

看起来您可能有存储在 DocumentGuid 中的值,这些值不是有效的 GUID,或者是您传递给查询的无效 GUID

要查找您的表是否包含无效的 GUID,您可以运行:

SELECT DocumentGuid
FROM  docsRepo
WHERE TRY_CAST(DocumentGuid AS UNIQUEIDENTIFIER) IS NULL

【讨论】:

【参考方案4】:

当您想将您的 uniqueIdentifer 作为 varchar 发送到您的执行 SQL 任务时,您可以在您的执行 SQL 任务中使用此查询:

DECLARE @docGuid AS UniqueIdentifier 
SET @docGuid  = CAST ( '' + ? + '' AS UniqueIdentifier )

SELECT DocData 
FROM  docsRepo
WHERE DocumentGuid = @docGuid

那个?通过参数作为 varchar 发送,大小为 50

【讨论】:

以上是关于将字符串变量转换为 GUID的主要内容,如果未能解决你的问题,请参考以下文章

guid与Base64编码互相转换

将字符串转换为 uniqueidentifier 时失败的解决方法

GUID转换成16位19位22位唯一字符串

使用python将字符串转换为字典的有用方法

C# GUID转换成16位字符串或19位数字并确保唯一

用于 URL 的 Guid 的最短编码