尝试做 go koan,我陷入了理解接口(interface)(结构)语法的困境,究竟是什么 是吗? 我想出了以下有趣的程序,这让我对界面转换的工作方式更加困惑:
package main
import "fmt"
type foo interface{ fn() }
type t struct { }
type q struct { }
func (_i t ) fn() { fmt.Print("t","\n") }
func (_i q ) fn() { fmt.Print("q","\n")}
func main() {
_j := t{}
_q := q{}
// This is alright ..
fmt.Print( _j.fn,"\n") //0x4015e0
fmt.Print( _q.fn,"\n") //0x401610
_j.fn() //t
_q.fn() //q
// both pointers same .. why ?
fmt.Print( foo(_j).fn,"\n") //0x401640
fmt.Print( foo(_q).fn,"\n") //0x401640
// but correct fns called .. how ?
foo(_j).fn() //t
foo(_q).fn() //q
// same thing again ...
_fj := foo(_j).fn
_fq := foo(_q).fn
// both pointers same .. as above
fmt.Print( _fj,"\n") //0x401640
fmt.Print( _fq,"\n") //0x401640
// correct fns called .. HOW !
_fj() //t
_fq() //q
}
指针是我的机器,YMMV。 我的问题是.. interface(struct) 究竟返回什么? 以及 interface(struct).func 是如何找到原始结构的…… 这里有一些 thunk/stub 魔法吗?
最佳答案
来自这里:http://research.swtch.com/interfaces
what exactly does
interface(struct)return?
它创建一个新的接口(interface)值(就像您在图中顶部看到的那样),包装一个具体的结构值。
how does
interface(struct).funcfind the original struct?
请参阅图中的数据 字段。大多数情况下,这将是指向现有值的指针。不过,有时它会包含值本身(如果合适的话)。
在 itable 中,您会看到一个函数表(fun[0] 所在的位置)。
我假设在你的机器上 0x401640 是指向 fn 的相应指针的地址,它在 foo 的表中。虽然这最好由从事 GC 编译器套件的人员验证。
请注意,您发现的行为并未严格定义为如此。编译器构建者可以根据需要采用其他方法来实现 Go 接口(interface),只要 the language semantics被保留下来。
编辑以回答评论中的问题:
package main
import "fmt"
type foo interface {
fn()
}
type t struct{}
type q struct{}
func (_i t) fn() { fmt.Print("t", "\n") }
func (_i q) fn() { fmt.Print("q", "\n") }
func main() {
_j := t{}
_j1 := t{}
fmt.Println(foo(_j) == foo(_j)) // true
fmt.Println(foo(_j) == foo(_j1)) // true
}
在图表上您看到 3 个 block :
左侧标记为 Binary 的是具体类型实例,例如您的结构实例 _j 和 _j1。
顶部中间的是一个接口(interface)值,这个包裹(读作:指向)一个具体的值。
右下方的 block 是Binary底层的接口(interface)定义。这就是跳转表/调用转发表所在的位置 (itable)。
_j 和 _j1 是具体类型 t 的两个实例。所以在内存中的某处有两个左下角的 block 。
现在您决定将 _j 和 _j1 都包装在 foo 类型的接口(interface)值中;现在你在内存中的某处有 2 个顶部中心 block ,指向 _j 和 _j1。
为了让接口(interface)值记住它的底层类型是什么以及这些类型的方法在哪里,它在内存中保留右下角 block 的单个实例,两个接口(interface)值都指向该实例_j和_j1分别指向
在该 block 中,您有一个跳转表,用于将对接口(interface)值进行的方法调用转发到具体的底层类型的实现。这就是两者相同的原因。
值得一提的是,与 Java 和 C++(Python 不确定)不同,所有 Go 方法都是静态的,点调用符号只是语法糖。所以 _j 和 _j1 没有不同的 fn 方法,它是用另一个隐式第一个参数调用的完全相同的方法,该参数是接收者方法被调用。
关于go - interface(struct) 和 interface(struct).function 到底是什么,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19395749/
类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
我正在使用的第三方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
为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返
它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput
我可以得到Infinity和NaNn=9.0/0#=>Infinityn.class#=>Floatm=0/0.0#=>NaNm.class#=>Float但是当我想直接访问Infinity或NaN时:Infinity#=>uninitializedconstantInfinity(NameError)NaN#=>uninitializedconstantNaN(NameError)什么是Infinity和NaN?它们是对象、关键字还是其他东西? 最佳答案 您看到打印为Infinity和NaN的只是Float类的两个特殊实例的字符串
如果您尝试在Ruby中的nil对象上调用方法,则会出现NoMethodError异常并显示消息:"undefinedmethod‘...’fornil:NilClass"然而,有一个tryRails中的方法,如果它被发送到一个nil对象,它只返回nil:require'rubygems'require'active_support/all'nil.try(:nonexisting_method)#noNoMethodErrorexceptionanymore那么try如何在内部工作以防止该异常? 最佳答案 像Ruby中的所有其他对象
关闭。这个问题需要detailsorclarity.它目前不接受答案。想改进这个问题吗?通过editingthispost添加细节并澄清问题.关闭8年前。Improvethisquestion为什么SecureRandom.uuid创建一个唯一的字符串?SecureRandom.uuid#=>"35cb4e30-54e1-49f9-b5ce-4134799eb2c0"SecureRandom.uuid方法创建的字符串从不重复?
我刚刚被困在这个问题上一段时间了。以这个基地为例:moduleTopclassTestendmoduleFooendend稍后,我可以通过这样做在Foo中定义扩展Test的类:moduleTopmoduleFooclassSomeTest但是,如果我尝试通过使用::指定模块来最小化缩进:moduleTop::FooclassFailure这失败了:NameError:uninitializedconstantTop::Foo::Test这是一个错误,还是仅仅是Ruby解析变量名的方式的逻辑结果? 最佳答案 Isthisabug,or