Record some of the data types in GO.

An array of

Type [n]T is an array with n values of type T.

Expressions:

var a [10]int
Copy the code

Define variable A to be an array of ten integers.

The length of an array is part of its type, so arrays cannot be resized.

var a [2]string
	a[0] = "Hello"
	a[1] = "World"

b := [6]int{2.3.5.7.11.13}		// Array declaration and assignment
Copy the code

slice

Slice the slice

[]T is a slice of element type T.

A slice points to a sequence of values and contains length information.

In human terms, Slice is similar to Java’s ArrayList, which is a dynamic array.

Unlike Java’s ArrayList, which maintains an array internally, Slice can either maintain an array internally or create slices on top of an existing array.

Each expanded slice points to an expanded array.

nil slice

The zero value of slice is nil.

A nil slice has a length and a capacity of 0.

Tectonic slice


/* * Slicing does not store any data, it only describes part of the underlying array. Changing an element of a slice modifies the corresponding element of its underlying array. Other slices that share the same underlying array will see these changes. * Slicing is like referencing an array. * /
     / / array
	names := [5]string{
		"0_John"."1_Paul"."2_George"."3_Ringo",}// Construct a slice of the Names array, [n:m] indicating which part to refer to
    slice := names[0:]
    
Slice is created by the make function. This allocates a zero-length array and returns a slice pointing to the array:

a := make([]int.5)  // len(a)=5
// To specify capacity, pass a third argument to 'make' :

b := make([]int.0.5) // len(b)=0, cap(b)=5

b = b[:cap(b)] // len(b)=5, cap(b)=5

b = b[1:]      // len(b)=4, cap(b)=4
Copy the code

slice [n:m]usage

  • Slice [n:m] represents slice elements from n to m-1, including both ends.

  • Slice [n:n] is empty, i.e. Nil.

  • Slice [n:n+1] has one element.

  • Slice [:m] represents slice elements from 0 to m-1, including both ends.

  • Slice [n:] Indicates the slice element from N to len(slice), including both ends.

Add elements to Slice

/* * It is common to attach new elements to a slice, so Go provides a built-in append function. Func append (s T, vs.T) the first argument s to T is a slice of type T, and the rest are T values attached to the slice. * The result of append is a slice containing all the elements of the original slice and the newly supplied values. If the backup array of S is too small to hold all the given values, a larger array is allocated. The returned slice points to the newly allocated array. * /
    slice = append(slice, 2.3.4)

    /*slice1 is another slice, * must be followed by... , * to place the element in the array to which the slice points into append */
    slice = append(slice,slice1...)
Copy the code

map

Map Maps keys to values.

A map must be created before it can be used; A map with a value of nil is empty and cannot be assigned.

Map variable declaration

//map[key type] The type of the value
var m map[string]int
Copy the code

The map constructed

/* * is key-value pair, map zero is nil. Nil maps have no keys and can't add keys. The make function returns a map of the specified type, initialized and ready to use. Make (map[key type] value type) */
m = make(map[string]int)
    
var ma = map[string]int{
	"Bell Labs":1."Google": 2,}// If the top-level type is just a type name, it can be omitted.
// omit Ver(struct).
var mapa = map[string]Ver{
	"Bell Labs": {40.68433.74.39967},
	"Google":    {37.42202.122.08408}},Copy the code

usage

// add a key:value pair.
// if key exists, override value m[key]= value
    m["Bell Labs"] = 1
    
V indicates the value of the key. Ok indicates whether the key exists. If yes, the value is true.
// Otherwise false. If key does not exist, v is the zero value of its type.
v, ok := m["Answer"]

/ / remove the key: value
delete(m, "Answer")
Copy the code

iota

Enumeration usage of Go

//1, ioTA constant automatic generator, every other line, automatically accumulate 1
   //2, ioTA is used to assign constants
   const (
   	a = iota / / 0
   	b = iota / / 1
   	c = iota / / 2
   )
   //3, ioTA encounters const, reset to 0
   const d = iota / / 0
   // You can write only one ioTA
   const (
   	e = iota / / 0
   	f        / / 1
   	g        / / 2
   )
   //5, if it's on the same line, it's the same value
   const (
   	h = iota / / 0
   	i,j,k = iota.iota.iota / / 1,1,1
   	l=iota  / / 2
   )
Copy the code

Feel this is a Go bug, if someone passes a numerical type in gg.

error

Error is an interface.

The definitions in the library are as follows:

// The error built-in interface type is the conventional interface for
// representing an error condition, with the nil value representing no error.
type error interface {
	Error() string
}
Copy the code

Any time you implement this interface, you’re implementing error.

Many times you will need to receive an error in a function’s multiple return values to determine if there is an error.

Error in Go is basically a string, used to Java exception mechanism, really not used to this.

Pointer to the

Pointers in GO don’t have pointer operations like C/C ++.

