GoLong学习-Mac版Go环境配置和基础语法(一)

Hello World这是开发的一个开发魔咒,几乎所有人都是从这开始的,今天我也从魔咒开始
先打印一份Hello World ,Let's Go! ↓

package main

import "fmt"

func main() {// main函数

    fmt.Printf("Hello, world or 你好,世界 or καλημ ́ρα κóσμ or こんにちはせかい\n")
}
// 以上代码打印输出: Hello, world or 你好,世界 or καλημ ́ρα κóσμ or こんにちはせかい

Go概念:

Go程序是通过package来组织的,每一个可独立运行的Go程序,必定包含一个package main,在这个main包中必定包含一个入口函数main,而这个函数既没有参数,也没有返回值。

代码解析:

  • package main 这一行代码告诉我们当前文件属于哪个包,而包名main则告诉我们他是一个可独立运行的包,它在编译后会产生可执行文件
  • 为了打印Hello world...我们调用了一个Printf函数,这个函数来自fmt包,所以导入了fmt包(import "fmt")

结论:

最后大家可以看到我们输出的内容里面包含了很多非ASCII码字符。实际上,Go是天生支持UTF-8的,任何字符都可以直接输出,你甚至可以用UTF-8中的任何字符作为标识符
Go使用package来组织代码。main.main()函数(这个函数位于主包)是每一个独立的可运行程序的入口点。Go使用UTF-8字符串和标识符(因为UTF-8的发明者也就是Go的发明者之一),所以它天生支持多语言。

Mac环境开发GOLONG

环境配置:访问下载地址,32位系统下载go1.4.2.darwin-386-osx10.8.pkg(更新的版本已无32位下载),64位系统下载go1.8.3.darwin-amd64.pkg,双击下载文件,一路默认安装点击下一步,这个时候go已经安装到你的系统中,默认已经在PATH中增加了相应的~/go/bin,这个时候打开终端,输入go
看到类似上面源码安装成功的图片说明已经安装成功如果出现go的Usage信息,那么说明go已经安装成功了;如果出现该命令不存在,那么可以检查一下自己的PATH环境变中是否包含了go的安装目录。
几个查看Go环境的命令:

  • go version 查看go当前的版本
  • go env 查看当前go的环境变量
  • go list 列出当前全部安装的package
  • go run 编译并运行Go程序

