操作数类型冲突:nvarchar与用户定义的表类型不兼容
我在这里是新手,目前遇到麻烦,我的方案是我想从Excel插入和更新数据到SQL Server表中。操作数类型冲突:nvarchar与用户定义的表类型不兼容
对于插入部分它完美的作品,但是当涉及到更新时,我不知道该怎么做。我搜索了很少的方法,我发现这对我来说最适合使用存储过程。
这是我现在使用的代码。当我尝试它给了我这个错误:
Operand type clash: 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:我再次确认我的用户定义表类型与我的表具有相同的数据类型。
在您的MERGE
声明中的INSERT
中,我会建议明确定义您要插入的列。最有可能的是,这是错误的原因 - 你插入你的列 - 但你没有指定哪些目标列应插入。
既然你不指定,你必须为表中的每一列提供值,在定义它们的确切顺序 - 是真的如此?例如。你在表格中插入什么ID
列?
假设您的实际数据库表中的列ID
是IDENTITY
列,我会用(否则,你必须列出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保留关键字status
或datetime
为您列名 - 你只是要求这样做的麻烦。使用更具表现的名字 - 这确实涉及到你的业务领域 - 不仅datetime
.....
感谢您的建议,并修复了错误。我会为刚刚发现的错误而自行处理;) – TheButterfly
你应该检查(HTTP [我们可以停止使用AddWithValue()了吗?]://blogs.msmvps .com/jcoehoorn/blog/2014/05/12/can-we-stop-using-addwithvalue-already /)并停止使用'.AddWithValue()' - 它可能会导致意想不到的结果... –
加载你的数据到一个'DataTable'中,如果你想传递一个表值参数,这是正确的。然后你忽略它,而是将字符串* path *传递到数据的起始位置,而不是将DateTable对象传入。 –
@Damien_The_Unbeliever谢谢我刚刚注意到这一点。 – TheButterfly