Pointer usage: * variable = value of variable & variable = memory address of variable

i, j := 42.2701

	p := &i         // point to I

	fmt.Println(*p) // Read I through the pointer
	fmt.Println(&p) // memory address of p
	fmt.Println(&i) // The memory address of I

	*p = 21         // Set I through the pointer Sets I
	fmt.Println(i)  // See the new value of I

	p = &j         // point to j
	*p = *p / 37   // Divide j through the pointer
    fmt.Println(j) // see the new value of j
Copy the code

function

Functions are also values. They can be passed like any other value. Function values can be used as function parameters and return values.

// The parameter to compute is a function,
// This function is named fn and its type is float64,
// Compute returns float64.
func compute(fn func(float64.float64) float64) float64 {
	return fn(3.4)}Copy the code
Hypot is a function defined with two parameters
// And a return value of both types float64. X times x plus y times y, take the square root
	hypot := func(x, y float64) float64 {
		return math.Sqrt(x*x + y*y)
	}
    fmt.Println(hypot(5.12))  // 5 times 5 plus 12 times 12

	fmt.Println(compute(hypot))		// Pass hypot to function compute. 3 times 3 plus 4 times 4 is 25, so I take the square root, and I get 5.
	fmt.Println(compute(math.Pow))	// The built-in function Pow(x, y float64) computes x to the y, 3 to the 4th and results in 81.
Copy the code

closure

 // The return value of this function is an anonymous function that returns a function of type func(int) int
func adder(a) func(int) int {
	sum := 0
	return func(x int) int {
		sum += x
		return sum
	}
}

/* * returns an anonymous function, returns a function type, calls the returned anonymous function through f, calls the closure function. * It doesn't care if the captured variables and constants are out of scope, as long as the closure uses them. * /
	f:=adder()
	fmt.Println(f(1))	1 / / results
	fmt.Println(f(2))	//结果3
	fmt.Println(f(3))	6 / / results
Copy the code

interface

Chapter 11: Interface and Reflection

  • The Go interface is similar to Java in that it declares methods that constrain the behavior of classes (structures) that implement the interface.

  • Go’s interface does not need to display inheritance, as long as the structure implements the methods declared by the interface, it implements the interface.

  • Interfaces in Go can nest methods declared by other interfaces into the interface by composition.

define

type Namer interface {
    Method1(param_list) return_type
    Method2(param_list) return_type
    ...
}

// Nested interfaces
type Interface interface {
	Len() int
	Less(i, j int) bool
	Swap(i, j int)}type Interface interface {
	sort.Interface
	Push(x interface{}) 
	Pop() interface{}}/ / equal to
type Interface interface {
	Len() int
	Less(i, j int) bool
	Swap(i, j int)
	Push(x interface{}) 
	Pop() interface{}}Copy the code

Namer above is an interface type.

Interface names (which by convention contain only one method) consist of method names with [e]r suffixes, such as Printer, Reader, Writer, Logger, Converter, and so on. There are also less common ways (when the suffix er is inappropriate), such as Recoverable, where the interface name ends in able or begins with I (as in.NET or Java).

Interfaces in the Go language are short, usually containing zero or at most three methods.

Unlike most object-oriented programming languages, in Go an interface can have a value, a variable of an interface type or an interface value: var ai Namer, ai is a multiword data structure whose value is nil. It’s essentially a pointer, though not quite the same thing. Pointers to interface values are illegal; not only are they useless, they cause code errors.

The interface type

An interface is also a type, so variables of that type can be assigned to implementations of that interface.

package main

import "fmt"

type Shaper interface {
    Area() float32
}

type Square struct {
    side float32
}

func (sq *Square) Area(a) float32 {
    return sq.side * sq.side
}

func main(a) {
    sq1 := new(Square)
    sq1.side = 5

    // var areaIntf Shaper
    // areaIntf = sq1
    // shorter,without separate declaration:
    // areaIntf := Shaper(sq1)
    // or even:
    areaIntf := sq1
    fmt.Printf("The square has area: %f\n", areaIntf.Area())
}
Copy the code

Types of assertions

The type in an interface variable can be any type of value. If you need to determine the type at run time, you can use type assertion.

/* * T is the type you need to assert, * when ok is true, v is converted to T, * when ok is false, v is T zero. * /
if v, ok := varI.(T); ok {  // checked type assertion
    Process(v)
    return
}
// varI is not of type T
Copy the code

Type check: type-switch

The type of an interface variable can also be detected using a special form of swtich: type-swtich

switch t := areaIntf.(type) {
case *Square:
    fmt.Printf("Type Square %T with value %v\n", t, t)
case *Circle:
    fmt.Printf("Type Circle %T with value %v\n", t, t)
case nil:
    fmt.Printf("nil value: nothing to check? \n")
default:
    fmt.Printf("Unexpected type %T\n", t)
}
Copy the code

Type-switch does not allow fallthrough.

Empty interface

An empty interface does not declare any methods. Similar to the Object class in Java, any type in Go implements a null interface.

The usage is similar to that of Object in Java. After all, there are no generics in Go now. I hope this feature will be officially added soon.

The structure of the body

Chapter 10: Structs and Methods

Go has no concept of classes, and certainly no concept of inheritance.

Fortunately, there are interfaces, so you can use a structure + interface + method + combination to implement OOP.

Defining structure

/* * structure definition */
type Vertex struct {
	X int
	Y int
}
// Structure with tag
type TagType struct { // tags
    field1 bool   "An important answer"
    field2 string "The name of the thing"
    field3 int    "How much there are"
}
Copy the code