Go开发工具,笔者选择的sublime,sublime3下载地址
sublime+GoSublime+GoCode一块使用,先安装package Control插件管理器:如下方法操作完重启,之后就可以开心安装想要的插件了:(安装插件方法:需安装GoSublime、SidebarEnhancements和Go Build,安装插件之后记得重启Sublime生效,Ctrl+Shift+p打开Package Controll 输入pcip(即“Package Control: Install Package”的缩写)

打开之后安装 Package Control:Ctrl+` 打开命令行,执行如下代码:
适用于 Sublime Text 3:
import  urllib.request,os;pf='Package Control.sublime-package';ipp=sublime.installed_packages_path();urllib.request.install_opener(urllib.request.build_opener(urllib.request.ProxyHandler()));open(os.path.join(ipp,pf),'wb').write(urllib.request.urlopen('http://sublime.wbond.net/'+pf.replace(' ','%20')).read())

适用于 Sublime Text 2: 
import  urllib2,
安装gocode
go get -u github.com/nsf/gocode
gocode 将会安装在默认`$GOBIN`
另外建议安装gotests(生成测试代码):
先在sublime安装gotests插件,再运行:
go get -u -v github.com/cweill/gotests/...
验证是否安装成功,你可以打开Sublime,打开main.go,看看语法是不是高亮了,输入import是不是自动化提示了,import "fmt"之后,输入fmt.是不是自动化提示有函数了

基础语法

一.定义变量

定义一个 nickName变量
var nickName type
    // 定义多个变量
    var nickName,nickName1,nickName2 type
    // 定义变量并初始化值
    var nickName type = "cocoaAhda"
    // 同事初始化多个变量
    var nickName,nickName1,nickName2 type = "1","2","3"
    // 当然Go语言也可以像sweift php一样通过值来进行推断
    var nickName,nickName1,nickName2 = "1","2","3"
    nickName,nickName1,nickName2 := "1","2","3"
    // :=这个符号直接取代了var和type,这种形式叫做简短声明。不过它有一个限制,那就是它只能用在函数内部;在函数外部使用则会无法编译通过,所以一般用var方式来定义全局变量
    // _在Go中是一个特殊的变量名,任何赋予它的值都会被丢弃
    _,i := 20,21
    // 上边一行代码中21会赋予给,儿20会被丢弃    
  !注意:Go对弈已声明但是未使用的变量会在编译时报错。!

二.定义常量

所谓常量,也就是在程序编译阶段就切丁下来的值,儿程序在运行时无法改变该值,Go中常量可以定义为数值,布尔值,或字符串等类型
语法如下

    const constantName = value 
    // 如果需要,也可以明确指定常量的类型
    const Pi  float32 = 3.14

    几种声明例子
    const Pi = 3.1415926
    const i  = 10000
    const MaxThread = 10
    const prefix    = "astaxie_"

    ```
###三.内置基础类型
* Boolean

Boolean:在Go中,布尔值的类型为bool,值是true或false,默认为false
// 实例代码
1.全局变量
var isActive bool
2.忽略类型的声明
var star,end = true,false
3.简短声明和赋值
func test() {
var available bool
valid := true
available = false
}

* 数值类型

数值类型:整数类型有无符号和带符号两种。Go同时支持int和uint,这两种类型的长度相同,但具体长度取决于不同编译器的实现。Go里面也有直接定义好位数的类型:rune, int8, int16, int32, int64和byte, uint8, uint16, uint32, uint64。其中rune是int32的别称,byte是uint8的别称.需要注意的一点是,这些类型的变量之间不允许互相赋值或操作,不然会在编译时引起编译器报错。
// 下边代码就会报类型不匹配的错误(invalid operation:a+b(mismatched types int8 and int32))
var a int8
var b int32
c := a+b
// 浮点数的类型有float32和float64两种(没有float类型),默认是float64。
// Go还有复数,它的默认类型是complex128(64位实数+64位虚数)。如果需要小一些的,也有complex64(32位实数+32位虚数)。复数的形式为RE + IMi,其中RE是实数部分,IM是虚数部分,而最后的i是虚数单位。下面是一个使用复数的例子:
var c complex64 = 5+5i
// output:(5+5i)
fmt.Printf("Value is:%v",c)

```
  • 字符串
字符串:Go中的字符串都是采用UTF-8字符集编码。字符串是用一对双引号("")或反引号(` `)括起来定义,它的类型是string。
    //示例代码
    var frenchHello string  // 声明变量为字符串的一般方法
    var emptyString string = ""  // 声明了一个字符串变量,初始化为空字符串
    func test() {
        no, yes, maybe := "no", "yes", "maybe"  // 简短声明,同时声明多个变量
        japaneseHello := "Konichiwa"  // 同上
        frenchHello = "Bonjour"  // 常规赋值
    }
    // 在Go中字符串是不可变的,例如下面的代码编译时会报错:cannot assign to s[0]
    var s string = "hello"
    s[0] = 'c'
    //但如果真的想要修改怎么办呢?下面的代码可以实现:
    s := "hello"
    c := []byte(s)  // 将字符串 s 转换为 []byte 类型
    c[0] = 'c'
    s2 := string(c)  // 再转换回 string 类型
    fmt.Printf("%s\n", s2)
    // Go中可以使用+操作符来连接两个字符串:
    s := "hello,"
    m := " world"
    a := s + m
    fmt.Printf("%s\n", a)
    // 修改字符串也可写为:
    s := "hello"
    s = "c" + s[1:] // 字符串虽不能更改,但可进行切片操作
    fmt.Printf("%s\n", s)
    // 如果要声明一个多行的字符串怎么办?可以通过`来声明:
    m := `hello
            world`
    // 错误类型:Go内置有一个error类型,专门用来处理错误信息,Go的package里面还专门有一个包errors来处理错误:
    err := errors.New("emit macho dwarf: elf header corrupted")
    if err != nil {
        fmt.Print(err)
    }
  • Go数据底层的存储
    Go数据底层的存储,下面这张图来源于Russ Cox Blog中一篇介绍Go数据结构的文章,大家可以看到这些基础类型底层都是分配了一块内存,然后存储了相应的值。
Go数据格式的存储.png
  • 一些技巧
    • 分组声明:
    import "fmt"
    import "os"

    const i = 100
    const pi = 3.1415
    const prefix = "Go_"

    var i int
    var pi float32
    var prefix string

    // 可以分组写成如下形式:
    import(
    
        "fmt"
        "os"
    )

    const(
        i = 100
        pi = 3.1415926
        prefix = "Go__"
    )

    var(
        i int 
        pi float32
        prefix string
    )

    ```
* iota枚举值
    iota枚举值:Go里面有一个关键字iota,这个关键字用来声明enum的时候采用,它默认开始值是0,const中每增加一行加1:
    ```
    package main

    import (
        "fmt"
    )

    const (
        x = iota // x == 0
        y = iota // y == 1
        z = iota // z == 2
        w        // 常量声明省略值时,默认和之前一个值的字面相同。这里隐式地说w = iota,因此w == 3。其实上面y和z可同样不用"= iota"
    )

    const v = iota // 每遇到一个const关键字,iota就会重置,此时v == 0

    const (
        h, i, j = iota, iota, iota //h=0,i=0,j=0 iota在同一行值相同
    )

    const (
        a       = iota //a=0
        b       = "B"
        c       = iota             //c=2
        d, e, f = iota, iota, iota //d=3,e=3,f=3
        g       = iota             //g = 4
    )

    func main() {
        fmt.Println(a, b, c, d, e, f, g, h, i, j, x, y, z, w, v)
    }
    除非被显式设置为其它值或iota,每个const分组的第一个常量被默认设置为它的0值,第二及后续的常量被默认设置为它前面那个常量的值,如果前面那个常量的值是iota,则它也被设置为iota。

    ```

* array丶slice丶map 
    array:array就是数组,它的定义方式↓
    ```
    
    var arr [n]type
    // 在[n]type中,n表示数组的长度,type表示存储元素的类型。对数组的操作和其它语言类似,都是通过[]来进行读取或赋值
    var arr [10]int  // 一个int类型的数组
    arr[0] = 42      // 数组下标从0开始
    arr[1] = 13      // 赋值操作
    fmt.Printf("The first element is %d\n",arr[0])  // 获取数据
    fmt.Printf("The last element is %d\n",arr[9])  // 最后元素
    
    ```
    由于长度也是数组类型的一部分,因此[3]int与[4]int是不同的类型,数组也就不能改变长度。数组之间的赋值是值的赋值,即当把一个数组作为参数传入函数的时候,传入的其实是该数组的副本,而不是它的指针。如果要使用指针,那么就需要用到后面介绍的slice类型了。

  另一种声明方式  :=
    ```
    a := [3]int{1, 2, 3} // 声明了一个长度为3的int数组

    b := [10]int{1, 2, 3} // 声明了一个长度为10的int数组,其中前三个元素初始化为1、2、3,其它默认为0

    c := [...]int{4, 5, 6} // 可以省略长度而采用`...`的方式,Go会自动根据元素个数来计算长度
    ```
    数组嵌套或者说二维数组:
    ```
    // 声明了一个二维数组,该数组以两个数组作为元素,其中每个数组中又有4个int类型的元素
    doubleArray := [2][4]int{[4]int{1, 2, 3, 4}, [4]int{5, 6, 7, 8}}

    // 上面的声明可以简化,直接忽略内部的类型
    easyArray := [2][4]int{{1, 2, 3, 4}, {5, 6, 7, 8}}

    ```
* slice
* slice :在很多应用场景中,数组并不能满足我们的需求。在初始定义数组时,我们并不知道需要多大的数组,因此我们就需要“动态数组”。在Go里面这种数据结构叫sliceslice并不是真正意义上的动态数组,而是一个引用类型。slice总是指向一个底层array,slice的声明也可以像array一样,只是不需要长度。
    ```
    // 和声明array一样,只是少了长度
    var fslice []int
    接下来我们可以声明一个slice,并初始化数据,如下所示:

    slice := []byte {'a', 'b', 'c', 'd'}
    slice可以从一个数组或一个已经存在的slice中再次声明。slice通过array[i:j]来获取,其中i是数组的开始位置,j是结束位置,但不包含array[j],它的长度是j-i。

    // 声明一个含有10个元素元素类型为byte的数组
    var ar = [10]byte {'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j'}

    // 声明两个含有byte的slice
    var a, b []byte

    // a指向数组的第3个元素开始,并到第五个元素结束,
    a = ar[2:5]
    //现在a含有的元素: ar[2]、ar[3]和ar[4]

    // b是数组ar的另一个slice
    b = ar[3:5]
    // b的元素是:ar[3]和ar[4]
    注意slice和数组在声明时的区别:声明数组时,方括号内写明了数组的长度或使用...自动计算长度,而声明slice时,方括号内没有任何字符。

    ```

    * slice有一些简便的操作
1.slice的默认开始位置是0,ar[:n]等价于ar[0:n]
2.slice的第二个序列默认是数组的长度,ar[n:]等价于ar[n:len(ar)]
3.如果从一个数组里面直接获取slice,可以这样ar[:],因为默认第一个序列是0,第二个是数组的长度,即等价于ar[0:len(ar)]

// 声明一个数组
var array = [10]byte{'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j'}
// 声明两个slice
var aSlice, bSlice []byte

// 演示一些简便操作
aSlice = array[:3] // 等价于aSlice = array[0:3] aSlice包含元素: a,b,c
aSlice = array[5:] // 等价于aSlice = array[5:10] aSlice包含元素: f,g,h,i,j
aSlice = array[:]  // 等价于aSlice = array[0:10] 这样aSlice包含了全部的元素

// 从slice中获取slice
aSlice = array[3:7]  // aSlice包含元素: d,e,f,g,len=4,cap=7
bSlice = aSlice[1:3] // bSlice 包含aSlice[1], aSlice[2] 也就是含有: e,f
bSlice = aSlice[:3]  // bSlice 包含 aSlice[0], aSlice[1], aSlice[2] 也就是含有: d,e,f
bSlice = aSlice[0:5] // 对slice的slice可以在cap范围内扩展,此时bSlice包含:d,e,f,g,h
bSlice = aSlice[:]   // bSlice包含所有aSlice的元素: d,e,f,g

```
* slice是引用类型,所以当引用改变其中元素的值时,其它的所有引用都会改变该值,例如上面的aSlice和bSlice,如果修改了aSlice中元素的值,那么bSlice相对应的值也会改变。

从概念上面来说slice像一个结构体,这个结构体包含了三个元素:

1.一个指针,指向数组中slice指定的开始位置
2.长度,即slice的长度
3.最大长度,也就是slice开始位置到数组的最后位置的长度

Array_a := [10]byte{'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j'}
Slice_a := Array_a[2:5]

上面代码的真正存储结构如下图所示
存储结构.png

对于slice有几个有用的内置函数

    1. len 获取slice的长度
    2.cap 获取slice的最大容量
    3.append 向slice里面追加一个或者多个元素,然后返回一个和slice一样类型的slice
    4.copy 函数copy从源slice的src中复制元素到目标dst,并且返回复制的元素的个数
    注:append函数会改变slice所引用的数组的内容,从而影响到引用同一数组的其它slice。 但当slice中没有剩余空间(即(cap-len) == 0)时,此时将动态分配新的数组空间。返回的slice数组指针将指向这个空间,而原数组的内容将保持不变;其它引用此数组的slice则不受影响。

从Go1.2开始slice支持了三个参数的slice,之前我们一直采用这种方式在slice或者array基础上来获取一个slice

    var array [10]int
    slice := array[2:4]
    这个例子里面slice的容量是8,新版本里面可以指定这个容量

    slice = array[2:4:7]
    上面这个的容量就是7-2,即5。这样这个产生的新的slice就没办法访问最后的三个元素。

    如果slice是这样的形式array[:i:j],即第一个参数为空,默认值就是0。
  • 引用类型map:
    引用类型map:也就是OC中字典的概念,它的格式为map[keyType]valueType我们看下面的代码,map的读取和设置也类似slice一样,通过key来操作,只是slice的index只能是`int`类型,而map多了很多类型,可以是int,可以是string及所有完全定义了==与!=操作的类型。
    // 声明一个key是字符串,值为int的字典,这种方式的声明需要在使用之前使用make初始化
    var numbers map[string]int
    // 另一种map的声明方式
    numbers := make(map[string]int)
    numbers["one"] = 1  //赋值
    numbers["ten"] = 10 //赋值
    numbers["three"] = 3

    fmt.Println("第三个数字是: ", numbers["three"]) // 读取数据
    // 打印出来如:第三个数字是: 3
    ```
    * 使用map过程中需要注意的几点:

     1.map是无序的,每次打印出来的map都会不一样,它不能通过index获取,而必须通过key获取
   2.map的长度是不固定的,也就是和slice一样,也是一种引用类型
    3.内置的len函数同样适用于map,返回map拥有的key的数量
    4.map的值可以很方便的修改,通过numbers["one"]=11可以很容易的把key为one的字典值改为11
    5.map和其他基本型别不同,它不是thread-safe,在多个go-routine存取时,必须使用mutex lock机制
    6.map的初始化可以通过key:val的方式初始化值,同时map内置有判断是否存在key的方式
* 通过delete删除map的元素:
    ```
    // 初始化一个字典
    rating := map[string]float32{"C":5, "Go":4.5, "Python":4.5, "C++":2 }
    // map有两个返回值,第二个返回值,如果不存在key,那么ok为false,如果存在ok为true
    csharpRating, ok := rating["C#"]
    if ok {
        fmt.Println("C# is in the map and its rating is ", csharpRating)
    } else {
        fmt.Println("We have no rating associated with C# in the map")
    }

    delete(rating, "C")  // 删除key为C的元素
    ```

    上面说过了,map也是一种引用类型,如果两个map同时指向一个内存,那么一个改变,另一个也相应的改变:
    ```
    m := make(map[string]string)
    m["Hello"] = "Bonjour"
    m1 := m
    m1["Hello"] = "Salut"  // 现在m["hello"]的值已经是Salut了
    ```
* make和new
make用于开辟空间(map,slice和channel),new用于各种类型的内存分配
    内建函数new本质上说跟其它语言中的同名函数功能一样:new(T)分配了零值填充的T类型的内存空间,并且返回其地址,即一个*T类型的值。用Go的术语说,它返回了一个指针,指向新分配的类型T的零值。有一点非常重要:
new返回指针。
    内建函数make(T, args)与new(T)有着不同的功能,make只能创建slice、map和channel,并且返回一个有初始值(非零)的T类型,而不是*T。本质来讲,导致这三个类型有所不同的原因是指向数据结构的引用在使用前必须被初始化。例如,一个slice,是一个包含指向数据(内部array)的指针、长度和容量的三项描述符;在这些项目被初始化之前,slice为nil。对于slice、map和channel来说,make初始化了内部的数据结构,填充适当的值。
make返回初始化后的(非零)值。
make返回内容,new返回指针
    下面这个图详细的解释了new和make之间的区别:
    
![new和make的区别.png](http://upload-images.jianshu.io/upload_images/1782677-7add29b0ce024b4b.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

* 零值
零值:关于“零值”,所指并非是空值,而是一种“变量未填充前”的默认值,通常为0。 此处罗列 部分类型 的 “零值”
int     0
int8    0
int32   0
int64   0
uint    0x0
rune    0 //rune的实际类型是 int32
byte    0x0 // byte的实际类型是 uint8
float32 0 //长度为 4 byte
float64 0 //长度为 8 byte
bool    false
string  ""
##Go程序设计的一些规则
* Go之所以会那么简洁,是因为它有一些默认的行为:
 * 大写字母开头的变量是可导出的,也就是其它包可以读取的,是公有变量;小写字母开头的就是不可导出的,是私有变量。
 * 大写字母开头的函数也是一样,相当于class中的带public关键词的公有函数;小写字母开头的就是有private关键词的私有函数。



                                             --End--
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,898评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,401评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,058评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,539评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,382评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,319评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,706评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,370评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,664评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,715评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,476评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,326评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,730评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,003评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,275评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,683评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,877评论 2 335

推荐阅读更多精彩内容

  • Go语言做Web编程非常方便,并且在开发效率和程序运行效率方面都非常优秀。相比于Java,其最大的优势就是简便易用...
    暗黑破坏球嘿哈阅读 8,981评论 6 67
  • 出处---Go编程语言 欢迎来到 Go 编程语言指南。本指南涵盖了该语言的大部分重要特性 Go 语言的交互式简介,...
    Tuberose阅读 18,390评论 1 46
  • fmt格式化字符串 格式:%[旗标][宽度][.精度][arg索引]动词旗标有以下几种:+: 对于数值类型总是输出...
    皮皮v阅读 1,085评论 0 3
  • 我 雀 操 咱们三是出门不洗头之交 买票的事情上我出了幺蛾子 我是我本人的信息还没有确认
    萌石专阅读 220评论 0 0
  • 读完罗胖的《我懂你的知识焦虑》,不但重新整顿了我的三观,并且还找到了我心心念念的要找的书,此时特别应景的想插入一句...
    雲朵兒阅读 435评论 0 1