async_util.dart 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130
  1. import 'dart:async';
  2. import 'package:flutter/widgets.dart';
  3. import 'async_typeof.dart';
  4. import 'cancel_future.dart';
  5. class AsyncUtil {
  6. AsyncUtil._();
  7. static CancelableFuture<T> retryWithExponentialBackoff<T>(
  8. FutureCallback<T> callback, int maxRetry, Predicate<dynamic> predicate) {
  9. const Duration initialInterval = Duration(seconds: 1);
  10. int retryCount = 0;
  11. Timer? timer;
  12. void attempt(Completer<T> completer) {
  13. callback().then((value) {
  14. if (!completer.isCompleted) {
  15. completer.complete(value);
  16. }
  17. }).catchError((error) {
  18. if (retryCount < maxRetry && predicate(error)) {
  19. retryCount++;
  20. Duration nextInterval = initialInterval * (1 << (retryCount - 1));
  21. timer = Timer(nextInterval, () => attempt(completer));
  22. } else {
  23. if (!completer.isCompleted) {
  24. completer.completeError(error);
  25. }
  26. }
  27. });
  28. }
  29. return CancelableFuture<T>((completer) {
  30. attempt(completer);
  31. }, () {
  32. timer?.cancel();
  33. });
  34. }
  35. static CancelableFuture<T> retryWhen<T>(FutureCallback<T> callback,
  36. int maxRetry, Duration interval, Predicate<dynamic> predicate) {
  37. int retryCount = 0;
  38. Timer? timer;
  39. void attempt(Completer<T> completer) {
  40. callback().then((value) {
  41. if (!completer.isCompleted) {
  42. completer.complete(value);
  43. }
  44. }).catchError((error) {
  45. if ((maxRetry <= 0 || retryCount < maxRetry) && predicate(error)) {
  46. retryCount++;
  47. timer = Timer(interval, () => attempt(completer));
  48. } else {
  49. if (!completer.isCompleted) {
  50. completer.completeError(error);
  51. }
  52. }
  53. });
  54. }
  55. return CancelableFuture<T>((completer) {
  56. attempt(completer);
  57. }, () {
  58. timer?.cancel();
  59. });
  60. }
  61. static CancelableFuture<T> retry<T>(
  62. FutureCallback<T> callback, int maxRetry, Duration interval) {
  63. return retryWhen(callback, maxRetry, interval, (error) => true);
  64. }
  65. static CancelableFuture<T> delay<T>(
  66. FutureCallback<T> callback, Duration interval) {
  67. Timer? timer;
  68. return CancelableFuture<T>((completer) {
  69. timer = Timer(interval, () {
  70. callback().then((value) {
  71. if (!completer.isCompleted) {
  72. completer.complete(value);
  73. }
  74. }).catchError((error) {
  75. if (!completer.isCompleted) {
  76. completer.completeError(error);
  77. }
  78. });
  79. });
  80. }, () {
  81. timer?.cancel();
  82. });
  83. }
  84. ///实际delay第一次执行的时间为delay+interval
  85. static CancelableFuture<T> interval<T>(FutureCallback<T> callback,
  86. Duration delay, Duration interval, int times) {
  87. Timer? timer;
  88. Timer? delayTimer;
  89. int counter = 0;
  90. void tick(Completer<T> completer) {
  91. if (counter < times) {
  92. callback().then((value) {
  93. if (!completer.isCompleted) {
  94. completer.complete(value);
  95. }
  96. }).catchError((error) {
  97. if (!completer.isCompleted) {
  98. completer.completeError(error);
  99. }
  100. });
  101. counter++;
  102. } else {
  103. timer?.cancel();
  104. }
  105. }
  106. return CancelableFuture<T>((completer) {
  107. delayTimer = Timer(delay, () {
  108. timer = Timer.periodic(interval, (Timer t) => tick(completer));
  109. });
  110. }, () {
  111. delayTimer?.cancel();
  112. timer?.cancel();
  113. });
  114. }
  115. }