189 8069 5689

flutter回调函数名,flutter flutter

flutter 同时执行多个异步请求回调

flutter中提供了Future.wait()函数,可以在执行多个异步请求之后有一个统一的回调结果,但是劣势在于,每一个异步函数的函数体中都需要执行return去返回结果,如果在异步请求中,有多层success或者fail这种函数的嵌套,那么可能会在某个地方忽略掉retrun,导致没有办法拿到正确的结果。所以这里封装了一个类似于js中Promise中的类去执行多个异步请求。

成都创新互联公司坚持“要么做到,要么别承诺”的工作理念,服务领域包括:成都网站设计、成都做网站、企业官网、英文网站、手机端网站、网站推广等服务,满足客户于互联网时代的陕西网站设计、移动媒体设计的需求,帮助企业找到有效的互联网解决方案。努力成为您成熟可靠的网络建设合作伙伴!

调用方式如下:

Flutter进阶篇(4)-- Flutter的Future异步详解

本文首发在公众号 Flutter那些事 ,欢迎大家多多关注。

工具安装:

Flutter基础篇:

Flutter进阶篇:

Dart语法基础篇:

Dart语法进阶篇:

说明:本文中的所有函数的引用在 main 函数中:

这里的执行结果是:

Futue直接new就可以了。我这里没有具体的返回数据,所以就用匿名函数代替了, Future future = new Future(() = null); 相当于 FutureNull future = new Future(() = null); 泛型如果为null可以省略不写,为了便于维护和管理,开发中建议加上泛型。

输出结果是:

future里面有几个函数:

then :异步操作逻辑在这里写。

whenComplete :异步完成时的回调。

catchError :捕获异常或者异步出错时的回调。

因为这里面的异步操作过程中没有遇到什么错误,所以catchError回调不会调用。

我们可以看到执行结果是:

我们可以看到输出结果是: 2 1 3 和我们创建Future对象的先后顺序完全一致。

我们可以看到结果为 1 2 3 ,和我们调用then的先后顺序无关。:

当then回调函数里面还有then回调的时候,这时候的流程跟前面就不太一样了,也是一个大坑,也是面试经常会被问到的一个知识点。

我们可以看到执行结果如下:

结果还是一样的:

运行结果是:

这里再次证明了上面我的猜想: 执行顺序和和创建Future的先后顺序有关,如果有多个then嵌套执行,先执行外面的then,然后执行里面的then。

执行结果如下,我们可以看到then内部创建的Future要等到then执行完了,最后再去执行的:

根据上文总结的特点,我们可以不用运行也能推断出输出结果:

为了验证我们的猜想,我们打印一下输出结果,果然我们的证明是正确的。

我们重点看看 then函数的文档说明:

then 注册在 Future 完成时调用的回调。

当这个 Future 用一个 value 完成时,将使用该值调用 onValue 回调。

如果 Future 已经完成,则不会立即调用回调,而是将在稍后的 microtask(微任务) 中调度。

如果回调返回 Future ,那么 then 返回的 future 将与 callback 返回的 future 结果相同。

onError 回调必须接受一个参数或两个参数,后者是[StackTrace]。

如果 onError 接受两个参数,则使用错误和堆栈跟踪时调用它,否则仅使用错误对象时候调用它。

onError 回调必须返回一个可用于完成返回的future的值或future,因此它必须是可赋值给 FutureOr R 的东西。

返回一个新的 Future ,该 Future 是通过调用 onValue (如果这个Future是通过一个value完成的)或' onError (如果这个Future是通过一个error完成的)的结果完成的。

如果调用的回调抛出异常,返回的 future 将使用抛出的错误和错误的堆栈跟踪完成。在 onError 的情况下,如果抛出的异常与 onError 的错误参数“相同(identical)”,则视为重新抛出,并使用原始堆栈跟踪替代

如果回调返回 Future ,则 then 返回的 Future 将以与回调返回的 Future 相同的结果完成。

如果未给出 onError ,并且后续程序走了刚出现了错误,则错误将直接转发给返回的 Future 。

在大多数情况下,单独使用 catchError 更可读,可能使用 test 参数,而不是在单个 then 调用中同时处理 value 和 error 。

请注意,在添加监听器(listener)之前, future 不会延迟报告错误。如果第一个 then 或 catchError 调用在 future 完成后发生 error ,那么 error 将报告为未处理的错误。

Flutter 组件回调

在网上找了很多没有关于这方面的例子,组件之间传值,可以通过构建函数,如果我要回调这个组件的值呢,给大家介绍一下这两个值 ValueChanged VoidCallback 自己也加深记忆一下

一ValueChanged

官方解释

这个值可以回调值,例如

二 VoidCallback 

这个值也可以回调但不能回调值,只能触发方法

hao((){

print("");

});

hao(VoidCallback voidCallback){

voidCallback();

}

