gpt4 book ai didi

C# 和反射不能连续工作两次

转载 作者:行者123 更新时间:2023-11-30 23:29:44 26 4
gpt4 key购买 nike

我有一个问题,但我不知道如何解决。

我有一个从数据库中获取数据的类,在这个类中我有一个简单的select *方法,这个方法叫做

List<T> All<T>(string tableName)

并且您必须指定要获取的资源,例如

All<User>("users")

而且,抛开经典的SQL Reader和SQL Command,方法的核心是这个

public override List<T> All<T>(string resource) 
{
List<T> result = new List<T>();
using (MySqlConnection sqlConnection = new MySqlConnection(connectionString))
{
sqlConnection.Open();
try
{
string query = "SELECT * FROM " + resource + " WHERE 1=1";
using (MySqlCommand sqlCommand = new MySqlCommand(query, sqlConnection))
{
lock (locker)
{
MySqlDataReader reader = sqlCommand.ExecuteReader();
if (reader.HasRows)
{
while (reader.Read())
{
T model = Activator.CreateInstance<T>();
Dictionary<string, object> _properties = new Dictionary<string, object>();
for (int i = 0; i < reader.FieldCount; i++)
{
string property = reader.GetName(i);
object value = reader.GetValue(i);
_properties.Add(property, value);
}
var type = model.GetType();
var method = type.GetMethod("SetProperties");
var invoked = method.Invoke(model, new object[] { _properties });

result.Add(model);
}
}
reader.Close();
}
}
}
catch (Exception ex)
{
Program.eventLogger.Add(new Event(EventType.Error, "SQL Data Providers", "Exception catched on All", ex));
}
finally
{
sqlConnection.Close();
}
}
return result;
}

基本上,基于方法头中的类型,该方法将尝试创建特定类型的新实例,稍后对于查询中的每个字段,它将在临时列表中填充该类的所有属性。一旦完成,它将尝试调用方法“SetProperties”,该方法基本上使用反射设置类的每个属性。

这是 SetProperties 的核心,对每个实体都是相等的:

    public virtual bool SetProperties(Dictionary<string,object> properties)
{
if(this.ValidateData(properties))
{
FillNullableAttributes(properties);
// Iterate trough every key : value pairs in properties
foreach (KeyValuePair<string, object> kvp in properties)
{
if (this.data.Contains(kvp.Key))
{
var property = this.GetType().GetProperty(kvp.Key);
PropertyInfo propertyInfo = this.GetType().GetProperty(kvp.Key);
// Set the current fetched key with the given value if !null
if (kvp.Value != null)
{
Type fetchedType = Nullable.GetUnderlyingType(property.PropertyType) ?? property.PropertyType;
object safeConversion = (kvp.Value == null || kvp.Value == DBNull.Value) ? null : Convert.ChangeType(kvp.Value, fetchedType);
if (propertyInfo.CanWrite)
{
propertyInfo.SetValue(this, safeConversion, null);
}
}
}
}
return true;
}
return false;
}

总而言之,结果是一个列表,将被返回,具体的实体将有自己的 BindingList 填充。每个实体的绑定(bind)列表描述如下:

public static BindingList<Seller> items = new BindingList<Seller>();

这段代码工作正常,即使我知道有很大的改进空间,但如果我像这样调用它两次:

User.items = new BindingList<User>(provider.All<User>("users"));
User.items = new BindingList<User>(provider.All<User>("users"));

第二个列表将由空实体填充,计数将是正确的,但它们将是空的......这不应该发生。

我从调试中唯一弄清楚的是在第二次调用时

var invoked = method.Invoke(model, new object[] { _properties });

invoked 设置为 false。

最佳答案

结果:

var invoked = method.Invoke(model, new object[] { _properties });

是您的 SetProperties 方法的返回值,不是该方法是否如您的问题所示被调用。您的 SetProperties 方法告诉您它无法完成其工作,对其进行调试,您会找到答案。

关于C# 和反射不能连续工作两次,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35271660/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com