将包含 Excel 公式的数据对象数组插入给定范围内的现有 Excel sheet
insert data object array which contains Excel formula's into existing Excel sheet on a given range
我有一个数据对象数组对象[] 来自通过 SqlDataAdapter 加载的 SQL 数据库 table,它包含 Excel 公式。
我想将对象粘贴到已打开的 Excel 文件中的特定范围内。
我应该补充一点,当数据table 包含像 "this is a test" 这样的字符串而不是像 "=SUM(A1:A5) 这样的 Excel 公式时,一切正常“
但我将 运行 保留在对我来说毫无意义的例外情况中。请指教!
这是我得到的。我忽略了什么?
private void InsertBridgeCalcBlock()
{
Excel.Application xlApp = (Excel.Application)System.Runtime.InteropServices.Marshal.GetActiveObject("Excel.Application");
try
{
xlActiveCell = xlApp.ActiveCell;
DataTable dt = new DataTable();
try
{
using (SqlDataAdapter da = new SqlDataAdapter("SELECT * FROM " + sDBBridgeCalcTable, conn))
{
da.Fill(dt);
}
}
catch (Exception ex)
{
MessageBox.Show("Error Message:" + Environment.NewLine + ex);
}
object[,] BridgeCalcTable = new object[dt.Rows.Count + 1,dt.Columns.Count];
for (var i = 0; i < dt.Rows.Count; i++)
for (var j = 1; j < dt.Columns.Count; j++)
{
BridgeCalcTable[i, j-1] = dt.Rows[i][j];
}
Excel.Range insertBridgeCalcTableRange = xlApp.Range[xlApp.ActiveSheet.Cells[xlActiveCell.Row-2, 11], xlApp.ActiveSheet.Cells[xlActiveCell.Row-2 + dt.Rows.Count - 1, 11 + dt.Columns.Count]]; // set insertrange
xlApp.ActiveSheet.EnableCalculation = true;
insertBridgeCalcTableRange.Value = BridgeCalcTable; // fill range with data
}
catch (Exception ex)
{
MessageBox.Show("Error Message:" + Environment.NewLine + ex);
}
}
异常显示为:
System.Runtime.InteropServices.COMException (0x800A03EC):
Exception from HRESULT: 0x800A03EC at System.RuntimeType.ForwardCallToInvokeMember(String memberName, BindingFlags flags, Object target, Int32[] aWarpperTypes, MessageData& msgData)
at Microsoft.Office.Interop.Excel.Rage.ser_Value(obejct value) at LookApp2016.Form1.InsertBridgeCalcBlock() in ~~myfilelocation~~ line 2886
虽然我仍然想知道为什么 Interop 在数据表中使用 Excel 公式时抛出异常,
我通过输入 Excel.Range.CopyFromRecordSet 找到了一个可能的答案。像这样:
ADODB.Recordset BridgeCalcRecordset = ConvertToRecordset(dt);
insertBridgeCalcTableRange.CopyFromRecordset(BridgeCalcRecordset);
但是,现在我的公式没有被自动识别为公式。我必须在计算开始之前编辑并退出每个单元格并且公式变为结果...
我使用以下方法解决了这个问题:
insertBridgeCalcTableRange.Formula = insertBridgeCalcTableRange.Value;
我的代码是相同的,但我使用此代码将我的数据集转换为 ADODB 记录集(他们说这样也更快):
归功于 http://www.nullskull.com/q/10057748/hi-all.aspx
的 Web Star
static public ADODB.Recordset ConvertToRecordset(DataTable inTable)
{
ADODB.Recordset result = new ADODB.Recordset();
result.CursorLocation = ADODB.CursorLocationEnum.adUseClient;
ADODB.Fields resultFields = result.Fields;
System.Data.DataColumnCollection inColumns = inTable.Columns;
foreach (DataColumn inColumn in inColumns)
{
resultFields.Append(inColumn.ColumnName
, TranslateType(inColumn.DataType)
, inColumn.MaxLength
, inColumn.AllowDBNull ? ADODB.FieldAttributeEnum.adFldIsNullable :
ADODB.FieldAttributeEnum.adFldUnspecified
, null);
}
result.Open(System.Reflection.Missing.Value
, System.Reflection.Missing.Value
, ADODB.CursorTypeEnum.adOpenStatic
, ADODB.LockTypeEnum.adLockOptimistic, 0);
foreach (DataRow dr in inTable.Rows)
{
result.AddNew(System.Reflection.Missing.Value,
System.Reflection.Missing.Value);
for (int columnIndex = 0; columnIndex < inColumns.Count; columnIndex++)
{
resultFields[columnIndex].Value = dr[columnIndex];
}
}
return result;
}
static ADODB.DataTypeEnum TranslateType(Type columnType)
{
switch (columnType.UnderlyingSystemType.ToString())
{
case "System.Boolean":
return ADODB.DataTypeEnum.adBoolean;
case "System.Byte":
return ADODB.DataTypeEnum.adUnsignedTinyInt;
case "System.Char":
return ADODB.DataTypeEnum.adChar;
case "System.DateTime":
return ADODB.DataTypeEnum.adDate;
case "System.Decimal":
return ADODB.DataTypeEnum.adCurrency;
case "System.Double":
return ADODB.DataTypeEnum.adDouble;
case "System.Int16":
return ADODB.DataTypeEnum.adSmallInt;
case "System.Int32":
return ADODB.DataTypeEnum.adInteger;
case "System.Int64":
return ADODB.DataTypeEnum.adBigInt;
case "System.SByte":
return ADODB.DataTypeEnum.adTinyInt;
case "System.Single":
return ADODB.DataTypeEnum.adSingle;
case "System.UInt16":
return ADODB.DataTypeEnum.adUnsignedSmallInt;
case "System.UInt32":
return ADODB.DataTypeEnum.adUnsignedInt;
case "System.UInt64":
return ADODB.DataTypeEnum.adUnsignedBigInt;
case "System.String":
default:
return ADODB.DataTypeEnum.adVarChar;
}
}
我有一个数据对象数组对象[] 来自通过 SqlDataAdapter 加载的 SQL 数据库 table,它包含 Excel 公式。
我想将对象粘贴到已打开的 Excel 文件中的特定范围内。
我应该补充一点,当数据table 包含像 "this is a test" 这样的字符串而不是像 "=SUM(A1:A5) 这样的 Excel 公式时,一切正常“
但我将 运行 保留在对我来说毫无意义的例外情况中。请指教!
这是我得到的。我忽略了什么?
private void InsertBridgeCalcBlock()
{
Excel.Application xlApp = (Excel.Application)System.Runtime.InteropServices.Marshal.GetActiveObject("Excel.Application");
try
{
xlActiveCell = xlApp.ActiveCell;
DataTable dt = new DataTable();
try
{
using (SqlDataAdapter da = new SqlDataAdapter("SELECT * FROM " + sDBBridgeCalcTable, conn))
{
da.Fill(dt);
}
}
catch (Exception ex)
{
MessageBox.Show("Error Message:" + Environment.NewLine + ex);
}
object[,] BridgeCalcTable = new object[dt.Rows.Count + 1,dt.Columns.Count];
for (var i = 0; i < dt.Rows.Count; i++)
for (var j = 1; j < dt.Columns.Count; j++)
{
BridgeCalcTable[i, j-1] = dt.Rows[i][j];
}
Excel.Range insertBridgeCalcTableRange = xlApp.Range[xlApp.ActiveSheet.Cells[xlActiveCell.Row-2, 11], xlApp.ActiveSheet.Cells[xlActiveCell.Row-2 + dt.Rows.Count - 1, 11 + dt.Columns.Count]]; // set insertrange
xlApp.ActiveSheet.EnableCalculation = true;
insertBridgeCalcTableRange.Value = BridgeCalcTable; // fill range with data
}
catch (Exception ex)
{
MessageBox.Show("Error Message:" + Environment.NewLine + ex);
}
}
异常显示为:
System.Runtime.InteropServices.COMException (0x800A03EC):
Exception from HRESULT: 0x800A03EC at System.RuntimeType.ForwardCallToInvokeMember(String memberName, BindingFlags flags, Object target, Int32[] aWarpperTypes, MessageData& msgData)
at Microsoft.Office.Interop.Excel.Rage.ser_Value(obejct value) at LookApp2016.Form1.InsertBridgeCalcBlock() in ~~myfilelocation~~ line 2886
虽然我仍然想知道为什么 Interop 在数据表中使用 Excel 公式时抛出异常,
我通过输入 Excel.Range.CopyFromRecordSet 找到了一个可能的答案。像这样:
ADODB.Recordset BridgeCalcRecordset = ConvertToRecordset(dt);
insertBridgeCalcTableRange.CopyFromRecordset(BridgeCalcRecordset);
但是,现在我的公式没有被自动识别为公式。我必须在计算开始之前编辑并退出每个单元格并且公式变为结果...
我使用以下方法解决了这个问题:
insertBridgeCalcTableRange.Formula = insertBridgeCalcTableRange.Value;
我的代码是相同的,但我使用此代码将我的数据集转换为 ADODB 记录集(他们说这样也更快): 归功于 http://www.nullskull.com/q/10057748/hi-all.aspx
的 Web Star static public ADODB.Recordset ConvertToRecordset(DataTable inTable)
{
ADODB.Recordset result = new ADODB.Recordset();
result.CursorLocation = ADODB.CursorLocationEnum.adUseClient;
ADODB.Fields resultFields = result.Fields;
System.Data.DataColumnCollection inColumns = inTable.Columns;
foreach (DataColumn inColumn in inColumns)
{
resultFields.Append(inColumn.ColumnName
, TranslateType(inColumn.DataType)
, inColumn.MaxLength
, inColumn.AllowDBNull ? ADODB.FieldAttributeEnum.adFldIsNullable :
ADODB.FieldAttributeEnum.adFldUnspecified
, null);
}
result.Open(System.Reflection.Missing.Value
, System.Reflection.Missing.Value
, ADODB.CursorTypeEnum.adOpenStatic
, ADODB.LockTypeEnum.adLockOptimistic, 0);
foreach (DataRow dr in inTable.Rows)
{
result.AddNew(System.Reflection.Missing.Value,
System.Reflection.Missing.Value);
for (int columnIndex = 0; columnIndex < inColumns.Count; columnIndex++)
{
resultFields[columnIndex].Value = dr[columnIndex];
}
}
return result;
}
static ADODB.DataTypeEnum TranslateType(Type columnType)
{
switch (columnType.UnderlyingSystemType.ToString())
{
case "System.Boolean":
return ADODB.DataTypeEnum.adBoolean;
case "System.Byte":
return ADODB.DataTypeEnum.adUnsignedTinyInt;
case "System.Char":
return ADODB.DataTypeEnum.adChar;
case "System.DateTime":
return ADODB.DataTypeEnum.adDate;
case "System.Decimal":
return ADODB.DataTypeEnum.adCurrency;
case "System.Double":
return ADODB.DataTypeEnum.adDouble;
case "System.Int16":
return ADODB.DataTypeEnum.adSmallInt;
case "System.Int32":
return ADODB.DataTypeEnum.adInteger;
case "System.Int64":
return ADODB.DataTypeEnum.adBigInt;
case "System.SByte":
return ADODB.DataTypeEnum.adTinyInt;
case "System.Single":
return ADODB.DataTypeEnum.adSingle;
case "System.UInt16":
return ADODB.DataTypeEnum.adUnsignedSmallInt;
case "System.UInt32":
return ADODB.DataTypeEnum.adUnsignedInt;
case "System.UInt64":
return ADODB.DataTypeEnum.adUnsignedBigInt;
case "System.String":
default:
return ADODB.DataTypeEnum.adVarChar;
}
}