背景:
前两天由于工作需要做个业务单据接口(支持批量处理),一般有接口发布为批量,但访问数据库时还是一张张单据处理,本次访问数据库也是批量操作。
内容:
研究发现Oracle批量操作有两种:1)BulkCopy 2)OracleCommand 。由于BulkCopy 的操作方式(表名、列映射、数据源)仅支持添加,无法做更新操作,所以不考虑此方式。
OracleCommand 在支持批量时,属性ArrayBindCount的值是至关重要的,它代表着一批的量,即如果你提供的参数的数组长度大于它的值,将以ArrayBindCount的值为长度标尺,大于这个的数据将被丢弃。本次准备:1)创建一张 student表 (ID int,Name varchar2(32)),引用Oracle.DataAccess
OracleConnection cn = new OracleConnection(); cn.ConnectionString = ConfigurationManager.ConnectionStrings["Oracle"].ConnectionString; OracleCommand cmd = new OracleCommand(); cmd.CommandText = "Insert into student(id,name) values(:id,:name)"; cmd.Connection = cn; List<int> lsID=new List<int>(); List<string> lsName=new List<string>(); for (int i = 0; i < 10; i++) { lsID.Add(i); lsName.Add("N"+i.ToString()); } cmd.ArrayBindCount = 10;//如果小于10 如8 则新增的行数 为8 而不是10 OracleParameter oracleParameterID = new OracleParameter(); oracleParameterID.ParameterName = "ID"; oracleParameterID.OracleDbType = OracleDbType.Int16; oracleParameterID.Direction = ParameterDirection.Input; oracleParameterID.Value = lsID.ToArray(); OracleParameter oracleParameterName = new OracleParameter(); oracleParameterName.ParameterName = "Name"; oracleParameterName.OracleDbType = OracleDbType.Varchar2; oracleParameterName.Direction = ParameterDirection.Input; oracleParameterName.Value = lsName.ToArray(); cmd.Parameters.Add(oracleParameterID); cmd.Parameters.Add(oracleParameterName); cn.Open(); var t=cmd.ExecuteNonQuery(); cn.Close();实验证明 批量新增、修改、删除 均 ok。但是在获取数据时,悲剧的发现不ok。
有点小小的遗憾,查了半天也没有发现原因,经讨论有三种办法:
1)拼接字符串,把条件拼接出来(例子中用in 就可以,但是实际中需要如((A=a and B=b)or(A=a1 and B=b1) …) ) 不易过长,大家知道这需要编译时间的;
2)创建临时表,把条件都插入到临时表中,然后关联临时表查询(比较麻烦,但速度还不错);
3)Command执行多次,把返回结果拼接起来 如(使用OracleDataAdapter.fill (datatable) 多次填充的时候,datatable 即为最终的查询结果集) 。
本次就记录到这吧,如果后续有更多思路再更新。