StatelessWidget 与在性能方面返回 Widget 的函数

Posted

技术标签:

【中文标题】StatelessWidget 与在性能方面返回 Widget 的函数【英文标题】:StatelessWidget vs a function returning Widgets in terms of performance 【发布时间】:2019-07-16 10:06:06 【问题描述】:

在性能方面,使用StatelessWidgetfunction returning a Widget 有什么区别吗?

至少我很清楚flutter's repo issue 中指出的与性能无关的差异。

事实上,我有一些同事声称 functional widgets 在性能方面最差,但在阅读了一些关于该主题的内容后,我找不到任何可以证明该断言的结论性文档,因此任何类型的非常欢迎您对此事进行澄清!

据我所知,它们之间的唯一区别在于使用const Widget,这似乎可以避免重建阶段。

【问题讨论】:

***.com/questions/53234825/…、***.com/questions/51689435/… 的副本 我的问题集中在性能而不是用例上。 在这些答案中提到,性能是主要区别之一。 我重新打开了它,因为我认为这个问题更多的是关于“为什么类的性能更高?” 您能否指出他们提到它的段落并对其进行推理?我在文章中看不到性能这个词。仅在其中一篇引用的文章中,并且只是声称 Class 允许性能优化(这并不是我问题的真正症结,因为这将是一个选项)。我的问题与使用函数与小部件时某种固有的性能不佳有关。 【参考方案1】:

首先,我想指出一个包可用于从函数中生成StatelessWidget:functional_widget


增益是性能不一定是真的。这取决于您如何使用小部件,主要是如何使用它们来管理您的状态。

默认情况下,与应用程序中不利用其功能的函数相比,类可能会降低性能。

真正的问题是:他们的力量是什么?

简单:类可以相互独立更新。函数不能

类可以部分更新小部件树。

考虑一个重建每一帧并返回其子帧的小部件:

class InfiniteLoop extends StatefulWidget 
  const InfiniteLoop(Key key, this.child) : super(key: key);
  final Widget child;
  @override
  _InfiniteLoopState createState() => _InfiniteLoopState();


class _InfiniteLoopState extends State<InfiniteLoop> 
  @override
  Widget build(BuildContext context) 
    WidgetsBinding.instance.addPostFrameCallback((_) => setState(() ));

    return widget.child;
  

现在,如果我们将整个应用程序包装在那个小部件中,会发生什么?

void main() => runApp(InfiniteLoop(child: MyApp()));

什么都没有

当然,您将拥有一个经常在您的树中重建的小部件。但实际上MyAppbuild方法只会被调用一次。

这是因为 Flutter 能够在小部件的实例未更改时中止树重建。


类可以滥用这种优化。

使用类可以巧妙地将小部件树的重建拆分为独立的部分。

列出一个类允许的所有潜在优化因素是不合理的,因为太多了。

以下示例是一个小部件,它采用int 并将其格式化为Text。问题是,如果int 传递了更改,此小部件将重建:

class Counter extends StatelessWidget 
  const Counter(Key key, this.value) : super(key: key);

  final int value;

  @override
  Widget build(BuildContext context) 
    return Text(value.toString());
  

  @override
  bool operator ==(Object other) =>
      identical(this, other) || (other is Counter && other.value == value);

  @override
  int get hashCode => value.hashCode;

这是因为 Flutter 使用 == 运算符来了解小部件是否应该更新(因此为什么 const 构造函数是一个很好的优化因素)。

这不是唯一的解决方案,但它是函数无法完成的一个很好的例子。

【讨论】:

感谢雷米的回答。我真的很感激 ;D 对我来说,关键是你的第一段:这取决于用法。这意味着您没有理由争辩说 StatelessWidget 本质上比函数性能最高(如果我们将自己从其他需要能够通过覆盖 operator== 等进行微调的用例中抽象出来) )。然后我的下一个问题,使用 InfiniteLoopWidget 的示例,我已经用一个函数对其进行了测试,但该函数只被调用一次 我无法合理地列出类在这里性能更高的所有原因。但我可以肯定地说,函数永远无法击败优化的类架构。 InfiniteLoop 示例与类与函数无关。它是关于展示类可以使用的优化因素之一。 为了明确我的观点,我绝对同意类允许在 Flutter 中进行更多优化,但在我看来,根据您的回答,您不认为返回小部件的函数本质上是在性能方面不好。这只是案例的用法和特殊性问题。很明显,如果您使用类,如果需要,您将始终打开案例进行优化。对吗?

以上是关于StatelessWidget 与在性能方面返回 Widget 的函数的主要内容,如果未能解决你的问题,请参考以下文章

StatefulWidget 和 StatelessWidget 的 Flutter 性能

MySQL使用默认值与在查询中设置值是不是有任何性能差异?

Function 或 StatelessWidget 哪个最好? [复制]

调用 Statelesswidget 中的方法以获取启动画面?

在 SQL Server 中运行 sum 与在代码中处理 sum 都有哪些性能问题? [关闭]

在 app.yaml 中定义路由与在 AppEngine 中的 WSGIApplication 中定义一个大型映射相比有性能提升吗?