Go 语言 数组(array)与数组切片( Slice)

数组
Go语言中的数组是定长的同一类型数据的集合,数组索引是从0开始的。
数组有以下几种创建方式

// 声明一个叫 balance的 10个元素的float32 数组
var balance [10] float32 
//数组初始化
var balance = [5]float32{1000.0, 2000.0, 3000.4, 7000.0, 5000.0} 
// 忽略 [] 中的数字不设置数组大小,Go 语言会根据元素的个数自动设置数组的大小
var balance = [...]float32{1000.0, 2000.0, 3000.4, 7000.0, 5000.0}

以下是一些特殊数组

[2*N] struct {x,y int32} //复杂类型数组
[1000]* float64  //指针数组
[3][5]int  //二维数组
[2][3][5]float64 // 等同于 [2]([3][5]float64)

当创建数组时,若没有被显示的初始化或者只是部分初始化,那么Go语言会自动的把数组其他的项都初始化为0(元素类型默认值)

获取数组长度,使用len函数;获取数组容量大小,使用cap函数,由于数组长度不可变,因此数组的容量等于长度。

len(arr) == cap(arr)

访问数组
使用 len 遍历

for i := 0; i < len(arr); i++ {
   fmt.Printf("%c", arr[i])
}

使用 range 遍历,有两个返回值,第一个是 元素的数组下标,第二个是元素的值

for _, v := range arr{
      fmt.Printf("%c", v)
 }

数组是值传递,因此在函数内操作数组只是数组的一个副本,不会影响数组本身,但可以通过传递指针来避免值传递。

func main() {
   array := [5]int{1,2,3,4,5}
   modify(array)
   fmt.Println("In main, array values:",array)
}

func modify(array [5]int)  {
   array[0] = 10
   fmt.Println("In modify, array values:",array)
}
------------output-----------
In modify, array values: [10 2 3 4 5]
In main, array values: [1 2 3 4 5]

数组切片
是引用类型,可以自动扩容但容量固定,弥补数组的长度在定义后无法再次修改,在函数体内无法对外部的数组内部结构进行修改的缺点。
数组切片的数据结构可以抽象为以下3个变量:
①一个指向原生数组的指针
②数组切片中的元素个数
③数组切片已分配的存储空间

数组与切片都可以使用下面所给出的语法进行切片

s[n]                     //切片s中索引为n的项
s[n:m]                 //从切片s的索引位置 n 到 m-1 处所获得的切片                 
s[n:]                    //从切片s的索引位置 n 到len(s)-1处所获得的切片
s[:m]                   //从切片s的索引位置 0 到 m-1 处所获得的切片
s[:]                      //切片s的索引位置0到len(s)-1处所获得的切片
cap(s)                 //获得切片的容量:总是>= len(s)
len(s)                  //获得切片包含元素的个数:总是<= cap(s) 
s[:cap(s)]            //增加切片s的长度到其容量,如果长度小于等于容量的话

注:s == s[:n]+s[n:] //s是一个字符串,n为整型,0<=n<=len(s)

创建数组切片有以下几种方式
①基于数组创建数组切片

var myArray [10]int = [10]int{1,2,3,4,5}
var mySlice []int = myArray[:5] //前五个元素创建数组切片
var mySlice []int = myArray[:] // 所有元素创建数组切片

②直接创建数组切片

//创建元素初始值为0,比如下面初始元素个数即长度为5(必须设定)
//预留10个元素的存储容量(可以不设定,默认跟初始元素相等),空间容量大于等于初始元素个数
mySlice :=make([]int,5,10) 
mySlice := []int{1,2,3,4,5}

③基于数组切片创建数组切片(指向同一个隐藏数组)

func main() {
   mySlice1 := []int{1,2,3,4,5} //容量与长度相同
   mySlice2 := mySlice1[:3]
   fmt.Println(mySlice2)
}

只要mySlice2选择的范围 mySlice1[:n] 这个n不超过 cap(mySlice1) 的值即可,自动补充0

当创建一个切片时,它会创建一个隐藏的初始化为零值的数组,然后返回引用该隐藏数组的切片。该隐藏数组也是固定长度的,该长度始终等于切片的容量。比如下图所示的切片x,基于切片x创建的切片y,隐藏数组。

数组与数组切片.png

由于数组切片是一个引用类型,因此若有多个指向同一个隐藏数组的切片中的某一个进行修改操作,那么其他切片都会受影响

