- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在包装一些 API 调用来假装套接字结果 - 以几秒的间隔循环调用并在更改时发出结果。
使用 DistinctUntilChanged()
运算符时效果很好。但是,当结果是列表时,运算符始终发出结果,因为它与默认比较器不同。
这是我自定义的可观察对象,用于重复任务,无论成功还是失败,都会有一定的延迟。
public class TaskRepeatObservable<T> : IObservable<T>
{
private readonly Func<Task<T>> _taskFactory;
private readonly TimeSpan _repeatDelayTimeSpan;
private readonly ILogger _logger;
private Func<Exception, bool> _onError;
public TaskRepeatObservable(Func<Task<T>> taskFactory, TimeSpan repeatDelayTimeSpan = default(TimeSpan), Func<Exception, bool> onError = null)
{
_logger = new Log4NetLogger(GetType());
_logger.IsEnabled = false;
_taskFactory = taskFactory;
_repeatDelayTimeSpan = repeatDelayTimeSpan;
_onError = onError;
}
public IDisposable Subscribe(IObserver<T> observer)
{
var tokenSource = new CancellationTokenSource();
var cancellationToken = tokenSource.Token;
Task.Factory.StartNew(async () =>
{
try
{
while (!cancellationToken.IsCancellationRequested)
{
try
{
var result = await _taskFactory();
observer.OnNext(result);
}
catch (Exception e)
{
_logger.Error(e, "Observable Error: " + e.Message);
if (_onError != null && !_onError.Invoke(e))
throw;
}
finally
{
try
{
if (_repeatDelayTimeSpan > TimeSpan.Zero)
await Task.Delay(_repeatDelayTimeSpan, cancellationToken);
}
catch (TaskCanceledException)
{
// ignored
}
}
}
}
catch (Exception e)
{
observer.OnError(e);
}
_logger.Debug("Observable is cancelled.");
}, cancellationToken, TaskCreationOptions.LongRunning, TaskScheduler.Default);
return Disposable.Create(() =>
{
tokenSource.Cancel();
});
}
}
这是包装 API 调用的扩展。
public static class ObservableBuilder
{
///<summary>
///<para>Convert Task to Observable and emit only changed result, it's useful to wrap the api as socket-like.</para>
///</summary>
public static IObservable<T> Repeat<T>(this Func<Task<T>> taskFactory, TimeSpan delay = default(TimeSpan),
Func<Exception, bool> onError = null)
{
return new TaskRepeatObservable<T>(taskFactory, delay, onError).DistinctUntilChanged();
}
我的问题是 - 如何使 DistinctUntilChanged()
适用于任何结果,包括 List
或 Enumerable
。
注意到我尝试实现我自己的比较器。但我仍然不知道如何检查 T
的类型来为 DistinctUntilChanged()
public class IEnumerableComparer<T> : IEqualityComparer<IEnumerable<T>>
{
public bool Equals(IEnumerable<T> x, IEnumerable<T> y)
{
return ReferenceEquals(x, y) || x != null && y != null && x.SequenceEqual(y);
}
public int GetHashCode(IEnumerable<T> obj)
{
// Will not throw an OverflowException
unchecked
{
return obj.Where(e => e != null).Select(e => e.GetHashCode()).Aggregate(17, (a, b) => 23 * a + b);
}
}
}
这是简单的测试代码:
ObservableBuilder.Repeat(async () =>
{
var i = new List<int>() { 1, 2, 3, 4 };
return i;
}, TimeSpan.FromSeconds(1)).ToHotObservable().Subscribe(x => Logger.Info($"Result = {x}"));
我希望列表只发出一次结果。
最佳答案
如果你只想拥有一个Repeat
函数可以转换所有情况,但对于 IEnumerable
的工作方式有所不同s - 你必须使用一些反射。您需要检查是否输入 T
实现IEnumerable<Something>
如果是这样 - 使用 DistinctUtilChanged
的特殊比较器- 否则使用默认比较器。
首先,修改你的EnumerableComparer
的签名不会有什么坏处。一点,因为您将特别需要 IEqualityComparer<T>
,不是IEqualityComparer<IEnumerable<T>>
:
private class EnumerableComparer<T, TItem> : IEqualityComparer<T> where T : IEnumerable<TItem> {
public bool Equals(T x, T y) {
return ReferenceEquals(x, y) || x != null && y != null && x.SequenceEqual(y);
}
public int GetHashCode(T obj) {
// Will not throw an OverflowException
unchecked {
return obj.Where(e => e != null).Select(e => e.GetHashCode()).Aggregate(17, (a, b) => 23 * a + b);
}
}
}
现在我们需要检查是否 T
是 IEnumerable
并通过反射创建此比较器的实例:
public static class ObservableBuilder {
public static IObservable<T> Repeat<T>(this Func<Task<T>> taskFactory, TimeSpan delay = default(TimeSpan),
Func<Exception, bool> onError = null) {
var ienum = typeof(T).GetInterfaces().FirstOrDefault(c => c.IsGenericType && c.GetGenericTypeDefinition() == typeof(IEnumerable<>));
if (ienum != null) {
// implements IEnumerable - create instance of comparer and use
var comparer = (IEqualityComparer<T>) Activator.CreateInstance(typeof(EnumerableComparer<,>).MakeGenericType(typeof(T), ienum.GenericTypeArguments[0]));
return new TaskRepeatObservable<T>(taskFactory, delay, onError).DistinctUntilChanged(comparer);
}
// otherwise - don't use
return new TaskRepeatObservable<T>(taskFactory, delay, onError).DistinctUntilChanged();
}
private class EnumerableComparer<T, TItem> : IEqualityComparer<T> where T : IEnumerable<TItem> {
public bool Equals(T x, T y) {
return ReferenceEquals(x, y) || x != null && y != null && x.SequenceEqual(y);
}
public int GetHashCode(T obj) {
// Will not throw an OverflowException
unchecked {
return obj.Where(e => e != null).Select(e => e.GetHashCode()).Aggregate(17, (a, b) => 23 * a + b);
}
}
}
}
或者,您可以创建 IEqualityComparer<T>
每次都会检查 x
和y
实现IEnumerable<Something>
并进行相应的比较,但我预计与构建序列时只进行一次比较相比,每次比较的效率都会较低。
关于C# Rx DistinctUntilChanged() 用于任何结果,包括列表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50383674/
127.0.0.1:8000/api/仅包含来自第二个应用程序的 url,但我将两个 url 模块链接到相同的模式。甚至有可能做到这一点吗? 第一个应用程序: from django.urls imp
我目前正在学习 ColdFusion。我有 PHP 背景,对此我有点困惑。 我有一个选择菜单,我希望将选项保存在不同的文件中。 (例如 options.cfm)当我调用文件时,我想在选择菜单中包含选项
字符串: "75,000", "is", "95%", "or", "95/100" "of", "monthly", "income" o/p: "is","%, "or", "/", "of",
我有 4 个 javascript 文件(每个文件对应一个 HTML 文件),所有 4 个文件中的 3 个函数都是相同的。我想找到一个顺利的解决方案,我可以以某种方式分别包含这 3 个函数...是否可
我在 PHP 中有这种情况,其中 include在一台服务器上被遗漏,但在另一台服务器上没有(我没有设置服务器,所以我不能告诉你更多;我不是真正的 devops 人,所以这就是我在这里问的原因)。两台
这是一个模式文件,midi.xsd定义类型,note ,用于存储 MIDI 音符值: 这是另一个模式文件,octaves.xsd使用
我想备份以下文件夹 /home /etc /usr/local /root /var /boot 并排除 /var/tmp /var/run /var/lock /home/*/.thumbnails
如何重新编码具有许多值(包括缺失值)的数值变量,以获得数字 0:n-1哪里n是唯一值的数量,包括 NA ,整齐? 例子: df 1 1000 0 2 1000 0 3 N
选择元素的 html(包括在内)的最佳方法是什么?例如: This is just a test. 而$('#testDiv').html()返回"This is just a test."
我正在尝试设置Varnish来处理本地环境中的ESI包含。 我在虚拟机中运行 Varnish ,内容在主机上运行。 我有两个文件“index.html”和“test.html”。它们都存储在apach
我有以下内容,并且想要检索“ FromEmail”不为空的数据 Simple email@gma
欧海,我正在编写一个小型 PHP 应用程序,使用一个单独的 config.php 文件和一个functions.php,其中包含我将在应用程序中使用的所有自定义函数。现在,我真的必须在每个函数中包含
我知道可以将 JavaScript 放在一个特定的 .js 文件中,然后通过执行以下操作将其包含在任何页面中...... 我注意到,对于包含的这些 .js 文件: 它们实际上不必以 .js 结尾 其
我使用 gwt UIBinder 添加了一些项目到我的 ComboBox。 --select one-- Dispute Referral Form Dispute Settlement Clause
我可以将一个 first.c 文件包含到另一个 second.c 中吗? (我正在做一些套接字编程,以将服务器收到的消息存储在链接列表中,因此在第一个程序中,我尝试保留链接列表和第二个程序套接字编程文
我有一个简单的 Spring MVC 数据项目设置,我试图选择 Admin 中尚不存在的用户列表。 table 。这是我的存储库方法 SELECT u FROM User u WHERE u.id N
在 bash 脚本中,使用什么实用程序以及如何删除两个字符串之间的文本,包括字符串。 原文: (ABC blah1)blah 2(def blah 5)blah 7)(DEF blah 8)blah
我有这个 BST 问题,我试图用 Java 解决,但我不知道为什么它不起作用。问题是: 二叉搜索树 (BST) 是一种二叉树,其中每个值节点大于或等于该节点的所有节点中的值左子树并且小于该树中所有节点
我有一个字符串,其中包含“Dollars”和“Cents”符号。我想删除它们。我试过了 string.replaceAll("[\"\\u00A2\" $]", "") 但它不起作用。正确的做法是什么
我在 stories 和 tags 之间有一个多对多的关系,为保存关系而创建的表是 taxonomies。我想搜索所有具有所有给定标签的故事。 到目前为止我使用的查询是这个,当然它对我不起作用,它返回
我是一名优秀的程序员,十分优秀!