【Dart】=> [06] Dart初体验-类Class-构造函数-继承-mixin-异步编程-链式调用-泛型-异常

本文主要是介绍【Dart】=> [06] Dart初体验-类Class-构造函数-继承-mixin-异步编程-链式调用-泛型-异常,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

    • 能够定义并使用Dart的类
    • 类的定义
    • 构造函数
    • 私有属性和方法
    • 继承
    • mixin
    • 异步编程
    • Future
    • Future链式调用
    • async - await
    • dynamic类型
    • 泛型
    • 异常

能够定义并使用Dart的类

Dart是一门面向对象的编程语言,所有的对象都是类的实例 通过类我们可以对数据和方法进行封装复用

学习内容:

  1. 类的定义
  2. 构造函数
  3. 私有属性和方法
  4. 继承
  5. mixin

类的定义

使用 class 关键字声明一个类,所有的类都是继承自 Object 类
类的组成:属性 和 方法
属性和方法都是通过 . 访问的
例子:定义一个Person类,属性是名字和年龄,方法是吃饭

在这里插入图片描述
在这里插入图片描述
整体代码

void main() {// 创建Person对象// Person person = Person();// // 给属性赋值// person.name = '张三';// person.age = 18;// 使用自定义类名构造函数创建对象// Person person = Person('李四', 19);// 使用命名构造函数创建对象// Person person = Person.withName('王五');// 使用工厂构造函数创建对象// Person person = Person.withInfo('赵六', 21);Person person = Person.withInfo('赵六', -21);print('我叫 ${person.name} 今年 ${person.age}');Person person1 = Person.withAge(20);print(person1.age);// 调用方法person.eat();
}// 例子:定义Person类,属性:名字和年龄,方法:吃饭
class Person {// 默认的构造函数(无参数,默认隐藏)// Person() {//   print('我是默认的构造函数');// }// 自定义类名构造函数:可以有参数// 注意点:与类同名的构造函数只能有一个,如果自定义了类名构造函数,那么默认的构造函数就失效// Person(String name, int age) {//   this.name = name;//   this.age = age;// }// 简写:自定义类名构造函数时,如果函数的参数和类的属性同名Person(this.name, this.age);// 定义命名构造函数// Person.withName(String name) {//   this.name = name;// }// 简写Person.withName(this.name);Person.withAge(this.age);// 定义工厂构造函数// factory Person.withInfo(String name, int age) {//   // 需要手动的创建对象并返回//   return Person(name, age);// }// 例子:如果age < 0,那么person对象的年龄默认设置为0factory Person.withInfo(String name, int age) {// 需要手动的创建对象并返回return age < 0 ? Person(name, 0) : Person(name, age);}// 属性String? name;int? age;// 方法void eat() {print('我是干饭人');}
}

构造函数

创建对象时调用的函数,常用的构造函数有:

  1. 类名构造函数
  2. 命名构造函数
  3. 工厂构造函数

类名构造函数,与类同名的构造函数

  1. 默认构造函数(无参数,默认隐藏)

在这里插入图片描述

  1. 自定义类名构造函数(可以有参数)
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

注意点:类名构造函数,只能有一个,如果自定义了类名构造函数,那么默认的类名构造函数就无效了

命名构造函数,可以为类提供多个不同的构造函数
定义方式:类名.函数名() {}

 // 简写:自定义类名构造函数时,如果函数的参数和类的属性同名Person(this.name, this.age);// 定义命名构造函数// Person.withName(String name) {//   this.name = name;// }// 简写Person.withName(this.name);Person.withAge(this.age);

工厂构造函数,不会直接创建对象,而是在构造函数内部通过代码来决定要创建的对象
定义方式: 使用 factory 关键字声明工厂构造函数

  // 定义工厂构造函数// factory Person.withInfo(String name, int age) {//   // 需要手动的创建对象并返回//   return Person(name, age);// }

