dart 基础

变量声明

  • var 关键字

  • 强类型语言,声明后无法改变

          var z;
          z = "hello world";
          // 下面代码在dart中会报错,因为变量t的类型已经确定为String,
          // 类型一旦确定后则不能再更改其类型。
          z = 1000;
          
    
  • dynamic和Object

    • Object 是Dart所有对象的根基类,也就是说所有类型都是Object的子类(包括Function和Null),所以任何类型的数据都可以赋值给Object声明的对象. dynamic与var一样都是关键词,声明的变量可以赋值任意对象。
    • dynamic与Object相同之处:"在于,他们声明的变量可以在后期改变赋值类型。"
    • dynamic与Object不同的是:"dynamic声明的对象编译器会提供所有可能的组合, 而Object声明的对象只能使用Object的属性与方法, 否则编译器会报错"
      dynamic t; Object x; t = "hello world"; x = 'Hello Object'; //下面代码没有问题 t = 1000; x = 1000;
  • final和const

    • 不改变变量类型或值,初始化就固定
    • final 或 const,不是var,也不是一个类型。 一个 final 变量只能被设置一次
    • 两者区别在于:const 变量是一个编译时常量,final变量在第一次使用时被初始化。被final或者const修饰的变量,变量类型可以省略

//可以省略String这个类型声明 final str = "hi world"; //final String str = "hi world"; const str1 = "hi world"; //const String str1 = "hi world";

函数

  • Dart是一种真正的面向对象的语言,所以即使是函数也是对象,并且有一个类型Function。这意味着函数可以赋值给变量或作为参数传递给其他函数,这是函数式编程的典型特征。
  • 声明一个函数:

bool isNoble(int atomicNumber) { return _nobleGases[atomicNumber] != null; }

  • Dart函数声明如果没有显式声明返回值类型时会默认当做“dynamic处理”,注意,函数返回值没有类型推断!

  • 对于只包含一个表达式的函数,可以使用简写语法

bool isNoble (int atomicNumber)=> _nobleGases [ atomicNumber ] != null ;

  • 函数可以作为变量:

var say = (str){ print(str); }; say("hi world");

  • 函数作为参数传递:

void execute(var callback) { callback(); } execute(() => print("xxx"))

  • 可选的位置参数
    • 包装一组函数参数,用[]标记为可选的位置参数,并放在参数列表的最后面:

String say(String from, String msg, [String device]) { var result = '$from says $msg'; if (device != null) { result = '$result with a $device'; } return result; }

测试=》say('Bob', 'Howdy'); //结果是: Bob says Howdy

测试=> say('Bob', 'Howdy', 'smoke signal'); //结果是:Bob says Howdy with a smoke signal

  • 可选的命名参数(用的比较多)
    • 定义函数时,使用{param1, param2, …},放在参数列表的最后面,用于指定命名参数。例如:

//设置[bold]和[hidden]标志 void enableFlags({bool bold, bool hidden}) { // ... }

  • 调用函数时,可以使用指定命名参数。例如:paramName: value

enableFlags(bold: true, hidden: false);

注意,不能同时使用可选的位置参数和可选的命名参数

异步支持

  • Dart类库有非常多的返回Future或者Stream对象的函数。 这些函数被称为异步函数:它们只会在设置好一些耗时操作之后返回,比如像 IO操作。而不是等到这个操作完成。

  • async和await关键词支持了异步编程,允许您写出和同步代码很像的异步代码。

  • Future

    • Future与JavaScript中的Promise非常相似,表示一个异步操作的最终完成(或失败)及其结果值的表示。
    • 简单来说,它就是用于处理异步操作的,异步处理成功了就执行成功的操作,异步处理失败了就捕获错误或者停止后续操作。
    • 一个Future只会对应一个结果,要么成功,要么失败。
    • Future的所有API的返回值仍然是一个Future对象,所以可以很方便的进行链式调用。
  • Future API及特性 例子讲解

    • Future.then 成功返回
    • Future.catchError 捕捉错误
    • Future.whenComplete 请求的中途想干些事
    • Future.wait 等待所有的Future成功才会走这里
    • 因为只有成功和失败,捕捉到错误then方法不在执行

// 延迟2s返回结果:"hi world!" Future.delayed(new Duration(seconds: 2),(){ return "hi world!"; }).then((data){ // 执行成功走这 print(data); }).catchError((e){ //执行失败会走到这里
print(e); }).whenComplete((){ //无论成功或失败都会走到这里 }); // wait等待结果,4s后看到hello world Future.wait([ // 2秒后返回结果
Future.delayed(new Duration(seconds: 2), () { return "hello"; }), // 4秒后返回结果
Future.delayed(new Duration(seconds: 4), () { return " world"; }) ]).then((results){ print(results[0]+results[1]); }).catchError((e){ print(e); });

  • Async/await
    • 同步方法写异步代码,利用Dart的Future返回都是Future特性可以连续调用,也可以直接用Async/await形式

// 登陆之后保存用户信息 task() async { try{ String id = await login("alice","******"); String userInfo = await getUserInfo(id); await saveUserInfo(userInfo); //执行接下来的操作
} catch(e){ //错误处理
print(e);
}
}

Stream

  • 也是用于接收异步事件数据,和Future 不同的是,它可以接收多个异步操作的结果(成功或失败)。
  • 也就是说,在执行异步任务时,可以通过多次触发成功或失败事件来传递结果数据或错误异常。
  • Stream 常用于会多次读取数据的异步任务场景,如网络内容下载、文件读写等。
kkdemian
hyperliquid