func main() {
   var  mySlice1 = []int{1,2,3,4,5,6,7}
   mySlice2 := mySlice1[:5]
   mySlice3 := mySlice1[3:]
   fmt.Println(mySlice1,mySlice2,mySlice3)
   mySlice3[0] = 100
   fmt.Println(mySlice1,mySlice2,mySlice3)
}
-----output-----
[1 2 3 4 5 6 7] [1 2 3 4 5] [4 5 6 7]
[1 2 3 100 5 6 7] [1 2 3 100 5] [100 5 6 7]

空数组切片
一个数组切片未被初始化时,默认为nil,存储空间长度为0,元素个数为0
len(): 返回当前切片存储的元素个数 cap(): 返回数组切片分配的存储空间

var numbers []int // len=0 cap=0 slice=[]

动态增减元素,合理的设置数组切片的存储空间,将会减少切片内部重新分配内存和搬送内存块的频率,提高性能。

接下来我们看一个创建切片综合运用实例

func main() {
   /* 创建切片 */
   numbers := []int{0,1,2,3,4,5,6,7,8}   
   printSlice(numbers) // len=9 cap=9 slice=[0 1 2 3 4 5 6 7 8]

   /* 打印原始切片 */
   fmt.Println("numbers ==", numbers) //numbers == [0 1 2 3 4 5 6 7 8]

   /* 打印子切片从索引1(包含) 到索引4(不包含)*/
   fmt.Println("numbers[1:4] ==", numbers[1:4]) //numbers[1:4] == [1 2 3]

   /* 默认下限为 0*/
   fmt.Println("numbers[:3] ==", numbers[:3]) //numbers[:3] == [0 1 2]

   /* 默认上限为 len(s)*/
   fmt.Println("numbers[4:] ==", numbers[4:]) //numbers[4:] == [4 5 6 7 8]

   numbers1 := make([]int,0,5)
   printSlice(numbers1)  //len=0 cap=5 slice=[]

   /* 打印子切片从索引  0(包含) 到索引 2(不包含) */
   number2 := numbers[:2]
   printSlice(number2) //  len=2 cap=9 slice=[0 1]

   /* 打印子切片从索引 2(包含) 到索引 5(不包含) */
   number3 := numbers[2:5]
   printSlice(number3) //len=3 cap=7 slice=[2 3 4]

}

func printSlice(x []int){
   fmt.Printf("len=%d cap=%d slice=%v\n",len(x),cap(x),x)
}

总结:基于一个数组切片a通过s[n:m]方式创建一个新的子切片b,那么子切片b的容量等于a切片容量减去n

向数组切片添加元素
切片不支持+=操作,所以要继续增加存储的元素,可以使用append 函数。append 函数可以直接将一个数组切片加到当前数组切片的后面,记住添加的数组切片后边必须加三个点,相当于把mySlice1的所有元素打散后传入mySlice,但是传入的mySlice1所有元素顺序不变,使用append函数的两个数组切片的元素类型必须是相同的。若原始切片的容量不足够容纳原始元素和新添加进来的元素,那么append 函数将会隐式的创建一个新的切片,并将原始元素与新元素都添加进来。
//numbers指切片,n指元素,这个n可以为空,那么依旧为原数组切片
numbers = append(numbers, n)

//示例一
func main() {
   var  mySlice1 = make([]int,5,10)
   fmt.Printf("mySlice1:%v, len(mySlice1):%v, cap(mySlice1):%v\n",mySlice1,len(mySlice1),cap(mySlice1))
   mySlice1 = append(mySlice1,1,2,3)
   fmt.Printf("mySlice1:%v, len(mySlice1):%v, cap(mySlice1):%v\n",mySlice1,len(mySlice1),cap(mySlice1))
   mySlice2 := []int{4,5,6}
   mySlice1 = append(mySlice1,mySlice2...)
   fmt.Printf("mySlice1:%v, len(mySlice1):%v, cap(mySlice1):%v\n",mySlice1,len(mySlice1),cap(mySlice1))
}
------output------
mySlice1:[0 0 0 0 0], len(mySlice1):5, cap(mySlice1):10
mySlice1:[0 0 0 0 0 1 2 3], len(mySlice1):8, cap(mySlice1):10
mySlice1:[0 0 0 0 0 1 2 3 4 5 6], len(mySlice1):11, cap(mySlice1):20