例子:如果创建Person对象时,age < 0,则该Person对象的年龄默认设置为0

  // 例子:如果age < 0,那么person对象的年龄默认设置为0factory Person.withInfo(String name, int age) {// 需要手动的创建对象并返回return age < 0 ? Person(name, 0) : Person(name, age);}

使用场景:当需要根据条件来决定要返回的对象时,比如:单例

私有属性和方法

  1. 私有属性和方法:
    在类中定义,不对外暴露,不能被其他Dart文件访问的属性和方法
  2. 如何定义私有属性和方法:
    使用 _ 定义属性和方法
// 导入Dart文件、库
import '28_类_私有属性和方法.dart';void main() {// 创建Dog对象Dog dog = Dog();dog.name = '旺财';print(dog.name);// 私有属性调用失败// dog._age;dog.eat();// 私有方法调用失败// dog._run();
}

继承

通过继承可以让子类拥有父类的一些属性和方法。如何实现继承:

  • 如何实现继承:
    • 子类使用 extends 关键字继承父类
  • 继承的特点:
    • Dart的继承是单继承,一个子类只能有一个父类
    • 子类只会继承父类里面可见的属性和方法,不会继承私有属性和方法
    • 子类只会继承父类默认的构造函数,不会继承其他构造函数
    • 子类可以重写父类的方法,也可以使用 super 调用父类方法
void main() {// 创建猫// Cat cat = Cat();// cat.name = 'Tom';Cat cat = Cat('Tom');print(cat.name);cat.eat();cat.walk();// 创建鱼// Fish fish = Fish();// fish.name = '鲨鱼';Fish fish = Fish('鲨鱼');print(fish.name);fish.eat();fish.swim();
}// 猫,吃饭和走路
class Cat extends Animal {Cat(String name) : super(name);// String? name;// void eat() {//   print('eat');// }// 重写父类的方法void eat() {// 执行子类自己的逻辑print('执行子类自己的逻辑');// 使用super去调用父类的方法// super.eat();}void walk() {print('walk');}
}// 鱼,吃饭和游泳
class Fish extends Animal {// 定义子类自己的构造函数,并且使用super调用父类的构造函数传递数据Fish(String name) : super(name);// String? name;// void eat() {//   print('eat');// }void swim() {print('swim');}
}// 定义一个父类,父类里面有其他类都有的属性和方法
// 使用继承,让子类继承父类,从而子类就自动拥有了父类的属性和方法
class Animal {// 自定义类名构造函数// 如果父类自定义构造函数,那么子类继承不到,所以子类需要自己定义构造函数Animal(this.name);String? name;void eat() {print('eat');}
}// class Animal1 {
//   String? age;//   void sleep() {
//     print('eat');
//   }
// }

mixin

mixin 可以理解为扩展类,可以为类扩展功能,而不需要使用继承,类似Vue里面的混入。

如何定义并使用 mixin

  • 定义:mixin 关键字
  • 使用:with 关键字

mixin 的特点:

  • 可以扩展属性和方法
  • 不能被实例化,不能被继承
void main() {Person person = Person();// name是继承的person.name = '张三';print(person.name);// height是通过mixin扩展的person.height = 180.0;print(person.height);// eat()是继承的person.eat();// walk()是通过mixin扩展的person.walk();person.study();// mixin不能被实例化,不能被继承// WalkMixin();
}// mixin可以使用多个
mixin StudyMixin {void study() {print('study');}
}// 定义mixin
mixin WalkMixin {double? height;void walk() {print('walk');}
}// 人,吃饭和走路
class Person extends Animal with WalkMixin, StudyMixin {// String? name;// void eat() {//   print('eat');// }// void walk() {//   print('walk');// }
}// 猫,吃饭和走路
class Cat extends Animal with WalkMixin {// void walk() {//   print('walk');// }
}// 鱼,吃饭和游泳
class Fish extends Animal {void swim() {print('swim');}
}// 定义基类:动物
class Animal {String? name;void eat() {print('eat');}// void walk() {//   print('walk');// }
}

异步编程

能够使用Dart异步编程解决耗时操作阻塞程序的问题