Tag: A string attached to a field that can be a document or other important tag.

The contents of the tag cannot be used in normal programming; only the package Reflect retrieves it.

It can introspect types, properties, and methods at run time.

For example, calling reflect.typeof () on a variable gets the correct TypeOf the variable.

If the variable is a structure type, the structure’s fields can be indexed by Field, and then the Tag attribute can be used.

Anonymous fields and embedded structures

A structure can contain one or more anonymous (or embedded) fields.

That is, these fields do not have an explicit name, only the type of the field is required, in which case the type is the name of the field.

An anonymous field can itself be a structure type, that is, a structure can contain an embedded structure.

You can roughly compare this to the concept of inheritance in object-oriented languages, and you’ll see later that it’s used to simulate inheritance-like behavior.

Inheritance in Go is implemented through embedding or composition, so it can be said that composition is favored over inheritance in Go.

Example 1:

package main

import "fmt"

type innerS struct {
    in1 int
    in2 int
}

type outerS struct {
    b    int
    c    float32
    int  // anonymous field
    innerS //anonymous field
}

func main(a) {
    outer := new(outerS)
    outer.b = 6
    outer.c = 7.5
    outer.int = 60
    outer.in1 = 5
    outer.in2 = 10

    fmt.Printf("outer.b is: %d\n", outer.b)
    fmt.Printf("outer.c is: %f\n", outer.c)
    fmt.Printf("outer.int is: %d\n", outer.int)
    fmt.Printf("outer.in1 is: %d\n", outer.in1)
    fmt.Printf("outer.in2 is: %d\n", outer.in2)

    // Use structure literals
    outer2 := outerS{6.7.5.60, innerS{5.10}}
    fmt.Printf("outer2 is:", outer2)
}
Copy the code

Output:

outer.b is: 6
outer.c is: 7.500000
outer.int is: 60
outer.in1 is: 5
outer.in2 is: 10
outer2 is:{6 7.5 60 {5 10}}
Copy the code

Example 2:

package main

import "fmt"

type A struct {
    ax, ay int
}

type B struct {
    A
    bx, by float32
}

func main(a) {
    b := B{A{1.2}, 3.0.4.0}
    fmt.Println(b.ax, b.ay, b.bx, b.by)
    fmt.Println(b.A)
}
Copy the code

Output:

1 2 3 4
{1 2}
Copy the code

Retrieve data stored in anonymous fields by the name of type outer. Int,

It follows that there can only be one anonymous field for each data type in a structure.

methods

Generally, there are two expressions in Go: function and method, which are essentially the same, but the method is a function belonging to a structure and can only be used by the variables of the structure. To call.

The definition is as follows:

func (recv struct_type) methodName(parameter_list) (return_value_list){... }Copy the code

The receiver is specified in parentheses before the method name and after the func keyword.

If recv is an instance of struct_type and Method1 is its method name, the method call follows the traditional object.name selector notation: recv.method1 ().

If recv is a pointer, Go is automatically dereferenced.

If the method does not need to use the value of recv, it can be replaced with _, as in:

func (_ struct_type) methodName(parameter_list) (return_value_list){... }Copy the code

Recv is like this or self in object-oriented languages, but not in Go.

You can use this or self as the name of the receiver.

Use structure

v1.X = 4				// Set the value of the structure variable
fmt.Println("Line 2:",v1.X)	// Read the value of the structure variable

p := &v1		//p points to the memory address of v1
p.X = 1e9		// reassign the X variable of p
fmt.Println("Line 3:",v1)	// Let's look at the new value of v1
Copy the code

container

The container data type provided by Go, in the Container package.

heap

Heap structure

Go only defines the interface, not the implementation.

type Interface interface {
	sort.Interface
	Push(x interface{}) // add x as element Len()
	Pop() interface{}   // remove and return element Len() - 1.
}
Copy the code

If you want to use a heap, simply implement the following five methods to implement the interface.

    Len() int
	// Less reports whether the element with
	// index i should sort before the element with index j.
	Less(i, j int) bool
	// Swap swaps the elements with indexes i and j.
    Swap(i, j int)
    Push(x interface{}) // add x as element Len()
	Pop() interface{}   // remove and return element Len() - 1.
