Golang - the good, the bad, the ugly 25 June 2015 - devnation.org Vincent Batts Engineer @vbatts vbatts@redhat.com https://github.com/vbatts/talks * howdy $> finger $(whoami) Login: vbatts Name: Vincent Batts Such mail. Plan: OHMAN $> id -Gn vbatts devel openshift docker * golang ## Notes # - libraries - source only, but can link to C *.so and *.a # - Fork/Exec are coupled together (for coroutine and GC reasons) # - Threading, and multiproc, concurrent logic # -- nice and easy to use # -- Make for tricky handling of C calls (i.e. setns) # - `go get` is handy # - cross-compile without hardly any bootstrapping # -- native compiler supported arches # -- gccgo works for the arch gcc is compiled for # - primitives can seem a bit magical # -- conditional returns # -- for ... range # -- iota # - no ifdefs, but build tags # - error handling, rather than exception catching * .image ./gopher.png * Overview - strongly typed - compiled - stylistically nice - opinionated * Good * .image ./rainbow.jpg * fully qualified imports .code ./imports.go /START1/,/STOP1/ * fast compiles (Perhaps other compilers are slow) .play ./hello.go /START1/,/STOP1/ * defer .code ./good0.go /START1/,/STOP1/ * Garbage Collected - Super convenient - references - completed goroutines * simple exports .code ./good1.go /START1/,/STOP1/ * concurrency .play ./pingpong.go /STARTMAIN1/,/STOPMAIN1/ .link http://talks.golang.org/2013/advconc.slide Sameer Ajmani - Advanced Concurrency * Marshalling .code ./marshal.go /START1/,/STOP1/ .play ./marshal.go /START2/,/STOP2/ * Marshalling .code ./marshal_xml.go /START1/,/STOP1/ .play ./marshal_xml.go /START2/,/STOP2/ * .image ./kanye_imma_bookmarklet.png 320 _ - easy learning curve - fast compiles - `go get` - cross-compiles * Bad * # they come along, and affect how you do your work .image ./cats20.gif * Packaging Addresses different concern than distributions * lack of generics? # i don't feel strongly about this, though many do - interfaces - good enough for most - go1.4 introduced go:generate # produce code for Set, Graph etc, for the types needed, but at compile time. No need to reflect. * Debugging - gdb is there, sort of - some known debugging tools for ELF are not useful - fmt.Printf("%#v\n", ...) - go1.5 has much focus on DWARF symbols * Concurrency and GC - Calls like setns(2) are rough (yes, even with runtime.LockOSThread()) - go1.5 will greatly improve garbage collections (performance and pauses) * Fork/Exec Not Separate, but together * Ugly? (maybe just new) * .image ./revenge-of-the-nerds-o.gif * build tags no #ifdef .code ./tags.go /START1/,/STOP1/ or files with *_linux.go like suffix. More like extern. * pointer reciever .play ./ugly2.go /START1/,/STOP1/ * _ bit bucket * channels .play ./ugly0.go /START1/,/STOP1/ * iota .code ./ugly1.go /START1/,/STOP1/ * for range array (or string) .play ./primitive1.go /START1/,/STOP1/ * for range map .play ./primitive2.go /START1/,/STOP1/ * for range channel (like an iterator) .play ./primitive3.go /START1/,/STOP1/ * for range channel .code ./primitive3.go /START2/,/STOP2/ * interfaces and marshalling .code ./marshal_ugly.go /START1/,/STOP1/ * Conclusions? * Conclusions? - easy to adopt - enjoyable to learn * use-case - get familiar enough to like and dislike it - like all languages, choose the one that aligns with your use-case - don't be afraid to try it out