還是先看看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);