Copy the code

list

Two-way linked list

The definitions in the library are as follows:

// List represents a doubly linked list.
// The zero value for List is an empty list ready to use.
type List struct {
	root Element // sentinel list element, only &root, root.prev, and root.next are used
	len  int     // current list length excluding (this) sentinel element
}
Copy the code
// Element is an element of a linked list.
type Element struct {
	// Next and previous pointers in the doubly-linked list of elements.
	// To simplify the implementation, internally a list l is implemented
	// as a ring, such that &l.root is both the next element of the last
	// list element (l.Back()) and the previous element of the first list
	// element (l.Front()).
	next, prev *Element

	// The list to which this element belongs.
	list *List

	// The value stored with this element.
	Value interface{}}Copy the code

The corresponding methods of list are:

type Element
    func (e *Element) Next(a) *Element
    func (e *Element) Prev(a) *Element
type List
    func New(a) *List
    func (l *List) Back(a) *Element// The last elementfunc (l *List) Front(a) *Element// The first elementfunc (l *List) Init(a) *List// List initializationfunc (l *List) InsertAfter(v interface{}, mark *Element) *Element// Insert after an elementfunc (l *List) InsertBefore(v interface{}, mark *Element) *Element// Insert before an elementfunc (l *List) Len(a) int// In the list lengthfunc (l *List) MoveAfter(e, mark *Element)/ / theeElement moves tomarkafterfunc (l *List) MoveBefore(e, mark *Element)/ / theeElement moves tomarkbeforefunc (l *List) MoveToBack(e *Element)/ / theeThe element moves to the end of the queuefunc (l *List) MoveToFront(e *Element)/ / theeElement moves to the top of the queuefunc (l *List) PushBack(v interface{}) *Element// Insert the element at the end of the queuefunc (l *List) PushBackList(other *List)// Insert a new queue at the end of the queuefunc (l *List) PushFront(v interface{}) *Element// Insert elements at the head of the queuefunc (l *List) PushFrontList(other *List)// Insert a new queue into the queue headfunc (l *List) Remove(e *Element) interface{} // Delete an element
Copy the code

ring

Bidirectional circular linked lists

Structure diagram of bidirectional lists and bidirectional cyclic lists:

The definitions in the library are as follows:

// A Ring is an element of a circular list, or ring.
// Rings do not have a beginning or end; a pointer to any ring element
// serves as reference to the entire ring. Empty rings are represented
// as nil Ring pointers. The zero value for a Ring is a one-element
// ring with a nil Value.
//
type Ring struct {
	next, prev *Ring
	Value      interface{} // for use by client; untouched by this library
}
Copy the code

The structure of the ring is a bit peculiar, with the tail of the ring being the head, so each element can actually represent its own ring. It doesn’t need to keep both list and Element like a list, it just needs to keep one.

Methods provided by Ring include:

type Ring
    func New(n int) *Ring// Initialize the ringfunc (r *Ring) Do(f func(interface{})) // Loop loop to operatefunc (r *Ring) Len(a) int/ / loop lengthfunc (r *Ring) Link(s *Ring) *Ring// Connect two ringsfunc (r *Ring) Move(n int) *Ring// The pointer moves backwards or forwards from the current element (nIt can be negative.func (r *Ring) Next(a) *Ring// The next element of the current elementfunc (r *Ring) Prev(a) *Ring// The last element of the current elementfunc (r *Ring) Unlink(n int) *Ring// Start with the current element and deletenAn elementCopy the code

Type aliases & type definitions

References fly snow ruthless blog

define

type D = int   // Type alias
type I int    // Type declaration
Copy the code

The type aliases are=Number, the type declaration does not.

Type the alias

A type alias is simply an extra name for the original type. The original type is basically used as it should be (unless you change the exportability with a type alias).

The main purpose of this feature is to be compatible with the movement of defined types between packages. For example, we have an export type flysnow.org/lib/T1 that we now want to migrate to another package flysnow.org/lib2/T1.

If we do this without the Type alias, it will cause other third parties to reference the code of the old package path. We need to modify it uniformly, otherwise it will not be used.

With a type alias, the implementation of type T1 can be migrated to lib2.

At the same time, we define an alias for T1 under lib2 in the original lib,

This allows third-party references to be used without modification,

It only needs to be compatible for a period of time, and then completely remove the type compatibility in the old package.

This allows us to refactor our code incrementally, rather than in a one-size-fits-all fashion.

The type definition

A type definition is the creation of a new type based on the original type. The new type and the original type are two types.

This feature is mostly used when you want to add a new method to type A, but type A is not a native type, so you cannot add it.

At this point, you can use the type definition to define it as type B, so you can add new methods.