Flutter GetX基礎(chǔ)教程(十二):RxList忽孽、Rx([])、.obs對比分析

前言

首先我們知道GetX組件里面obs狀態(tài)管理有三種創(chuàng)建屬性的方式盯另,我們這里以List為例

  • Rx<t style="box-sizing: border-box;">([])</t>
  • RxList
  • .obs

視頻講解

視頻講解鏈接

三種方式對比分析

我們聲明了一個類ListController繼承自GetxController性含,用于屬性創(chuàng)建以及狀態(tài)通知的方法,首先我們用三種方式來創(chuàng)建屬性并且通過convertToUpperCase方法進(jìn)行對值的改變鸳惯,然后我們通過調(diào)用update()`方法來進(jìn)行數(shù)據(jù)更新商蕴,最后我們使用該屬性狀態(tài)的值,接下來我們看一下三種使用方式的對比悲敷。

  • 第一種Rx<t style="box-sizing: border-box;">([])</t>
  • 第二種RxList
  • 第三種 .obs

import 'dart:convert';
import 'package:get/get.dart';

class ListController extends GetxController {
// 第一種
final listOne = Rx<List<Map>>([
{
"name": "Jimi",
"age": 18
}
]);

// 第二種
final listTwo = RxList([
{
"name": "Jimi",
"age": 18
}
]);

// 第三種
final listThree = [{
"name": "Jimi",
"age": 18
}].obs;

void convertToUpperCase() {
listOne.value[0]["name"] = listOne.value.first["name"].toUpperCase();
listTwo.toList().first["name"] = listTwo.toList().first["name"].toString().toUpperCase();
listThree.toList().first["name"] = listTwo.toList().first["name"].toString().toUpperCase();
update();
}
}

我們在頁面中獲取狀態(tài)更新的值

import 'package:flutter/material.dart';
import 'package:flutter_getx_dvanced_example/ListController.dart';
import 'package:get/get.dart';

void main() {
runApp(MyApp());
}

class MyApp extends StatelessWidget {

ListController listController = Get.put(ListController());

@override
Widget build(BuildContext context) {
return GetMaterialApp(
title: "GetX",
home: Scaffold(
appBar: AppBar(
title: Text("GetX"),
),
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
GetBuilder<ListController>(
init: listController,
builder: (controller) {
return Text(
"我的名字是 {controller.listOne.value.first['name']}", style: TextStyle(color: Colors.orange, fontSize: 30), ); }, ), SizedBox(height: 20,), GetBuilder<ListController>( init: listController, builder: (controller) { return Text( "我的名字是{controller.listTwo.first['name']}",
style: TextStyle(color: Colors.green, fontSize: 30),
);
},
),
SizedBox(height: 20,),
GetBuilder<ListController>(
init: listController,
builder: (controller) {
return Text(
"我的名字是 ${controller.listThree.first['name']}",
style: TextStyle(color: Colors.green, fontSize: 30),
);
},
),
SizedBox(height: 20,),
ElevatedButton(
onPressed: () {
listController.convertToUpperCase();
},
child: Text("轉(zhuǎn)換為大寫"))
],
),
),
),
);
}
}
</pre>

|`

效果展示

Rx<t style="box-sizing: border-box;">([])源碼分析</t>

Rx<T> 繼承自_RxImpl<T>究恤,_RxImpl<T>又繼承RxNotifier<T>并混合 RxObjectMixin<T>

RxImpl<T>它主要的作用是管理泛型的所有邏輯的。

RxObjectMixin<T> 它主要的作用是管理注冊到GetXObx的全局對象后德,比如WidgetRx