在Web端和APP中,有很多耗时操作都需要 异步执行
Web端的异步解决方案是 Promise,再配合 async – await 更能以同步的方式编写异步代码
Dart也同样提供了异步解决方案 Future ,也可以配合 async – await 使用

学习内容:
1.Future
2.async - await

Future

  • Future是一个表示延迟计算的对象。代表一些计算将异步进行
  • Future会在耗时操作执行完毕前直接返回,而不会等待耗时操作执行结束
  • 例子:模拟耗时操作阻塞程序,并使用Future解决程序阻塞问题
// 模拟耗时操作阻塞程序的问题,并使用Future解决阻塞问题print('开始喽');Future(() {// 耗时任务执行的地方sleep(Duration(seconds: 5));// 故意编写的异常代码(测试)// dynamic str = 'hehe';// str.haha();// 返回异步任务执行的结果return '假装这是异步任务执行的结果';}).then((value) {// 监听异步任务执行结束print(value);}).catchError((e) {// 捕获异常信息print(e);});

Future链式调用

例子:用户先登录并获取用户信息,再保存用户信息

// Future链式调用// 用户先登录拿到用户信息,然后再保存用户信息Future login(String name, String password) {return Future(() {sleep(Duration(seconds: 2));print('登录操作');return 'userInfo';});}Future saveUserInfo(String userInfo) {return Future(() {sleep(Duration(seconds: 2));print('保存用户信息');return 'OK';});}// Future链式调用login('张三', '123456').then((value) {saveUserInfo(value);});print('假装这是个不能被阻塞的代码');

async - await

  1. Future 配合 async – await 以同步的方式编写异步代码
  2. 例子:用户先登录并获取用户信息,再保存用户信息
 // Future链式调用// 用户先登录拿到用户信息,然后再保存用户信息Future login(String name, String password) {return Future(() {sleep(Duration(seconds: 2));print('登录操作');return 'userInfo';});}Future saveUserInfo(String userInfo) {return Future(() {sleep(Duration(seconds: 2));print('保存用户信息');return 'OK';});}// Future链式调用// login('张三', '123456').then((value) {//   saveUserInfo(value);// });// async awaitvoid doLogin() async {String userInfo = await login('李四', '123456');await saveUserInfo(userInfo);}doLogin();print('假装这是个不能被阻塞的代码');

dynamic类型

能够知道dynamic类型的特点

在Dart中,虽然有类型推断和类型检查
但是,还可以使用 dynamic 关键字关闭变量的类型检查

特点:
编码灵活,可以保存任意类型的数据
容易产生 NoSuchMethodError 的异常

void main() {int a = 15;// a = 'itcast';a = 20;// dynamic会关闭编译器的类型检查dynamic b = 100;b = 'itheima';print(b);// b.haha();// b.hehe;
}

泛型

能够知道泛型的作用

泛型是指类型的不确定性,数据具体的类型可以在使用时确定

  1. 使用泛型可以限定类型
  // 泛型限定数据的类型:List Map// 保存视频分类名称时,不应该出现100 true这样类型的数据// List categories = ['居家', '美食', 100, true];List<String> categories = ['居家', '美食'];
使用泛型可以明确约束列表元素的类型使用泛型可以明确约束字典的key和value的类型
 Map<String, String> category = {'id': '1','name': '居家',};
  1. 使用泛型可以减少重复代码
  // 封装方法:接收什么类型的数据,就返回什么类型T demo<T>(T parm) {return parm;}

异常

能够知道如何捕获并处理异常

  1. 如何捕获异常:
    使用关键字 try catch 捕获并处理异常
    finally:无论是否有异常都会执行到的语句块
  // 捕获异常:try catch// try {//   dynamic name = 'zzm';//   name.haha();// } catch (e) {//   print(e);// } finally {//   // 无论是否有异常都会执行这个代码块//   print('finally');// }
  1. 如何手动抛出异常:使用关键字 throw 手动抛出异常
