操作数类型冲突:nvarchar与用户定义的表类型不兼容

问题描述:

我在这里是新手,目前遇到麻烦,我的方案是我想从Excel插入和更新数据到SQL Server表中。操作数类型冲突:nvarchar与用户定义的表类型不兼容

对于插入部分它完美的作品,但是当涉及到更新时,我不知道该怎么做。我搜索了很少的方法,我发现这对我来说最适合使用存储过程。

这是我现在使用的代码。当我尝试它给了我这个错误:

Operand type *: nvarchar is incompatible with user-defined table type

--- Stored procedure --- 
CREATE PROCEDURE [dbo].[chkUpdate] 
    @Operator IC_CHK READONLY 
AS 
BEGIN 
    set nocount on; 

    MERGE INTO tb_Operator c1 
    USING @Operator c2 ON c1.IC = c2.IC 

    WHEN MATCHED THEN 
     UPDATE SET 
      c1.Name = c2.Name, 
      --c1.IC = c2.IC, 
      c1.Email = c2.Email, 
      c1.Status = c2.Status, 
      c1.Datetime = c2.Datetime 

    WHEN NOT MATCHED THEN 
     INSERT VALUES(c2.Name, c2.IC, c2.Email, c2.[Status], c2.[Datetime]); 
end 

--- User-defined table type --- 
CREATE TYPE [dbo].[IC_CHK] as table 
(
    [Id] [int] NULL, 
    [Name] [nvarchar](50) NULL, 
    [IC] [bigint] NULL, 
    [Email] [nvarchar](MAX) NULL, 
    [Status] [nvarchar](50) NULL, 
    [Datetime] [datetime] NULL 
) 

VS 2010代码:

protected void btnImport_Click1(object sender, EventArgs e) 
{ 
    int i = 0; 

    try 
    { 
     string path = string.Concat(Server.MapPath("~/Excel/" + UploadExcel.FileName)); 
     UploadExcel.SaveAs(path); 

     String strCon = string.Format("Provider=Microsoft.Ace.OLEDB.12.0;Data Source={0}; Extended Properties=Excel 12.0;",path); 
     OleDbDataAdapter myda = new OleDbDataAdapter("SELECT * FROM [sheet1$]", strCon); 

     DataTable myds = new DataTable(); 
     myda.Fill(myds); 

     for (i = 0; i <= myds.Rows.Count - 1; i++) 
     { 
      String constr = ConfigurationManager.ConnectionStrings["conn"].ConnectionString; 

      using (SqlConnection con = new SqlConnection(constr)) 
      using (SqlCommand cmd = new SqlCommand("chkUpdate")) 
      { 
       cmd.CommandType = CommandType.StoredProcedure; 
       cmd.Connection = con; 

       cmd.Parameters.AddWithValue("@Operator", path); 

       con.Open(); 
       cmd.ExecuteNonQuery(); 
       con.Close(); 
      } 
     } 

     MsgBox1.alert("Import success"); 
     View.Visible = true; 

     vBinds(); 
    } 
    catch (Exception ex) 
    { 
     MsgBox1.alert(ex.Message); 
    } 
} 

做检查,我和我欣赏它。谢谢

P/S:我再次确认我的用户定义表类型与我的表具有相同的数据类型。

+0

你应该检查(HTTP [我们可以停止使用AddWithValue()了吗?]://blogs.msmvps .com/jcoehoorn/blog/2014/05/12/can-we-stop-using-addwithvalue-already /)并停止使用'.AddWithValue()' - 它可能会导致意想不到的结果... –

+0

加载你的数据到一个'DataTable'中,如果你想传递一个表值参数,这是正确的。然后你忽略它,而是将字符串* path *传递到数据的起始位置,而不是将DateTable对象传入。 –

+0

@Damien_The_Unbeliever谢谢我刚刚注意到这一点。 – TheButterfly

在您的MERGE声明中的INSERT中,我会建议明确定义您要插入的列。最有可能的是,这是错误的原因 - 你插入你的列 - 但你没有指定哪些目标列应插入。

既然你不指定,你必须为表中的每一列提供值,在定义它们的确切顺序 - 是真的如此?例如。你在表格中插入什么ID列?

假设您的实际数据库表中的列IDIDENTITY列,我会用(否则,你必须列出ID在列的列表中插入以及和的VALUES列表提供一个值值):

WHEN NOT MATCHED THEN 
    INSERT(Name, IC, Email, [Status], [DateTime]) 
    VALUES(c2.Name, c2.IC, c2.Email, c2.[Status], c2.[Datetime]); 

,我会还建议不使用T-SQL保留关键字statusdatetime为您列名 - 你只是要求这样做的麻烦。使用更具表现的名字 - 这确实涉及到你的业务领域 - 不仅datetime .....

+0

感谢您的建议,并修复了错误。我会为刚刚发现的错误而自行处理;) – TheButterfly