我在 C# 中看到的大多数 monad 示例都是这样写的:
public static Identity<B> Bind<A, B>(this Identity<A> a, Func<A, Identity<B>> func) {
return func(a.Value);
}
例如,参见 http://mikehadlow.blogspot.com/2011/01/monads-in-c-3-creating-our-first-monad.html .
问题是,要求 func 有什么意义?返回 Identity<B> ?如果我使用以下定义:
public interface IValue<A> {
public IValue<B> Bind<B>(Func<A, B> func)
}
然后我实际上可以使用相同的 func对于 Lazy<T> , Task<T> , Maybe<T>等实际上不依赖于实际类型实现 IValue .
我在这里遗漏了什么重要的东西吗?
最佳答案
首先,考虑组合 的概念。我们可以轻松地将组合表达为对委托(delegate)的操作:
public static Func<T, V> Compose<T, U, V>(this Func<U, V> f, Func<T, U> g)
{
return x => f(g(x));
}
所以如果我有一个函数 g 是 (int x) => x.ToString()和一个函数 f 是 (string s) => s.Length然后我可以创建一个组合函数 h,它是 (int x) => x.ToString().Length调用f.Compose(g) .
这应该很清楚了。
现在假设我有一个来自 T 的函数 g至 Monad<U>和来自 U 的函数 f至 Monad<V> .我希望编写一个方法,将这两个返回 monad 的函数组合成一个接受 T 的函数。并返回 Monad<V> .所以我试着这样写:
public static Func<T, Monad<V>> Compose<T, U, V>(this Func<U, Monad<V>> f, Func<T, Monad<U>> g)
{
return x => f(g(x));
}
不起作用。 g返回 Monad<U>但是f需要 U .我有办法“包装”一个 U进入Monad<U>但我没有办法“打开”一个。
但是,如果我有一个方法
public static Monad<V> Bind<U, V>(this Monad<U> m, Func<U, Monad<V>> k)
{ whatever }
然后我可以编写一个方法,该方法由两个都返回 monad 的方法组成:
public static Func<T, Monad<V>> Compose<T, U, V>(this Func<U, Monad<V>> f, Func<T, Monad<U>> g)
{
return x => Bind(g(x), f);
}
这就是为什么 Bind 从 T 获取函数的原因至 Monad<U> -- 因为事情的重点是能够从 T 中获取函数 g至 Monad<U>和来自 U 的函数 f至 Monad<V>并将它们组合成来自 T 的函数 h至 Monad<V> .
如果你想从T中获取函数g至 U和来自 U 的函数 f至 Monad<V>那么您一开始就不需要 Bind。只需正常编写函数 即可从T 获取方法。至 Monad<V> ! Bind 的全部目的就是解决这个问题;如果您解决了这个问题,那么您一开始就不需要 Bind。
更新:
In most cases I want to compose function g from
TtoMonad<U>and function f fromUtoV.
然后我假设您希望将其组合成来自 T 的函数至 V .但是你不能保证定义了这样的操作!例如,以“Maybe monad”作为monad,在C#中表示为T? .假设你有 g 作为 (int x)=>(double?)null你有一个函数 f 是 (double y)=>(decimal)y .您应该如何将 f 和 g 组合成一个方法,该方法接受一个 int 并返回不可为 null 的 decimal类型?没有将可为 null 的 double 解包为 f 可以采用的 double 值的“解包”!
您可以使用 Bind 将 f 和 g 组合成一个接受 int 并返回可为 null 的小数的方法:
public static Func<T, Monad<V>> Compose<T, U, V>(this Func<U, V> f, Func<T, Monad<U>> g)
{
return x => Bind(g(x), x=>Unit(f(x)));
}
其中 Unit 是一个接受 V 的函数并返回 Monad<V> .
但是如果 g 返回一个 monad 而 f 不返回 monad,那么 f 和 g 根本就没有组合——不能保证有一种方法可以从 monad 的实例返回到一个“未包装的” “类型。也许在一些单子(monad)的情况下总是有——比如 Lazy<T> .或者有时可能会有,就像“可能”monad 一样。通常有办法做到这一点,但没有要求您可以这样做。
顺便说一下,请注意我们是如何将“Bind”用作瑞士军刀来制作一种新型组合的。绑定(bind)可以进行任何操作!例如,假设我们在序列 monad 上有 Bind 操作,我们在 IEnumerable<T> 上称之为“SelectMany”在 C# 中输入:
static IEnumerable<V> SelectMany<U, V>(this IEnumerable<U> sequence, Func<U, IEnumerable<V>> f)
{
foreach(U u in sequence)
foreach(V v in f(u))
yield return v;
}
你可能还有一个序列操作符:
static IEnumerable<A> Where<A>(this IEnumerable<A> sequence, Func<A, bool> predicate)
{
foreach(A item in sequence)
if (predicate(item))
yield return item;
}
您真的需要在 Where 中编写该代码吗? ?不!您可以完全使用“Bind/SelectMany”构建它:
static IEnumerable<A> Where<A>(this IEnumerable<A> sequence, Func<A, bool> predicate)
{
return sequence.SelectMany((A a)=>predicate(a) ? new A[] { a } : new A[] { } );
}
高效?不,但是没有什么是 Bind/SelectMany 做不到的。如果你真的想要,你可以只用 SelectMany 构建所有的 LINQ 序列运算符。
关于c# - C# 中的 Monads——为什么 Bind 实现需要传递的函数来返回 monad?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7828528/
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时
作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代
Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何
为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返