0

我刚刚开始使用 Dapper.Contrib 来帮助我进行插入和获取,但是由于我的枚举作为字符串存储在数据库中(出于多种原因),我在插入时遇到了问题。Dapper 在读取时与字符串枚举无缝协作,但插入总是会将序数值放入数据库中。

我已经为此阅读了许多向 Dapper 提出的建议,并且打开了很多问题,但没有找到可行的解决方案。我的简化类如下所示:

public class Person {
  public long Id { get; set; }
  public string Name { get; set; }
  public Gender Gender { get; set; }
}

public enum Gender { Female, Male, TransWoman, TransMan }

我期待我可以配置 Dapper.Contrib 以使用枚举名称而不是序数值发出插入,这样下面的代码就会神奇地工作并在 varchar(20) 数据库字段性别中插入“男性”:

void InsertPersonFelipe(SqlConnection conn) {
  var person = new Person { Name = "Felipe", Gender = Gender.Male };
  conn.Insert(person);
}

有没有办法添加自定义映射typeof(Gender)

或者,更好的是,Dapper.Contrib 是否提供了一种配置以使其使用枚举名称而不是它们的序数值?

4

2 回答 2

0

我将 Dan 的答案改写为更现代的 C# 并且不尝试插入 ID(因为我有自动递增的标识列),以及采用表名而不是查看属性。

    public static long InsertB<T>(this SqlConnection sqlConnection, T obj, string tableName)
    {
        Dictionary<string, object> propertyValuesMap = new Dictionary<string, object>();
        var columnList = new List<String>();
        var valueList = new List<String>();
        var relevantProperties = obj.GetType().GetProperties().Where(x => !Attribute.IsDefined(x, typeof(ComputedAttribute))).ToList();

        foreach (var propertyInfo in relevantProperties)
        {
            if (propertyInfo.Name.ToLower() == "id") continue; // do not try to insert id
            
            var val = propertyInfo.PropertyType.IsEnum
                ? Enum.GetName(propertyInfo.PropertyType, propertyInfo.GetValue(obj))
                : propertyInfo.GetValue(obj);
            
            propertyValuesMap.Add(propertyInfo.Name, val);
            columnList.Add(propertyInfo.Name);
            valueList.Add($"@{propertyInfo.Name}");
        }

        return sqlConnection.Execute($"Insert Into {tableName} ({String.Join(", ", columnList)}) values ({String.Join(", ", valueList)})", propertyValuesMap);
    }
于 2021-06-28T08:18:03.503 回答
0

我编写了一个扩展方法来处理将枚举转换为字符串,并考虑了Dapper.Contrib 中的TableComputed属性。如果您不想引用 Dapper.Contrib,也可以轻松删除这些内容。

用法:

  using (var sql = new SqlConnection(_connString))
  {
      sql.Open();
      sql.InsertB(person);
  }

扩展方法:

    public static long InsertB<T>(this SqlConnection sqlConnection, T obj)
    {
        Dictionary<string, object> propertyValuesMap = new Dictionary<string, object>();

        var columns = new StringBuilder();
        var values = new StringBuilder();
        var tableName = ((TableAttribute)obj.GetType().GetCustomAttribute(typeof(TableAttribute))).Name;
        var relevantProperties = obj.GetType().GetProperties().Where(x => !Attribute.IsDefined(x, typeof(ComputedAttribute))).ToList();

        for (int i = 0; i < relevantProperties.Count(); i++)
        {
            object val = null;
            var propertyInfo = relevantProperties[i];
            if (propertyInfo.PropertyType.IsEnum)
            {
                val = Enum.GetName(propertyInfo.PropertyType, propertyInfo.GetValue(obj));
            }
            else
            {
                val = propertyInfo.GetValue(obj);
            }

            propertyValuesMap.Add(propertyInfo.Name, val);
            var propName = i == relevantProperties.Count() - 1 ? $"{propertyInfo.Name}" : $"{propertyInfo.Name},";
            columns.Append(propName);
            values.Append($"@{propName}");
        }

        return sqlConnection.Execute($"Insert Into {tableName} ({columns}) values ({values})", propertyValuesMap);
    }
于 2018-11-23T19:21:21.490 回答