之前说切片的底层实现是通过共享数组的方式实现的,append在进行添加元素时,会首先检查原切片的可用容量,也就是底层共享数组的长度是否满足,如果底层数组长度不够,那么就会分配一个新的数组,将被引用的所有的值复制到新数组当中,再继续添加新元素。

如果是新切片进行append添加新元素,那么原切片的容量与长度都不会改变,哪怕新切片扩容超过原切片的容量

//示例二
func main() {
   a := []int{1, 2, 3, 4, 5}
   b := a[2:3]
   fmt.Println("a: ", a, " len: ", len(a), " cap: ", cap(a))
   fmt.Println("b: ", b, " len: ", len(b), " cap: ", cap(b))

   b = append(b, 6)
   fmt.Println("a: ", a, " len: ", len(a), " cap: ", cap(a))
   fmt.Println("b: ", b, " len: ", len(b), " cap: ", cap(b))
}
---output---
a:  [1 2 3 4 5]  len:  5  cap:  5
b:  [3]  len:  1  cap:  3
a:  [1 2 3 6 5]  len:  5  cap:  5
b:  [3 6]  len:  2  cap:  3

在上面的示例中使用newSlice = Slice[n:m],新切片的容量会随着旧切片走。如果使用索引参数就可以来指定新切片的容量

//示例三
a := []int{1, 2, 3, 4, 5}
c := a[2:3:4] //注:a[i:j:k]:容量cap = k-i,长度len = j-i,因为a的容量为5,因此这里k的值最大不能大于5,k< cap(a) 
fmt.Println("c: ", c, " len: ", len(c), " cap: ", cap(c))
----output----
c:  [3]  len:  1  cap:  2

若将k的值大于cap(a),也就是设定新切片容量大于旧切片容量时,那么就会报运行时错误,这个很难找到错误原因。

c := a[2:3:6]
---------------------------------------------------------
panic: runtime error: slice bounds out of range

goroutine 1 [running]:
main.main()
    D:/GoDemo/src/MyGo/Demo_05.go:17 +0x74a

解决方法:如果在创建新切片时,设定长度与容量一致,新切片进行append操作时,会强制在底层立即创建新的数组,这就跟原切片在底层上不是共享同一数组,这样就可以安全的操作新切片,将上面的示例二中 b := a[2:3] 改为 b := a[2:3:3] 进行结果对比

//设置新切片长度与容量都相等的情况
a := []int{1, 2, 3, 4, 5}
b := a[2:3:3] //只取出原切片索引为2的值,并设定新切片容量与长度都为1
fmt.Println("a: ", a, " len: ", len(a), " cap: ", cap(a))
fmt.Println("b: ", b, " len: ", len(b), " cap: ", cap(b))

//新切片添加新元素操作
b = append(b,6 )
fmt.Println("a: ", a, " len: ", len(a), " cap: ", cap(a))
fmt.Println("b: ", b, " len: ", len(b), " cap: ", cap(b))

---output---
a:  [1 2 3 4 5]  len:  5  cap:  5
b:  [3]  len:  1  cap:  1
a:  [1 2 3 4 5]  len:  5  cap:  5
b:  [3 6]  len:  2  cap:  2

内容复制
使用Go语言另一个内置函数copy函数,它接受两个包含相同类型元素的切片,并将源切片的元素复制到目标切片,同时返回所复制元素的数量。若这两个切片不是一样大,那么自动按照较小的数组切片的元素个数进行复制。这个复制只是复制数值,原切片索引值改变,不会影响到新的切片

func main() {
   mySlice1 := []int{1,2,3,4,5}
   mySlice2 := []int{6,7,8}
   copy(mySlice1,mySlice2) //将mySlice2 复制到 mySlice1 的前3个位置
   fmt.Println(mySlice1) //  [6 7 8 4 5]
  
   copy(mySlice2,mySlice1) //因为mySlice2的长度比mySlice1小,反过来就是将 mySlice1 的前3个元素复制到mySlice2
   fmt.Println(mySlice2) // [1 2 3]

   mySlice1[0] = 100
   fmt.Println(mySlice1) //[100 2 3 4 5]
   fmt.Println(mySlice2) //[1 2 3]
}

切片迭代
同样可以使用range配合for循环迭代输出切片中的元素,但是要注意这里的range迭代输出的两个值:第一个值是当前迭代的索引位置,第二个值是该位置对应元素值的副本。

func main() {
   slice := []int{10, 20, 30, 40}
   // 迭代每一个元素,并显示其值
   for index, value := range slice {
      fmt.Printf("Index: %d Value: %d\n", index, value)
   }
}
---output---
Index: 0 Value: 10
Index: 1 Value: 20
Index: 2 Value: 30
Index: 3 Value: 40