Rx<T>它主要的作用是將自定義模型類用Rx`來進(jìn)行包裝,

class Rx<T> extends _RxImpl<T> {
Rx(T initial) : super(initial);

@override
dynamic toJson() {
try {
return (value as dynamic)?.toJson();
} on Exception catch (_) {
throw '$T has not method [toJson]';
}
}
}

abstract class _RxImpl<T> extends RxNotifier<T> with RxObjectMixin<T> {
_RxImpl(T initial) {
_value = initial;
}

void addError(Object error, [StackTrace? stackTrace]) {
subject.addError(error, stackTrace);
}

Stream<R> map<R>(R mapper(T? data)) => stream.map(mapper);

void update(void fn(T? val)) {
fn(_value);
subject.add(_value);
}

void trigger(T v) {
var firstRebuild = this.firstRebuild;
value = v;
if (!firstRebuild) {
subject.add(v);
}
}
}
</pre>

|`

RxList<e style="box-sizing: border-box;">源碼分析</e>

RxList<E>繼承自ListMixin<E>實現(xiàn)了RxInterface<List<E>>并混合了NotifyManager<List<E>>, RxObjectMixin<List<E>>

RxList<E>它的主要作用是創(chuàng)建一個類似于List<T> 的一個列表

class RxList<E> extends ListMixin<E>
with NotifyManager<List<E>>, RxObjectMixin<List<E>>
implements RxInterface<List<E>> {
RxList([List<E> initial = const []]) {
_value = List.from(initial);
}

factory RxList.filled(int length, E fill, {bool growable = false}) {
return RxList(List.filled(length, fill, growable: growable));
}

factory RxList.empty({bool growable = false}) {
return RxList(List.empty(growable: growable));
}

/// Creates a list containing all [elements].
factory RxList.from(Iterable elements, {bool growable = true}) {
return RxList(List.from(elements, growable: growable));
}

/// Creates a list from [elements].
factory RxList.of(Iterable<E> elements, {bool growable = true}) {
return RxList(List.of(elements, growable: growable));
}

/// Generates a list of values.
factory RxList.generate(int length, E generator(int index),
{bool growable = true}) {
return RxList(List.generate(length, generator, growable: growable));
}

/// Creates an unmodifiable list containing all [elements].
factory RxList.unmodifiable(Iterable elements) {
return RxList(List.unmodifiable(elements));
}

@override
Iterator<E> get iterator => value.iterator;

@override
void operator []=(int index, E val) {
_value[index] = val;
refresh();
}

/// Special override to push() element(s) in a reactive way
/// inside the List,
@override
RxList<E> operator +(Iterable<E> val) {
addAll(val);
refresh();
return this;
}

@override
E operator [](int index) {
return value[index];
}

@override
void add(E item) {
_value.add(item);
refresh();
}

@override
void addAll(Iterable<E> item) {
_value.addAll(item);
refresh();
}

@override
int get length => value.length;

@override
@protected
List<E> get value {
RxInterface.proxy?.addListener(subject);
return _value;
}

@override
set length(int newLength) {
_value.length = newLength;
refresh();
}

@override
void insertAll(int index, Iterable<E> iterable) {
_value.insertAll(index, iterable);
refresh();
}

@override
Iterable<E> get reversed => value.reversed;

@override
Iterable<E> where(bool Function(E) test) {
return value.where(test);
}

@override
Iterable<T> whereType<T>() {
return value.whereType<T>();
}

@override
void sort([int compare(E a, E b)?]) {
_value.sort(compare);
refresh();
}
}
</pre>

|`

.obs源碼分析

當(dāng)我們在調(diào)用.obs的時候其實內(nèi)部的實現(xiàn)源碼還是通過RxList<e>(this)進(jìn)行了一層包裝抄腔,設(shè)計這個主要的目的就是為了方便開發(fā)者進(jìn)行使用

ListExtension<E> on List<E> {
RxList<E> get obs => RxList<E>(this);

/// Add [item] to [List<E>] only if [item] is not null.
void addNonNull(E item) {
if (item != null) add(item);
}

// /// Add [Iterable<E>] to [List<E>] only if [Iterable<E>] is not null.
// void addAllNonNull(Iterable<E> item) {
// if (item != null) addAll(item);
// }

/// Add [item] to List<E> only if [condition] is true.
void addIf(dynamic condition, E item) {
if (condition is Condition) condition = condition();
if (condition is bool && condition) add(item);
}

/// Adds [Iterable<E>] to [List<E>] only if [condition] is true.
void addAllIf(dynamic condition, Iterable<E> items) {
if (condition is Condition) condition = condition();
if (condition is bool && condition) addAll(items);
}

/// Replaces all existing items of this list with [item]
void assign(E item) {
// if (this is RxList) {
// (this as RxList)._value;
// }

clear();
add(item);

}

/// Replaces all existing items of this list with [items]
void assignAll(Iterable<E> items) {
// if (this is RxList) {
// (this as RxList)._value;
// }
clear();
addAll(items);
}
}
</pre>

|`

