还是先看看C#中属性的这定义,在初始化和访问上有哪些方式,就能看出required属性有什么不一样的地方了。
属性,是封装字段的,通过get和set访问器可以很好地验证数据的有效性。
public record Order_00
{public Guid Id { get; set; }private string? _orderNo;public string? OrderNo{get => _orderNo;set{if (!string.IsNullOrEmpty(value) && value.Length >= 8){_orderNo = value;}else{throw new ApplicationException("OrderNo is error");}}}public DateTime OrderDate { get; set; }
}
属性差异访问修饰符,如下面代码,当把Id的set定义成private后,只能在Order_01内访问到,构造也好,内部方法也好,默认值也好,都可以赋值操作,更可以验证。
public record Order_01
{public Order_01(){Id = Guid.NewGuid();}public void SetID(Guid id){Id = id;}public Guid Id { get; private set; } = Guid.NewGuid();public string? OrderNo { get; set; }public DateTime OrderDate { get; set; }
}
属性初始化,通过把set换成init,该属性就变成了只能在类型初始化时赋值,如下代码,不能在内部方法中赋值,更不能在外部赋值。
public record Order_02
{public Order_02(){Id = Guid.NewGuid();}//public void SetID(Guid id)//{// Id = id;//}public Guid Id { get; init; } = Guid.NewGuid();public string? OrderNo { get; set; }public DateTime OrderDate { get; set; }
}
必填属性,当给属性添加required时,最大的区别在于,实例化对象时,需要给这两个属性赋值,之前如果遇到这种需求,通常是在构造函数中,定义两个参数,以达到实例化完后,就有初始值。
var order_0N_b = new Order_0N { Id = Guid.NewGuid(),OrderNo="N2022098813594" };
Console.WriteLine(order_0N_b);public record Order_0N
{public Order_0N(){Id = Guid.NewGuid();}//因为Id是init,所以不能在内部方法赋值//public void SetID(Guid id)//{// Id = id;//}public void SetOrderNo(string no){OrderNo = no;}public required Guid Id { get; init; } = Guid.NewGuid();public required string? OrderNo { get; set; }public DateTime OrderDate { get; set; }
}
如果在构造中传必填值 ,在反射时,可以在调用构造方法时,给Invoke传参数,但用required时,该怎么传值呢?肯定不是构造,其实,当普通属性一样赋值就可以,代码如下:
var orderType = typeof(Order_0N);
var orderObj = orderType.GetConstructor(new Type[0])?.Invoke(null);
var idPro = orderType.GetProperty("Id");
idPro?.SetValue(orderObj, Guid.NewGuid());
var noPro = orderType.GetProperty("OrderNo");
noPro?.SetValue(orderObj, "N12345678");
Console.WriteLine(orderObj);