我的代码一般是这样的:
func BulkInsert(docs interface{}) {
switch data := docs.(type) {
case map[string] *model.SnapshotByConv, map[string] *model.UserSnapshotsMap:
for ver, _ := range data {
// other logics ...
}
case map[int64] map[string] *model.Delta:
for ver, _ := range data {
// other logics ...
}
}
}
然后在编译时出现错误:
不能覆盖数据(类型接口(interface) {}),它由第一个 range 引发。
如果我在第一种情况下删除多个类型,这意味着将其保留为 case map[string] *model.SnapshotByConv:
然后编译错误就消失了,它是 wried,我需要对这两种类型进行完全相同的逻辑,那么为什么我不能将它们放在相同的 case 中?
请帮忙,谢谢。
我认为这里的情况与这里已经有答案的情况不同:golang multiple case in type switch ,它试图找到一种识别类型的方法,但我只是不想识别某些类型,只是在其上运行一些逻辑,我找不到一种优雅的方法来做到这一点。
最佳答案
如果案例枚举多种类型(就像您的示例中那样),变量的类型将不会缩小,因此在您的案例中它将是 interface{}。这就是您链接到的问题所说的内容,没有办法解决它。
要为两种不同类型的情况分派(dispatch)一些通用代码,您需要在两者之间引入一个接口(interface)。 (如果 Go 在某个时候引入泛型,这可能会改变,但从外观上看,这不会很快发生。)有几种方法可以做到这一点,这里是三个:
不要直接使用映射,而是定义映射类型,并为其添加函数。
type SnapshotMap map[string]*model.SnapshotByConv
type UserSnapshotMap map[string]*model.UserSnapshotsMap
func (sm *SnapshotMap) DoLogic() {/*...*/}
func (usm *UserSnapshotMap) DoLogic() {/*...*/}
type LogicDoer interface {
DoLogic()
}
// ...
switch data := docs.(type) {
case LogicDoer:
data.DoLogic()
// ...
}
不要创建指针映射,而是创建快照类型共享的接口(interface)映射。
type Snapshot interface {
LogicCommonToSnapshots()
}
var doc interface{}
// Somewhere else, this happens rather than a creating map[string]*Something
doc = make(map[string]Snapshot)
// ...
switch data := doc.(type) {
case map[string]Snapshot:
for ver, _ := range data {
ver.LogicCommonToSnapshots()
}
// ...
}
将 switch case 拆分为两个独立的 case,但将逻辑分解为一个可以对任一快照类型进行操作的函数。这样,您就不必重复逻辑。
type Snapshot interface {
LogicCommonToSnapshots()
}
func ComplexLogic(s Snapshot) {/*...*/}
switch data := doc.(type) {
case map[string] *model.SnapshotByConv:
for ver, _ := range data
ComplexLogic(ver)
}
case map[string] *model.UserSnapshotsMap:
for ver, _ := range data {
ComplexLogic(ver)
}
// ...
}
请注意,在所有这三种情况下,都有一个接口(interface)将公共(public)逻辑与逻辑作用的对象分开。您必须必要地在逻辑和对象之间引入动态绑定(bind)以完成您描述的内容,无论是通过接口(interface)还是通过例如创建一个捕获对象的匿名函数将传递给逻辑。
关于Go:类型切换断言中的这种多案例条件有什么问题?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43017440/
类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
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我想为Heroku构建一个Rails3应用程序。他们使用Postgres作为他们的数据库,所以我通过MacPorts安装了postgres9.0。现在我需要一个postgresgem并且共识是出于性能原因你想要pggem。但是我对我得到的错误感到非常困惑当我尝试在rvm下通过geminstall安装pg时。我已经非常明确地指定了所有postgres目录的位置可以找到但仍然无法完成安装:$envARCHFLAGS='-archx86_64'geminstallpg--\--with-pg-config=/opt/local/var/db/postgresql90/defaultdb/po
我正在使用的第三方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
尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub
为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返
我正在从erb文件切换到HAML。我将hamlgem添加到我的系统中。我创建了app/views/layouts/application.html.haml文件。我应该只删除application.html.erb文件吗?此外,仍然有/public/index.html文件被呈现为默认页面。我想创建自己的默认index.html.haml页面。我应该把它放在哪里以及如何使系统呈现该文件而不是默认索引文件?谢谢! 最佳答案 是的,您可以删除任何已转换为HAML的View的ERB版本。至于你的另一个问题,删除public/index/h
它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput
我的最终目标是安装当前版本的RubyonRails。我在OSXMountainLion上运行。到目前为止,这是我的过程:已安装的RVM$\curl-Lhttps://get.rvm.io|bash-sstable检查已知(我假设已批准)安装$rvmlistknown我看到当前的稳定版本可用[ruby-]2.0.0[-p247]输入命令安装$rvminstall2.0.0-p247注意:我也试过这些安装命令$rvminstallruby-2.0.0-p247$rvminstallruby=2.0.0-p247我很快就无处可去了。结果:$rvminstall2.0.0-p247Search