總結(jié)

我們對Rx<T>([])瓢湃、RxList<E>理张、.obs進(jìn)行了一個總結(jié),在我們平時的開發(fā)過程中建議大家使用.obs即可绵患,因為這是最簡單的方式雾叭。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市落蝙,隨后出現(xiàn)的幾起案子织狐,更是在濱河造成了極大的恐慌,老刑警劉巖筏勒,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件移迫,死亡現(xiàn)場離奇詭異,居然都是意外死亡管行,警方通過查閱死者的電腦和手機(jī)厨埋,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來捐顷,“玉大人荡陷,你說我怎么就攤上這事⊙镐蹋” “怎么了废赞?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長叮姑。 經(jīng)常有香客問我蛹头,道長,這世上最難降的妖魔是什么戏溺? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任渣蜗,我火速辦了婚禮,結(jié)果婚禮上旷祸,老公的妹妹穿的比我還像新娘耕拷。我一直安慰自己,他們只是感情好托享,可當(dāng)我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布骚烧。 她就那樣靜靜地躺著,像睡著了一般闰围。 火紅的嫁衣襯著肌膚如雪赃绊。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天羡榴,我揣著相機(jī)與錄音碧查,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛忠售,可吹牛的內(nèi)容都是我干的传惠。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼稻扬,長吁一口氣:“原來是場噩夢啊……” “哼卦方!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起泰佳,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤盼砍,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后逝她,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體浇坐,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年汽绢,在試婚紗的時候發(fā)現(xiàn)自己被綠了吗跋。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡宁昭,死狀恐怖跌宛,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情积仗,我是刑警寧澤疆拘,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站寂曹,受9級特大地震影響哎迄,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜隆圆,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一漱挚、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧渺氧,春花似錦旨涝、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至贩耐,卻和暖如春弧腥,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背潮太。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工管搪, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓抛蚤,卻偏偏與公主長得像台谢,于是被迫代替她去往敵國和親寻狂。 傳聞我的和親對象是個殘疾皇子岁经,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 42,916評論 2 344

推薦閱讀更多精彩內(nèi)容

  • GetxController介紹 在實際的項目開發(fā)過程中,我們不可能把UI代碼蛇券、業(yè)務(wù)邏輯都放在一起處理缀壤,這樣對項目...
    kadis閱讀 4,011評論 1 5
  • 一、基礎(chǔ)知識:1纠亚、JVM塘慕、JRE和JDK的區(qū)別:JVM(Java Virtual Machine):java虛擬機(jī)...
    殺小賊閱讀 2,365評論 0 4
  • Python基礎(chǔ) 介紹 輸入和輸出 所有的通過input獲取的數(shù)據(jù),都是字符串類型 print() 變量 程序就是...
    沙漠星海說遠(yuǎn)方近閱讀 352評論 0 0
  • [TOC] Dart基礎(chǔ) 在Dart中,所有能夠使用變量引用的都是對象object蒂胞,每個對象都是一個類class的...
    FreeRain77閱讀 870評論 0 1
  • 〇图呢、前言 本文共108張圖,流量黨請慎重骗随! 歷時1個半月蛤织,我把自己學(xué)習(xí)Python基礎(chǔ)知識的框架詳細(xì)梳理了一遍。 ...
    Raxxie閱讀 18,922評論 17 410