range迭代切片时,会返回当前迭代的索引位置与该位置对应元素值的副本,range为每个元素都创建了副本,而不是直接返回对该元素的引用。

使用 range 迭代切片会创建每个元素的副本.png

如果使用返回value值的地址作为指向每个元素的指针,就会造成错误,因为每个value值的地址都是相同的,无法区分.

func main() {
   slice := []int{10, 20, 30, 40}
   // 迭代每一个元素,并显示其值
   for index, value := range slice {
      fmt.Printf("Value: %d Value-Addr: %X ElemAddr: %X\n",
         value, &value, &slice[index])
   }
}
----output----
Value: 10 Value-Addr: C042060080 ElemAddr: C04205E0C0
Value: 20 Value-Addr: C042060080 ElemAddr: C04205E0C8
Value: 30 Value-Addr: C042060080 ElemAddr: C04205E0D0
Value: 40 Value-Addr: C042060080 ElemAddr: C04205E0D8

结论:每次迭代返回的变量value值其实是在迭代过程中据切片依次赋值的新变量,不是切片中原来的值了,因此如果需要求得每个元素的地址,还是使用 &slice[index] 的方式

切片操作实战
①插入元素。先保存后面的元素,再取前面的元素添加元素,再组合起来

第一种:利用append函数
func main() {
   var name = []int{1,2,3,4,5,6,7,8,9,10}
   index := 5
   insertSlice := []int{1000}
   name1 := append([]int{},name[index:]...) //name1:  [6 7 8 9 10]
   name2 := append(name[:index],insertSlice...) //name2:  [1 2 3 4 5 1000]
   name2 = append(name2,name1...)
   fmt.Println(name2) //name2:  [1 2 3 4 5 1000 6 7 8 9 10]
}

上面的方法可以再简化如下
func main() {
   var name = []int{1,2,3,4,5,6,7,8,9,10}
   index := 5
   insertSlice := []int{1000}
   name = append(name[:index],append(insertSlice, name[index:]...)...)
   fmt.Println(name) //name2:  [1 2 3 4 5 1000 6 7 8 9 10]
}

第二种:利用copy函数
func main() {
   var slice = []int{1,2,3,4,5,6,7,8,9,10}
   insertSlice := []int{1000}
   index := 5 //插入的切片索引
   //根据原始切片与新切片创建新的切片
   result := make([]int,len(slice)+len(insertSlice))
   at := copy(result,slice[:index])
   at += copy(result[at:],insertSlice)
   copy(result[at:],slice[index:])
   fmt.Println(result) // result:[1 2 3 4 5 1000 6 7 8 9 10]
}

②删除元素。

从开头删除某个索引处的元素
func main() {
   var name = []int{1,2,3,4,5,6,7,8,9,10}
   name = name[1:]
}

从结尾删除某个索引处的元素
func main() {
   var name = []int{1,2,3,4,5,6,7,8,9,10}
   name = name[:9]
}

从中间删除某个索引处的元素
func main()  {
   var name = []int{1,2,3,4,5,6,7,8,9,10}
   a := len(name)/2
   name = append(name[:a],name[a+1:]...) //name:  [1 2 3 4 5 7 8 9 10]
}

删除某一个索引区间的全部元素
func main() {
   var name = []int{1,2,3,4,5,6,7,8,9,10}//name:  [1 2 3 4 5 6 7 8 9 10]
   start := 1
   end := 5
   name = append(name[:start],name[end:]...) //name: [1 6 7 8 9 10]
}

使用copy同样能达到目的

③切片尾部追加元素

func main()  {
   var name = []int{1,2,3,4,5,6,7,8,9,10}
   fmt.Println("name: ",name)
   //尾部追加元素
   for i:=11;i<=15 ;i++  {
      name = append(name,i)
   }
   fmt.Println("name: ",name)
}

append函数修改切片会改变原始切片,而copy函数修改不会。

关于切片的指针

推荐阅读:http://www.cnblogs.com/dajianshi/p/4235142.html?hmsr=studygolang.com&utm_medium=studygolang.com&utm_source=studygolang.com

①当我们用append追加元素到切片时,如果容量不够,go就会创建一个新的切片变量,看下面程序的执行结果:

func main() {
   var sa []int
   fmt.Printf("addr:%p \tlen:%v \tcontent:%v\n",sa,len(sa),sa);
   for i:=0;i<10;i++{
      sa=append(sa,i)
      fmt.Printf("addr:%p \tlen:%v \t content:%v\n",sa,len(sa),sa);
   }
   fmt.Printf("addr:%p \tlen:%v \t content:%v\n",sa,len(sa),sa);
}
------output-------
addr:0x0     len:0     content:[]
addr:0xc042060088     len:1        content:[0]
addr:0xc0420600c0     len:2        content:[0 1]
addr:0xc04205e0e0     len:3        content:[0 1 2]
addr:0xc04205e0e0     len:4        content:[0 1 2 3]
addr:0xc042084100     len:5        content:[0 1 2 3 4]
addr:0xc042084100     len:6        content:[0 1 2 3 4 5]
addr:0xc042084100     len:7        content:[0 1 2 3 4 5 6]
addr:0xc042084100     len:8        content:[0 1 2 3 4 5 6 7]
addr:0xc04208e000     len:9        content:[0 1 2 3 4 5 6 7 8]
addr:0xc04208e000     len:10       content:[0 1 2 3 4 5 6 7 8 9]
addr:0xc04208e000     len:10       content:[0 1 2 3 4 5 6 7 8 9]

因为初始时指定的切片容量不足,因此切片在进行append操作时,会自动扩容产生新的切片变量,因此切片变量地址会频繁变动

因此在不能预估切片的容量情况下,又要防止切片变量地址频繁变动,我们就需要使用指针来操作切片变量,其本质上是:append操作亦然会在需要的时候构造新的切片,不过是将地址都保存到了sa中,因此我们通过该指针始终可以访问到真正的数据。

func main() {
   var osa = make ([]int,0);
   sa:=&osa; 
   for i:=0;i<10;i++{
      *sa=append(*sa,i)
      fmt.Printf("addr of osa:%p,\taddr:%p \t content:%v\n",osa,sa,sa);
   }
   fmt.Printf("addr of osa:%p,\taddr:%p \t content:%v\n",osa,sa,sa);
}
-------output--------
addr of osa:0xc042060080,    addr:0xc04205a3e0      content:&[0]
addr of osa:0xc0420600b0,    addr:0xc04205a3e0      content:&[0 1]
addr of osa:0xc04205e0e0,    addr:0xc04205a3e0      content:&[0 1 2]
addr of osa:0xc04205e0e0,    addr:0xc04205a3e0      content:&[0 1 2 3]
addr of osa:0xc042084100,    addr:0xc04205a3e0      content:&[0 1 2 3 4]
addr of osa:0xc042084100,    addr:0xc04205a3e0      content:&[0 1 2 3 4 5]
addr of osa:0xc042084100,    addr:0xc04205a3e0      content:&[0 1 2 3 4 5 6]
addr of osa:0xc042084100,    addr:0xc04205a3e0      content:&[0 1 2 3 4 5 6 7]
addr of osa:0xc04208e080,    addr:0xc04205a3e0      content:&[0 1 2 3 4 5 6 7 8]
addr of osa:0xc04208e080,    addr:0xc04205a3e0      content:&[0 1 2 3 4 5 6 7 8 9]
addr of osa:0xc04208e080,    addr:0xc04205a3e0      content:&[0 1 2 3 4 5 6 7 8 9]
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,236评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,867评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,715评论 0 340
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,899评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,895评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,733评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,085评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,722评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,025评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,696评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,816评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,447评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,057评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,254评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,204评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,561评论 2 343

推荐阅读更多精彩内容

  • 出处---Go编程语言 欢迎来到 Go 编程语言指南。本指南涵盖了该语言的大部分重要特性 Go 语言的交互式简介,...
    Tuberose阅读 18,398评论 1 46
  • 1.安装 https://studygolang.com/dl 2.使用vscode编辑器安装go插件 3.go语...
    go含羞草阅读 1,538评论 0 6
  • 本文翻译自Rob Pike的文章《Arrays, slices (and strings): The mechan...
    大蟒传奇阅读 4,883评论 2 8
  • 克服害羞其实没有那么难,各种办法很多,但是说到底是在两个地方动刀。 1.认知上做改变:A.“没有那么多人关注我和在...
    不二侠阅读 280评论 0 0
  • 1、了解产品(客户需求、产品路线) 一款好的产品可以为用户解决问题,可以开拓用户的思维。 当用户使用一款好的产品时...
    小马哥志峰阅读 227评论 0 0