【问题标题】:"The argument type 'String?' can't be assigned to the parameter type 'String'" when using stdin.readLineSync()“参数类型'字符串?'使用 stdin.readLineSync() 时无法将参数类型分配给“字符串”
【发布时间】:2021-06-16 02:30:37
【问题描述】:

我是飞镖新手。我不知道我犯了什么样的错误,但是这段代码不起作用。 是一个简单的代码,只需在终端中读取年龄并说它是未成年人或超过 18 岁。

import 'dart:io'; 

main(){
  print("Entre com a sua idade: ");
  var input = stdin.readLineSync();
  var idade = int.parse(input);

  if(idade >= 18){
    print("É maior de idade");
  }else{
    print("É menor de idade");
  }
}

我得到这个错误:

algoritmo01.dart:15:25:错误:参数类型“字符串?”不能分配给参数类型“字符串”,因为“字符串?”可以为空,而 'String' 不是。 var idade = int.parse(输入);

【问题讨论】:

标签: dart dart-null-safety null-safety


【解决方案1】:

欢迎来到飞镖。您所体验的是新的 null 安全功能(在 Dart 2.12.0 中引入),默认情况下变量不能包含 null 值。这是静态检查的,因此即使在程序执行之前您也会收到错误。

在您的示例中,问题在于以下行:

var input = stdin.readLineSync();

如果我们查看手册,我们可以看到此方法具有以下签名:

String? readLineSync (
    {Encoding encoding = systemEncoding,
    bool retainNewlines = false}
) 

https://api.dart.dev/stable/2.12.2/dart-io/Stdin/readLineSync.html

String? 表示它是可空类型,因此允许包含任何字符串或空值。如果类型改为String,则意味着结果只能是String,而不能是null

这意味着您的变量input 现在具有String? 类型。如果您查看示例的第二行,这(在 2.12.0 中引入)是一个问题:

var idade = int.parse(input);

由于int.parse的签名是:

int parse (

    String source,
    {int? radix,
    @deprecated int onError(
        String source
    )}

) 

https://api.dart.dev/stable/2.12.2/dart-core/int/parse.html

这需要String(因此永远不可能是null)。因此,您不能将String? 作为参数提供给仅处理String 的方法。相反的情况是允许的(如果方法采用String?,而你给它一个String)。

那么对于这种情况我们能做些什么呢?好吧,你需要处理 null 的情况,或者告诉 Dart 如果 inputnull,它应该让你的程序崩溃。

所以你可以这样处理:

import 'dart:io';

void main() {
  print("Entre com a sua idade: ");
  var input = stdin.readLineSync();
  
  if (input != null) {
    var idade = int.parse(input);

    if (idade >= 18) {
      print("É maior de idade");
    } else {
      print("É menor de idade");
    }
  } else {
    print('Input was null!');
  }
}

如您所见,即使input 在技术上仍属于String? 类型,这也是允许的。但是 Dart 可以看到,您的 if 语句将阻止 input 具有值 null,因此只要您在此 if 语句中,它将被“提升”为 String

另一个解决方案是告诉 Dart 它应该停止抱怨关于 input 的静态错误,并在编译代码时假设 inputString

import 'dart:io';

void main() {
  print("Entre com a sua idade: ");
  var input = stdin.readLineSync()!;
  var idade = int.parse(input);

  if (idade >= 18) {
    print("É maior de idade");
  } else {
    print("É menor de idade");
  }
}

(变化是readLineSync()后面加的!

在这种情况下,Dart 将添加自己的 null-check,如果 stdin.readLineSync 确实给出了 null 值,则程序会崩溃。这是为了确保我们永远不会给int.parse 一个空值,这可能会使您的代码在其他地方崩溃。通过添加null-check 在我们得到可疑值的地方,我们可以确保我们快速失败并且在事情与我们预期不同的地方失败。

您可以在此处阅读更多关于 Dart 中的 null-safety(也由 Stephen 链接):https://dart.dev/null-safety/understanding-null-safety

【讨论】:

  • 好详细的解释!
【解决方案2】:

接受的答案是正确的。但是还有另一种替代解决方案。您可以像这样使用“if null”运算符:

var idade = int.parse(input ?? "-1");

所以完整的代码看起来像这样:

import 'dart:io'; 

main(){
  print("Entre com a sua idade: ");
  var input = stdin.readLineSync();
  var idade = int.parse(input ?? "-1");

  if(idade >= 18){
    print("É maior de idade");
  }else if(idade < 18 && idade > 0){
    print("É menor de idade");
  } else {
    print("entrada errada");
  }
}

【讨论】:

    猜你喜欢
    • 2021-10-21
    • 2021-10-13
    • 2021-09-04
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多