package stack

import "errors"

type Stack []interface{}

func (s *Stack) Len() int {
	return len(*s)
}

func (s *Stack) Cap() int {
	return cap(*s)
}

func (s *Stack) Push(x interface{}) *Stack {

	*s = append(*s, x)

	/**

	如果改为
	z := append(*s, x)
	s = &z

	1. 不将函数返回值赋给一个变量,而是直接对一个函数取地址:象这样:s = &(append(*s,x)) 是不允许的
	原因可能是:函数的返回值是一个在函数内部作用域的局部变量,在函数结束后如果没有传值到外部调用者的话作用域就会结束(同时空间被GC回收)
	对这样的局部变量取地址是不行的。
	如果返回值直接是一个指针(地址) 当然可以直接赋给外部变量,所以应该可以对返回指针的函数取值 如下
		func foo1() *int { x := 4; return &x }
		func foo2() int  { x := 4; return x }

		fmt.Println(*foo1()) //ok
		fmt.Println(&foo2()) //no



	2. 直接修改地址指向,在外部调
		s := stack.Stack{1, 2, 3}
		s.Push(4).Push(5).Push(6)
		fmt.Println(s)
	返回的仍然是 {1,2,3}

	原因是:方法的接受者虽然是指针,但方法内部只不过是将地址的值传值引用进来,
	在方法内部你可以利用这个传进来的地址值 对地址所在位置的内存进行修改,但你如果改变地址指向本身
	却是对外部没有影响的,总之从这点上看,以前学C语言时有个高人说过:
	“本质上,函数其实都是传值调用的,只不过有时传递是变量值,有时传的是地址值”
	 */

	return s
}

func (s *Stack) Top() (interface{}, error) {

	n := s.Len()
	if n == 0 {
		return nil, errors.New("can not get top from an empty stack")
	}

	return (*s)[n-1], nil
}

func (s *Stack) Pop() (interface{}, error) {

	n := s.Len()
	if n == 0 {
		return nil, errors.New("can not pop an empty stack")
	}

	top := (*s)[n-1]
	*s = (*s)[:n-1]

	return top, nil
}

  

posted on 2018-09-01 12:02  scala  阅读(162)  评论(0编辑  收藏  举报