ORA-01008绑定了所有变量

时间:2011-04-15 15:20:55

标签: c# oracle oracleclient ora-01008

我正在使用System.Data.OracleClient按名称进行参数绑定,并验证CommandText和Parameters是否同步:

    public string CommandText { get; set; }
    public IEnumerable<OracleParameter> Parameters { get; set; }

    private void VerifyThatAllParametersAreBound()
    {
        var variableNames = Regex.Matches(CommandText, ":\\w+")
            .Cast<Match>().Select(m => m.Value).ToArray();
        var parameteterNames = Parameters.Select(p => p.ParameterName).ToArray();

        var unboundVariables = variableNames.Except(parameteterNames).ToArray();
        if (unboundVariables.Length > 0)
        {
            throw new Exception("Variable in CommandText missing parameter: "
                + string.Join(", ", unboundVariables) + ".");
        }

        var unboundParameters = parameteterNames.Except(variableNames).ToArray();
        if (unboundParameters.Length > 0)
        {
            throw new Exception("Parameter that is not used in CommandText: "
                + string.Join(", ", unboundParameters) + ".");
        }
    }

还有一个查询会引发ORA-01008: not all variables bound。当手动将参数值插入有问题的CommandText时,查询运行,因此CommandText和Parameters-values应该没问题。我正在使用:作为变量和参数名的前缀,它适用于其他查询。

如何查明此异常的原因?

5 个答案:

答案 0 :(得分:9)

错误是没有为null值指定DBNull.Value。所以

new OracleParameter(":Foo", item.Foo)

必须预先放置

item.Foo == null 
    ? new OracleParameter(":Foo", DBNull.Value) 
    : new OracleParameter(":Foo", item.Foo)

我认为它早期使用ODT.NET而没有空检查,但尚未确认。显然System.Data.OracleClient正在删除带有空值的参数。

答案 1 :(得分:3)

如果将 null 作为参数值传递,则会得到“并非所有变量都绑定”如果传递 DBNull.Value ,则会在OracleClient中的某处产生运行时错误。要传递NULL,请使用 string.Empty ,OracleClient会将其转换为任何数据库类型的 NULL

答案 2 :(得分:0)

我相信微软大约2年前已将deprecated OracleClient作为ADO.NET的一部分。

您可能需要考虑使用Oracle的数据访问组件(ODAC odp.net)。使用OracleParameter类可以轻松构建(并检查计数)参数。设置并安装找到here的文档。哦,你也可以进入他们的实体框架(和LINQ)支持(我觉得还是beta?)。

无论如何要认真考虑的事情。

答案 3 :(得分:0)

如果您有多个参数,则需要将 BindByName 设置为 true 。例如:

OracleCommand cmd = con.CreateCommand();

cmd.BindByName = true;

cmd.Parameters.Add(new OracleParameter("parameter1", parameter1));
cmd.Parameters.Add(new OracleParameter("parameter2", parameter2));

答案 4 :(得分:0)

基于上述答案和评论,我确保采取以下措施来解决此问题:

  • 参数的绑定顺序与查询中出现的顺序相同
  • 指定了参数类型
  • 如果在SQL中多次要求相同的参数值,请在SQL中以不同的方式命名每个参数(不确定是否需要)

    OracleParameter [] orclParams =新的OracleParameter [] { 新的OracleParameter { ParameterName =“ param1”, OracleDbType = OracleDbType.Varchar2, 值=“ abc”}, 新的OracleParameter { ParameterName =“ param2”, OracleDbType = OracleDbType.Varchar2, 值=“ abc”}, 新的OracleParameter { ParameterName =“ date1”, OracleDbType = OracleDbType.Date, 值= myDate} }; SomeFunction(sqlQuery,orclParams.ToList());