错误方法下面会报错的

Flutter 之 动画1

对动画系统而言,为了实现动画,它需要做三件事儿:1.确定画面变化的规律;2.根据这个规律,设定动画周期,启动动画;3.定期获取当前动画的值,不断地微调、重绘画面。

这三件事情对应到 Flutter 中,就是 Animation、AnimationController 与 Listener:

1.Animation 是 Flutter 动画库中的核心类,会根据预定规则,在单位时间内持续输出动画的当前状态。Animation 知道当前动画的状态(比如,动画是否开始、停止、前进或者后退,以及动画的当前值),但却不知道这些状态究竟应用在哪个组件对象上。换句话说,Animation 仅仅是用来提供动画数据,而不负责动画的渲染。

2.AnimationController 用于管理 Animation,可以用来设置动画的时长、启动动画、暂停动画、反转动画等。

3.Listener 是 Animation 的回调函数,用来监听动画的进度变化,我们需要在这个回调函数中,根据动画的当前值重新渲染组件,实现动画的渲染。

class NormalAnimateWidget extends StatefulWidget {

@override

StatecreateState()=_NormalAnimateState();

}

class _NormalAnimateState extends Statewith SingleTickerProviderStateMixin{

AnimationController?controller;

Animation?animation;

@override

void initState() {

// TODO: implement initState

super.initState();

/*

* AnimationController

AnimationController用于控制动画,它包含动画的启动forward()、停止stop() 、反向播放 reverse()等方法。

* AnimationController会在动画的每一帧,就会生成一个新的值。

* 默认情况下,AnimationController在给定的时间段内线性的生成从 0.0 到1.0(默认区间)的数字。

* */

/*Ticker

当创建一个AnimationController时,需要传递一个vsync参数,

它接收一个TickerProvider类型的对象,它的主要职责是创建Ticker,定义如下:

abstract class TickerProvider {

//通过一个回调创建一个Ticker

Ticker createTicker(TickerCallback onTick);

}

Flutter 应用在启动时都会绑定一个SchedulerBinding,

通过SchedulerBinding可以给每一次屏幕刷新添加回调,

而Ticker就是通过SchedulerBinding来添加屏幕刷新回调,这样一来,

每次屏幕刷新都会调用TickerCallback。

使用Ticker(而不是Timer)来驱动动画会防止屏幕外动画(动画的UI不在当前屏幕时,如锁屏时)

消耗不必要的资源,因为Flutter中屏幕刷新时会通知到绑定的SchedulerBinding,

而Ticker是受SchedulerBinding驱动的,

由于锁屏后屏幕会停止刷新,所以Ticker就不会再触发。

*/

// 创建动画周期为1秒的AnimationController对象

controller =AnimationController(

vsync:this, duration:const Duration(milliseconds:3000));

/*

* Curve

* 动画过程可以是匀速的、匀加速的或者先加速后减速等。

* Flutter中通过Curve(曲线)来描述动画过程,

* 我们把匀速动画称为线性的(Curves.linear),而非匀速动画称为非线性的。

* 我们可以通过CurvedAnimation来指定动画的曲线,如:

final CurvedAnimation curve =

CurvedAnimation(parent: controller, curve: Curves.easeIn);

*

Curves曲线 动画过程

linear 匀速的

decelerate 匀减速

ease 开始加速,后面减速

easeIn 开始慢,后面快

easeOut  开始快,后面慢

easeInOut  开始慢,然后加速,最后再减速

*

* 当然我们也可以创建自己Curve,例如我们定义一个正弦曲线:

class ShakeCurve extends Curve {

@override

double transform(double t) {

return math.sin(t * math.PI * 2);

}

}

* */

final CurvedAnimation curve =CurvedAnimation(

parent:controller!, curve:Curves.linear);

/*

* Animation

*Animation是一个抽象类,它本身和UI渲染没有任何关系,

* 而它主要的功能是保存动画的插值和状态;其中一个比较常用的Animation类是Animation。

* Animation对象是一个在一段时间内依次生成一个区间(Tween)之间值的类。

* Animation对象在整个动画执行过程中输出的值可以是线性的、曲线的、一个步进函数或者任何其他曲线函数等等,

* 这由Curve来决定。 根据Animation对象的控制方式,

* 动画可以正向运行(从起始状态开始,到终止状态结束),

* 也可以反向运行,甚至可以在中间切换方向。

* Animation还可以生成除double之外的其他类型值

* ,如:Animation 或Animation。

* 在动画的每一帧中,我们可以通过Animation对象的value属性获取动画的当前状态值。

#动画通知

我们可以通过Animation来监听动画每一帧以及执行状态的变化,Animation有如下两个方法:

addListener();它可以用于给Animation添加帧监听器,

* 在每一帧都会被调用。

* 帧监听器中最常见的行为是改变状态后调用setState()来触发UI重建。

addStatusListener();

* 它可以给Animation添加“动画状态改变”监听器;

* 动画开始、结束、正向或反向(见AnimationStatus定义)时会调用状态改变的监听器。

* */

// 创建从50到200线性变化的Animation对象

// 普通动画需要手动监听动画状态,刷新UI

animation =Tween(begin:10.0, end:200.0).animate(curve)

..addListener(()=setState((){}));

/*

* Tween

* 默认情况下,AnimationController对象值的范围是[0.0,1.0]。

* 如果我们需要构建UI的动画值在不同的范围或不同的数据类型,

* 则可以使用Tween来添加映射以生成不同的范围或数据类型的值。

*Tween构造函数需要begin和end两个参数。

* Tween的唯一职责就是定义从输入范围到输出范围的映射。

* 输入范围通常为[0.0,1.0],但这不是必须的,我们可以自定义需要的范围。

* */

// 启动动画

controller!.repeat(reverse:true);

//

// 第二段

// animation!.addStatusListener((status) {

//  if (status == AnimationStatus.completed) {

//    controller!.reverse();// 动画结束时反向执行

//  } else if (status == AnimationStatus.dismissed) {

//    controller!.forward();// 动画反向执行完毕时,重新执行

//  }

// });

// controller!.forward();// 启动动画

}

@override

Widget build(BuildContext context) {

return MaterialApp(

home:Scaffold(

body:Center(

child:Container(

width:animation!.value,// 将动画的值赋给 widget 的宽高

              height:animation!.value,//

              child:FlutterLogo(),

)

)

)

);

}

@override

void dispose() {

// 释放资源

controller!.dispose();

super.dispose();

}

}

