1- Подключитесь к Oracle XE с помощью Oracle.DataAccess.
2- Выполните команду для добавления столбца в таблицу: alter table TABLE add COLUMN b int;
3- Выполните команду, чтобы выбрать этот столбец
4- Чтение с помощью DataReader. Приложение вызывает IndexOutOfRangeException: невозможно найти указанный столбец в наборе результатов
5- Перезапустите приложение, и запрос будет выполнен правильно.
Почему DataReader не может получить доступ к столбцу, который я только что создал?
Вот большой, но простой код для тестирования:
private void button1_Click(object sender, EventArgs e)
{
using (OracleConnection con = new OracleConnection(Settings.Default.CS))
{
con.Open();
try
{
using (OracleCommand com = new OracleCommand())
{
com.Connection = con;
// Create a test table
com.CommandText = "CREATE TABLE Test (a int)";
com.ExecuteNonQuery();
// Add one column
com.CommandText = "ALTER TABLE Test ADD b int";
com.ExecuteNonQuery();
com.CommandText = "SELECT * FROM Test";
using (DbDataReader dr = com.ExecuteReader())
{
MessageBox.Show(dr.FieldCount.ToString());
// Here is showing "2", thats ok
}
}
}
finally
{
con.Close();
}
}
}
private void button2_Click(object sender, EventArgs e)
{
using (OracleConnection con = new OracleConnection(Settings.Default.CS))
{
con.Open();
try
{
using (OracleCommand com = new OracleCommand())
{
OracleTransaction trans = con.BeginTransaction();
try
{
// Add a column to table already created
com.Connection = con;
com.CommandText = "ALTER TABLE Test ADD c int";
com.ExecuteNonQuery();
// Insert a value, ok
com.CommandText = "INSERT INTO TEST (a, b, c) VALUES (1, 2, 3)";
com.ExecuteNonQuery();
trans.Commit();
}
catch
{
trans.Rollback();
throw;
}
// Selecting only "c" column
com.CommandText = "SELECT c FROM Test";
using (DbDataReader dr = com.ExecuteReader())
{
if (dr.Read())
MessageBox.Show(Convert.ToInt32(dr["c"]).ToString());
// Showing correct value, ok
}
// Uncomment these lines to solve problem
//con.Close();
//OracleConnection.ClearAllPools();
//con.Open();
// Selecting all fields * from table
com.CommandText = "SELECT * FROM Test";
using (DbDataReader dr = com.ExecuteReader())
{
MessageBox.Show(dr.GetSchemaTable().Rows.Count.ToString() + " / " + dr.FieldCount.ToString());
// HERE IS THE PROBLEM: message are showing 2/2, but table haves 3 fields
if (dr.Read())
MessageBox.Show(Convert.ToInt32(dr["c"]).ToString());
// Here throws IndexOutOfRangeException: Unable to find specified column in result set
}
}
}
finally
{
con.Close();
}
}
}
OracleConnection.PurgeStatementCache()
? - person Randy supports Monica   schedule 18.04.2012OracleTransaction
должен быть удален и поэтому должен иметь свой собственный блокusing
. Вы можете использоватьOracleCommand.CreateCommand
, так как это избавит вас от необходимости назначьте соединение команде самостоятельно. - person WhiteKnight   schedule 19.04.2012