我使用通用方法从任何Oracle表接收单行并使用下面的代码将其显示在datagridview中。但是,如果表包含一个float类型的列,并且该值具有大量小数位,那么我会在“MyReader.GetValues(objCells)”行中获得“算术运算导致溢出”;
oCmd.CommandText = "OTCMIADM.OTCMI_GUI.GET_ROW";
oCmd.CommandType = CommandType.StoredProcedure;
oCmd.Parameters.Add("PI_TABLE_NAME", OracleDbType.Varchar2, 40).Value = cmbStagingTables.SelectedItem;
oCmd.Parameters.Add("PI_ROWID", OracleDbType.Varchar2, 40).Value = txtRowID.Text;
oCmd.Parameters.Add(new OracleParameter("PIO_CURSOR", OracleDbType.RefCursor)).Direction = ParameterDirection.Output;
oCmd.ExecuteNonQuery();
// clear the datagrid in preperation for loading
dgvStagingTable.Columns.Clear();
dgvStagingTable.Rows.Clear();
using (OracleDataReader MyReader = oCmd.ExecuteReader())
{
int ColumnCount = MyReader.FieldCount;
// add the column headers
DataGridViewColumn[] columns = new DataGridViewColumn[ColumnCount];
for (int i = 0; i < columns.Length; ++i)
{
DataGridViewColumn column = new DataGridViewTextBoxColumn();
column.FillWeight = 1;
column.HeaderText = MyReader.GetName(i);
column.Name = MyReader.GetName(i);
columns[i] = column;
}
dgvStagingTable.Columns.AddRange(columns);
// get the data and add the row
while (MyReader.Read())
{
//get all row values into an array
object[] objCells = new object[ColumnCount];
MyReader.GetValues(objCells);
//add array as a row to grid
dgvStagingTable.Rows.Add(objCells);
}
}
堆栈跟踪显示: 在Oracle.DataAccess.Types.DecimalConv.GetDecimal(IntPtr numCtx) 在Oracle.DataAccess.Client.OracleDataReader.GetDecimal(Int32 i) 在Oracle.DataAccess.Client.OracleDataReader.GetValue(Int32 i) 在Oracle.DataAccess.Client.OracleDataReader.GetValues(Object [] values)
所以我可以看出它导致错误的原因(假设是十进制转换);但是我该如何绕过这个?
我在使用以下命令加载数据之前尝试显式设置列的类型:
dgvStagingTable.Columns [“TR_THROUGHPUT_TIME_NO”]。ValueType = typeof(string);
和其他几种类型,但没有任何区别。
任何帮助表示感谢。
答案 0 :(得分:2)
我最初建议使用OracleDbTypeEx(http://download.oracle.com/docs/html/E15167_01/OracleParameterClass.htm#CHDJHDGE)为您解决此问题,这是错误的,所以这是一个新建议。
所以我做了什么:
Create Table Testdecimalteable(
Acol number(10) ,
DecCol NUMBER(38,38)
);
/
Insert Into Testdecimalteable
Select level,Level/(power(2,level))
From Dual
Connect By Level < 100 ;
/
Create or replace Procedure Testprocdecimal(Crs OUT Sys_Refcursor)
AS
Begin
Open Crs For
Select *
FROM Testdecimalteable ;
END Testprocdecimal ;
现在这将获得一些已知超出.net的数据。
然后是.net方:
OracleConnection _conn = new OracleConnection("" );
_conn.Open();
OracleCommand oCmd = new OracleCommand();
oCmd.CommandText = "Testprocdecimal";
oCmd.CommandType = CommandType.StoredProcedure;
oCmd.Connection = _conn;
OracleParameter crs = new OracleParameter();
crs.OracleDbType = OracleDbType.RefCursor ;
crs.Direction = ParameterDirection.Output;
crs.ParameterName = "crs";
oCmd.Parameters.Add(crs);
using (OracleDataReader MyReader = oCmd.ExecuteReader())
{
int ColumnCount = MyReader.FieldCount;
// get the data and add the row
while (MyReader.Read())
{
//MyReader.GetOracleValue(1).ToString()
Console.WriteLine(string.Format("{0}/{1}", MyReader.GetValue(0),MyReader.GetOracleValue(1).ToString() ));
}
}
这会将所有内容转换为字符串,但它会起作用。
http://download-east.oracle.com/docs/html/A96160_01/features.htm#1048038
我再次查看了您的初始查询,您正在调用两次查询:
oCmd.ExecuteNonQuery();
...
using (OracleDataReader MyReader = oCmd.ExecuteReader())
你不需要ExecuteNonQuery; ExecuteReader执行sp