FlutterWeb 和 WebView 原生交互调用

需要创建两个工程,一个是FlutterWeb工程最终打包成Web页面,一个是Flutter原生工程承载一个WebView用来加载Web页面。这样做的好处在于只需要一种语言开发iOS和Android不用对接两次,可以直接使用社区Flutter原生工程的插件,只需要封装给Web调用。

FlutterWeb工程pubspec.yaml添加依赖

Flutter原生工程pubspec.yaml添加依赖

创建一个 toast_channel.dart,定义一个类实现 JavascriptChannel 重写name指定channel名称和onMessageReceived指定调用函数

在WebView的 javascriptChannels 配置上定义的Channel

创建一个 native_channel.dart ,定义一个外部函数通过 @JS("调用的channel和函数名") 注解指定调用的原生函数(JavascriptChannel固定名称为postMessage)

需要使用的地方直接调用

创建一个 js_function.dart,存放被原生调用的函数名称

将要提供给原生调用的函数,通过 js.context[原生调用名称] = 函数 开放给外部调用

如果在FlutterWeb工程要使用这个函数也可以使用@JS注解

WebView 创建时会回调 onWebViewCreated 获得 WebViewController ,WebViewController 调用 runJavascript 会执行JS函数无返回值,调用 runJavascriptReturningResult 会执行JS函数有返回值。

FutureBuilder获取WebViewController, 需要使用的地方直接调用

使用 HTML,CSS,Canvas 和 SVG 元素来渲染。

缺点:会存在不同平台效果不一样。

优点:不加载canvaskit默认使用系统字体,加载过程没有多余开销。

需要用到wasm,WebAssembly 要求需要浏览器支持,WebView Android需要最低需要57,Safari iOS 需要最低需要 11。

缺点:canvaskit 有7m大默认地址在国外首次加载耗时;中文会加载字体库默认地址在国外加载慢。

优点:性能更好,渲染效果一致。

--web-renderer=auto 默认移动端浏览器选择 HTML,桌面端浏览器选择 CanvasKit。

--web-renderer=html 使用 HTML 渲染器

--web-renderer=canvaskit 使用 CanvasKit 渲染器

综上所诉推荐移动端使用HTML渲染更合适,在编译和打包时指定渲染器 --web-renderer=html 。

--debug 模式构建的 Web 应用没有被压缩,且 Tree-shaking 没有执行。

--profile 模式构建的 Web 应用没有被压缩,但 Tree-shaking 执行了。

--release 模式构建的 Web 应用被压缩了,并且 Tree-shaking 执行了

运行命令

flutter run web --dart-define=FLUTTER_WEB_CANVASKIT_URL=./canvaskit/ --web-renderer=html

flutter run web --dart-define=FLUTTER_WEB_CANVASKIT_URL=./canvaskit/ --web-renderer=html --profile

打包命令

flutter build web --dart-define=FLUTTER_WEB_CANVASKIT_URL=./canvaskit/ --web-renderer=html --release


当前题目:flutter回调函数名,flutter flutter
URL地址:http://gzruizhi.cn/article/hoohec.html

其他资讯