// 手动抛出异常:判断字符串是否相等,如果不相等手动抛出异常try {String str = 'zzm';if (str == 'zxc') {print('ok');} else {// 手动抛出异常throw '字符串不相等';}} catch (e) {print(e);}

这篇关于【Dart】=> [06] Dart初体验-类Class-构造函数-继承-mixin-异步编程-链式调用-泛型-异常的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/609743

相关文章

Python调用Orator ORM进行数据库操作

《Python调用OratorORM进行数据库操作》OratorORM是一个功能丰富且灵活的PythonORM库,旨在简化数据库操作,它支持多种数据库并提供了简洁且直观的API,下面我们就... 目录Orator ORM 主要特点安装使用示例总结Orator ORM 是一个功能丰富且灵活的 python O

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

Deepseek R1模型本地化部署+API接口调用详细教程(释放AI生产力)

《DeepseekR1模型本地化部署+API接口调用详细教程(释放AI生产力)》本文介绍了本地部署DeepSeekR1模型和通过API调用将其集成到VSCode中的过程,作者详细步骤展示了如何下载和... 目录前言一、deepseek R1模型与chatGPT o1系列模型对比二、本地部署步骤1.安装oll

一分钟带你上手Python调用DeepSeek的API

《一分钟带你上手Python调用DeepSeek的API》最近DeepSeek非常火,作为一枚对前言技术非常关注的程序员来说,自然都想对接DeepSeek的API来体验一把,下面小编就来为大家介绍一下... 目录前言免费体验API-Key申请首次调用API基本概念最小单元推理模型智能体自定义界面总结前言最

JAVA调用Deepseek的api完成基本对话简单代码示例

《JAVA调用Deepseek的api完成基本对话简单代码示例》:本文主要介绍JAVA调用Deepseek的api完成基本对话的相关资料,文中详细讲解了如何获取DeepSeekAPI密钥、添加H... 获取API密钥首先,从DeepSeek平台获取API密钥,用于身份验证。添加HTTP客户端依赖使用Jav

异步线程traceId如何实现传递

《异步线程traceId如何实现传递》文章介绍了如何在异步请求中传递traceId,通过重写ThreadPoolTaskExecutor的方法和实现TaskDecorator接口来增强线程池,确保异步... 目录前言重写ThreadPoolTaskExecutor中方法线程池增强总结前言在日常问题排查中,

Go语言利用泛型封装常见的Map操作

《Go语言利用泛型封装常见的Map操作》Go语言在1.18版本中引入了泛型,这是Go语言发展的一个重要里程碑,它极大地增强了语言的表达能力和灵活性,本文将通过泛型实现封装常见的Map操作,感... 目录什么是泛型泛型解决了什么问题Go泛型基于泛型的常见Map操作代码合集总结什么是泛型泛型是一种编程范式,允

redis防止短信恶意调用的实现

《redis防止短信恶意调用的实现》本文主要介绍了在场景登录或注册接口中使用短信验证码时遇到的恶意调用问题,并通过使用Redis分布式锁来解决,具有一定的参考价值,感兴趣的可以了解一下... 目录1.场景2.排查3.解决方案3.1 Redis锁实现3.2 方法调用1.场景登录或注册接口中,使用短信验证码场

C#多线程编程中导致死锁的常见陷阱和避免方法

《C#多线程编程中导致死锁的常见陷阱和避免方法》在C#多线程编程中,死锁(Deadlock)是一种常见的、令人头疼的错误,死锁通常发生在多个线程试图获取多个资源的锁时,导致相互等待对方释放资源,最终形... 目录引言1. 什么是死锁?死锁的典型条件:2. 导致死锁的常见原因2.1 锁的顺序问题错误示例:不同

使用C/C++调用libcurl调试消息的方式

《使用C/C++调用libcurl调试消息的方式》在使用C/C++调用libcurl进行HTTP请求时,有时我们需要查看请求的/应答消息的内容(包括请求头和请求体)以方便调试,libcurl提供了多种... 目录1. libcurl 调试工具简介2. 输出请求消息使用 CURLOPT_VERBOSE使用 C