- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我刚刚在我的 .net core 3.1 项目中启用了空检查。
问题是我有一个响应类
public class DecryptResponse
{
public DecryptStatus Status { get; set; }
//This is the attribute in question
[NotNullWhen(Status==DecryptStatus.Ok)]
public Stream? Stream { get; set; }
public string? ErrorMessage { get; set; }
}
public enum DecryptStatus
{
Ok,
InvalidData,
KeyChecksumFailure,
NoData,
UnhandledError
}
Verify
方法不允许空值。
DecryptStatus==Ok
if (decryptResponse.Status != DecryptStatus.Ok)
return (decryptResponse, null);
var verifyResponse = Verify(customerId, decryptResponse.Stream);
return (decryptResponse, verifyResponse);
最佳答案
.NET 5+ 的简短回答:使用新的 MemberNotNull
和 MemberNotNullWhen
属性。
这些是在 .NET 5.0 和 C# 9.0 中引入的。 (你也可以使用 init
属性)。
问题在于 MemberNotNullWhen
仅适用于 Boolean
-typed 属性,但您可以添加一个新属性,该属性根据其他一些条件返回 true/false - 在您的情况下,这意味着如下所示:
public class DecryptResponse
{
public DecryptStatus Status { get; init; }
private Boolean StatusIsOK => this.Status == DecryptStatus.Ok;
[MemberNotNullWhen(true, nameof(StatusIsOK))]
public Stream? Stream { get; init; }
[MemberNotNullWhen(false, nameof(StatusIsOK))]
public string? ErrorMessage { get; init; }
}
当然,这种方法有一个巨大的漏洞:编译器无法验证
Status
,
Stream
, 和
ErrorMessage
设置正确。没有什么可以阻止您的程序执行
return new DecryptResponse();
无需设置任何属性。这意味着对象处于无效状态。
get; set;
的可变 DTO在每个属性上,因为如果没有主构造函数,就无法保证对象实例将被正确初始化。
MemberNotNullWhen
的情况下为较旧的 .NET 平台编写代码,然后我会有这个:
public abstract class DecryptResponse
{
public static implicit DecryptResponse( Stream okStream )
{
return new DecryptResponse.OK( okStream );
}
public static implicit DecryptResponse( DecryptStatus status, String errorMessage )
{
return new DecryptResponse.Failed( status, errorMessage );
}
private DecryptResponse( DecryptStatus status )
{
this.Status = status;
}
public DecryptStatus Status { get; }
public sealed class OK : DecryptResponse
{
public OK( Stream stream )
: base( DecryptStatus.OK )
{
this.Stream = stream ?? throw new ArgumentNullException(nameof(stream));
}
public Stream Stream { get; }
}
public sealed class Failed : DecryptResponse
{
public Failed ( DecryptStatus status, String errorMessage )
: base( status )
{
if( status == DecryptStatus.OK ) throw new ArgumentException( message: "Value cannot be " + nameof(DecryptStatus.OK) + "." );
this.ErrorMessage = errorMessage ?? throw new ArgumentNullException(nameof(errorMessage));
}
public String ErrorMessage { get; }
}
}
(从 CS 理论的角度来看,
the above class is a union type)。
Stream
和 ErrorMessage
)。 abstract
类型有一个 private
构造函数)并且它的两个子类型都是 sealed
,使得除了 OK
之外不可能有结果或 Failed
.enum
-类。而 C# 是 enum
更像是一个命名常量,编译器和语言不保证 C# enum
value 在运行时有效(例如,即使 MyEnum v = (MyEnum)123
不是定义的值,您也可以始终执行 123
)。 OK
中的验证逻辑和 Failed
构造函数保证 DecryptStatus.OK
始终表示结果类型为 DecryptResponse.OK
与非 null
Stream
属性(property)。同样,如果 Status != DecryptStatus.OK
你有一个 DecryptResponse.Failed
对象。 implicit
运算符定义意味着返回 DecryptResponse
的方法可以直接回一个Stream
或 ValueTuple<DecryptStatus,String>
并且 C# 编译器会自动为您执行转换。 public DecryptResponse DecryptSomething()
{
Stream someStream = ... // do stuff
if( itWorked )
{
return someStream; // Returning a `Stream` invokes the DecryptResponse conversion operator method.
}
else
{
DecryptStatus errorStatus = ...
return ( errorStatus, "someErrorMessage" ); // ditto for `ValueTuple<DecryptStatus,String>`
}
}
或者如果你想明确:
public DecryptResponse DecryptSomething()
{
Stream someStream = ... // do stuff
if( itWorked )
{
return new DecryptResponse.OK( someStream );
}
else
{
DecryptStatus errorStatus = ...
return new DecryptResponse.Failed( errorStatus, "someErrorMessage" );
}
}
并像这样消耗:
DecryptResponse response = DecryptSomething();
if( response is DecryptResponse.OK ok )
{
using( ok.Stream )
{
// do stuff
}
}
else if( response is DecryptResponse.Failed fail )
{
Console.WriteLine( fail.ErrorMessage );
}
else throw new InvalidOperationException("This will never happen.");
(不幸的是,C# 编译器还不够智能,无法识别封闭类型的层次结构,因此需要
else throw new...
语句,
but hopefully eventually that won't be necessary )。
关于C# 可空 : Make nullchecking dependent on another property/variable,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/62065860/
for /f "tokens=*" %%a in ('find /v ":" "%appdata%\gamelauncher\options.txt" ^| find "menu=a"') do (
我在 Javascript 中有一组全局计数器变量: var counter_0 = 0; var counter_1 = 0; var counter_2 = 0; 等等 然后我有一个 Javasc
好的,我正在阅读一些有关 RedBlackTrees 的代码。我注意到这一行“v1 = v2 = v3 = v4;”我理解类似“v1 += v2”(将 v2 添加到 v1 的当前值)和“v1 = v2
我正在为 C# 中的游戏数据加载制作一个 csv 阅读器,我想做的就是从数组(变量)的值声明一个变量,我们可以在 php 中像 $$foo 那样做。喜欢 void csvReader(string s
假设我有变量 内容为“ 123 ”和变量 b123 里面有一些文字。出于某种原因,我想使用变量 作为第二个 var 名称的一部分。像这样的东西: SET a=123 SET b123=some_tex
我对 javascript 有点陌生,我无法通过谷歌搜索找到任何内容,我正在编写一个程序,并且能够执行我所要求的操作: if (Variable == 1 或 Variable == 2 或 Vari
我发现我自己在做这种类型的 IF 语句分配。例如: if($variable == 1 || $variable == "whatever" || $variable == '492') { ...
我的虚拟 PC 在 MS-DOS 6.22 上运行时出现问题。 我需要使用变量 Date ,但我无法得到它,因为每当我尝试回显变量时,它都会显示 %variable%反而。 我在 Windows 控制
尝试运行此代码时: List list = em.createQuery("select balance b from Users where b.userName = '" + user_name.
我有一些代码,其中变量可以是 undefined、null 或正常值。无论变量是 undefined 还是 null,代码都需要做同样的事情。说有没有危险 for (var cur = this.bu
我正在编写一个批处理命令脚本,其中检查环境变量。我需要通过传递所有必需的变量来编写一个 FOR 循环,然后验证它是否已定义,如果未定义,则提示该键的值并永久设置该变量。 问题是我无法取消引用循环变量并
我知道这些是 Rails 的基础知识,但我仍然不知道 = 符号和 => 之间的全部区别以及 @some_variable 之间的区别、@@some_variable 和 :some_variable
我正在使用以下内容创建一个动态变量(PHP 术语中的“变量变量”): foo: "test1" set to-word (rejoin [foo "_result_data"]) array 5 但是
我一直在啃 PHP 套接字服务器和客户端的基础知识 here . 然后我偶然发现了这些行(摘自上面链接的第一个示例,发生在 while 中): if (false === ($buf = socket
这个问题在这里已经有了答案: What does "|=" mean? (pipe equal operator) (6 个答案) 关闭 9 年前。 我正在寻找一些编码来扩展我在 Java 方面的知
如何在 C++ 中从其他变量的值打印变量我只是 C++ 的新手。 在 php 中,我们可以通过其他变量的值来制作/打印一个变量。像这样。 $example = 'foo'; $foo = 'abc';
作为 Ruby on Rails 新手,我明白“@”和“:”引用有不同的含义。我看到了this post在 SO 中,其中描述了一些差异。 @ 表示实例变量(例如@my_selection) :表示别
编程新手/甚至更新。一个小的 go 程序有问题 - 不会编译带有 undefined variable 错误。代码: package main import ( "fmt" "io" "o
我知道其他一些语言,如PHP,支持“变量变量名”的概念--即,字符串的内容可以用作变量名的一部分。。我听说总的来说这不是一个好主意,但我认为它可以解决我在Python代码中遇到的一些问题。。有没有可能
我有两个版本的代码。 版本 1 Launcher.java class Launcher { public static void main(String[] args) {
我是一名优秀的程序员,十分优秀!