浅谈Socket编程

说到Socket,想必大家会觉得陌生又熟悉。许多同学听说过Socket,但仅仅知道它翻译成中文叫做套接字,除此之外似乎并没有太多的了解了。那么今天我就来抛砖引玉地聊一聊Socket。有人说

The lower application layers are all about socket programming
应用的底层全是和socket打交道

一看到涉及底层,有的同学就表示:
其实这些东西并不深奥,只要花一些时间去看,肯定是能够看懂的,并且一但找到了点儿感觉,会觉得Socket非常有趣。
你难道不好奇浏览器是怎样和web服务器勾搭在一起来取悦你的吗?许多网络应用都通过Socket来交流,所以Socket在网络编程里占有了很重要的地位。
那么言归正传,到底什么是Socket的呢?——大学教材上的答案是套接字。我个人觉得这是一个不太好的翻译,虽然说仔细一想有那么点儿意思,但是99%的人即使看见套接字这个词,依然不知道是什么鬼,所以没有翻译的必要。就像Rap你硬要说中文翻译叫拉普也没啥意义对吧。
在Unix中,有一种说法叫

Everything is a file
一切皆文件

所以你只需要记住Socket是某种类型文件的抽象
怎么理解这句话呢?想象一下,假设你要开发一个网络应用,需要在两个客户端之间发消息。整个过程可能包含以下步骤:

  • 客户端组装数据
  • 客户端之间约定好数据格式
  • 客户端向指定地址发送请求
  • DNS服务器解析请求地址
  • DNS没有找到地址,然后跳转到另一个DNS,一直到找到为止
  • 返回客户端真实的IP
  • 客户端向对应IP建立连接请求(三次握手)
  • 开始发送数据,窗口以2的k次幂大小滑动
    ……
    以上省略一本页数为1047的《TCP/IP》
    有人可能已经喷了,你不是说Socket编程很简单吗,这还叫简单?
    正因为这很复杂,所以前人们对这个过程进行了一种抽象,来帮助我们编程。
    你不就是想把数据发给对方吗?组装数据然后发给某人这个过程,和组装好数据然后写到某个文件里有什么区别呢?对了,没有区别。
    Socket就是一种特殊的文件。它是一个连接了两个用户的文件,任何一个用户向Socket里写数据,另一个用户都能看得到,不管这两个用户分布在世界上相距多么遥远的角落,感觉就像坐在一起传纸条一样。
    这么讲Socket应该更容易理解吧?这种抽象是非常重要的,因为它屏蔽了更底层的东西,我就想写个程序发送下数据,为什么要关系物理层怎么传输呢,对吧。
    所以有了Socket的概念之后,我们在两个客户端之间发送消息可能就是这样的:
  • 指定对方的地址
  • 打开一个和对方连接的Socket
  • 把Socket当成普通的文件,往里写数据
  • 要是发现Socket里有数据,就读出来,那必然是对方发过来的

这样的话,网络编程是不是就非常简单了呢?
那么下面我们用Go语言作为示例,演示一下。

package main

import (
"fmt"
"net"
"os"
) func main() {
//使用tcp协议,要监听的是6666端口
tcpAddr, err := net.ResolveTcpAddr("tcp", "localhost:6666")
checkError(err)
//开始监听
fd, err := net.ListenTcp("tcp", tcpAddr)
checkError(err)
for {
//获取Socket
conn, err := fd.Accept()
if err != nil {
continue
}
//你的逻辑
go Handle(conn)
}
} func Handle(conn *net.Conn) {
//You can do anything you want to here...
} func checkError(err error) {
if err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
}

端口是什么概念?可以近似这么想:一台电脑就是你家的小区,你买东西如果填的地址是你家小区,那么快递员最多能把东西送到小区门口;但是如果你写上了你家的门牌号,那么快递员就能送到你家门口。同样的,电脑上同时运行着很多程序,比如QQ,旺旺…但是电脑只有只有一个IP地址,一条消息来了没人知道这个消息是给谁的,于是就有了端口的概念。QQ在这台电脑的4567端口,旺旺在这台电脑1234端口。发消息的人只要知道它在什么端口,就能准确地把消息发过来了。
同样的,网络通信两端的人得事先约定好一个端口,然后一个人守着这个端口,待另一方连接了这个端口,这才算建立了Socket连接。就好两个人打电话,不需要关心信号怎么转换和传输,但在建立这次通话之前必须有人拨号,同时有人守在电话旁。
于是上面的代码应该就可以理解了吧?
一个程序猿走到"localhost:6666"这个“电话”旁边

tcpAddr,err := net.ResolveTcpAddr("tcp","localhost:6666")`

然后坐下来等电话响

fd,err := net.ListenTcp("tcp", tcpAddr)

他也不知道女朋友什么时候打电话过来,于是开始了漫长的等待

//一个死循环
for {
conn,err := fd.Accept()
//电话没有响就一直堵在上面这条语句
}

在漫长地等待中,突然电话响了,然后开始了一段佳话(程序终于不堵了,接着向下执行)

go Handle(conn)

Handle方法就用来处理对话,数据都在conn里面,只需要学习相关的API就能知道怎么把具体的内容取出来了。
整个过程是不是很简单?
有些机智的同学可能已经发现了,要这样的话,两个人都在等对方打电话过来,岂不是就终身无缘了(这种误会就像韩剧)。对的,所以我们还需要知道,如何给对方拨号。这是很关键的一步,一般妹子不好意思,那么自然我们得上了。
怎么拨号呢?请看代码:

import (
"fmt"
"io/ioutil"
"net"
"os"
) func main() {
tcpAddr, err := net.ResolveTcpAddr("tcp", "localhost:6666")
checkError(err)
conn, err := net.DialTcp("tcp", nil, tcpAddr)
checkError(err)
_, err = conn.Write([]byte("妹子,约吗?"))
result, err := ioutil.ReadAll(conn)
//不约,叔叔我们不约
} func checkError(err error) {
if err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(1)
}
}

解释一下,一般妹子都比较含蓄,告诉你联系方式不那么直接,你得破解一下

tcpAddr := net.ResolveTcpAddr("tcp", "localhost:6666")

呵呵,嘴上说不要身体却很诚实嘛,这么容易就破解了。(其实是Go的包比较好用好吗!)
然后按着电话号拨打电话

conn, err := net.DialTcp("tcp", nil, tcpAddr)

电话打通了,conn就代表这次通话。屌丝们已经急不可耐了,于是大喊一句:

_, err = conn.Write([]byte("妹子,约吗?"))

为什么我第一个返回值用_,这表示我不想知道函数的返回结果,即Write了多少个字节。我问妹子约不约,你说我关不关心这句话包含几个字节?

result, err := ioutil.ReadAll(conn)

妹子给的回复就在result里,慢慢去琢磨吧……


以上示例并不完整,完整的示例网上到处可见,希望大家能自己写一写。
本篇只是粗浅地讲了讲什么是Socket编程以及基本过程,之后会有更细致地讲解(比如:并发)。

作者: PHPBird 
链接:http://www.imooc.com/article/1668
来源:慕课网

浅谈Socket编程的更多相关文章

  1. iOS开发--浅谈CocoaAsyncSocket编程

    Socket就是一种特殊的文件.它是一个连接了两个用户的文件,任何一个用户向Socket里写数据,另一个用户都能看得到,不管这两个用户分布在世界上相距多么遥远的角落,感觉就像坐在一起传纸条一样. 这么 ...

  2. 浅谈JDBC编程

    一.概述 1.为什么要用JDBC 数据库是程序不可或缺的一部分,每一个网站和服务器的建设都需要数据库.对于大多数应用程序员(此处不包含数据库开发人员)来说,我们更多的不是在DBMS中对数据库进行操纵, ...

  3. 浅谈Socket长连+多线程[原创,欢迎指点]

    前戏 [PS:原文手打,转载说明出处] [PS:博主自认为适用于云平台设备管控,且适用于IM主控] 好久没来了,13年时还想着多写一些博客,这都17年过年,年前也写一写Scoket+多线程,不足之处, ...

  4. day10 浅谈面向对象编程

    面向对象编程:第一步找名词,名词是问题域中的. 第二步概括名词设计成类.某些名词可以浓缩包含到其它名词中,成为其属性. 第三步找动词,动词也是问题域中的.   第四步概括动词设计成方法.动作的产生往往 ...

  5. 浅谈Socket长连+多线程

    缘由 不知各位同仁有没有发现,用简单,无外乎就是都是一个流程 1)监听链接 2)校验链接是否是正常链接 3)保存链接至全局静态字典 4)开启新线程监听监听到的线程报文 5)执行对应命令或者发送对应命令 ...

  6. 浅谈React编程思想

    React是Facebook推出的面向视图层开发的一个框架,用于解决大型应用,包括如何很好地管理DOM结构,是构建大型,快速Web app的首选方式. React使用JavaScript来构建用户界面 ...

  7. 浅谈 JavaScript 编程语言的编码规范

    对于熟悉 C/C++ 或 Java 语言的工程师来说,JavaScript 显得灵活,简单易懂,对代码的格式的要求也相对松散.很容易学习,并运用到自己的代码中.也正因为这样,JavaScript 的编 ...

  8. 浅谈JAVA中如何利用socket进行网络编程(二)

    转自:http://developer.51cto.com/art/201106/268386.htm Socket是网络上运行的两个程序间双向通讯的一端,它既可以接受请求,也可以发送请求,利用它可以 ...

  9. Java网络编程和NIO详解7:浅谈 Linux 中NIO Selector 的实现原理

    Java网络编程和NIO详解7:浅谈 Linux 中NIO Selector 的实现原理 转自:https://www.jianshu.com/p/2b71ea919d49 本系列文章首发于我的个人博 ...

随机推荐

  1. Rails应用系列(1):初识Rails

    第一个Rails应用 Rails是一个"模型-视图-控制器"框架(MVC).是用Ruby写的,所以要对Ruby要有一定的了解才能对rails框架深入学习.其实Ruby与Rails就 ...

  2. js 中文汉字转Unicode、Unicode转中文汉字、ASCII转换Unicode、Unicode转换ASCII、中文转换&#XXX函数代码

    最近看不少在线工具里面都有一些编码转换的代码,很多情况下我们都用得到,这里脚本之家小编就跟大家分享一下这些资料 Unicode介绍 Unicode(统一码.万国码.单一码)是一种在计算机上使用的字符编 ...

  3. hadoop 配置安装

    1.   下载hadoop 压缩包,   拷贝到 /usr/hadoop目录下   tar -zxvf  hadoop-2.7.1.tar.gz, 比如: 127.0.0.1 localhost 19 ...

  4. [转] JAVA从本机获取IP地址

    [From] https://www.cnblogs.com/xiaoBlog2016/p/7076230.html 论述: 此篇博客是在工作的时候,需要获得当前网络下面正确的ip地址,在网上查阅很多 ...

  5. [转] node.js如何获取时间戳与时间差

    [From] http://www.jb51.net/article/89767.htm Nodejs中获取时间戳的方法有很多种,例如: 1.new Date().getTime()  2.Date. ...

  6. docker 镜像编译

    docker为我们提供了,包含源码的镜像, 可以要从docker hub上下载镜像来编译docker源码. 1.  docker pull docker-dev:v1.2.0,其他版本就到docker ...

  7. PIE SDK克里金插值法

    1.算法功能简介 克里金插值法基于一般最小二乘法的随机插值技术没用方差图作为权重函数,被应用于任何点数据估计其在地表上分布的现象,被称为空间自协方差最佳插值法,是一种最优内插法也是一种最常用的空间插值 ...

  8. Rsa2加密报错java.security.spec.InvalidKeySpecException的解决办法

    最近在和支付宝支付做个对接,Java项目中用到了RSA2进行加解密,在加密过程中遇到了错误: java.security.spec.InvalidKeySpecException: java.secu ...

  9. 你还在把Java当成Android官方开发语言吗?Kotlin了解一下!

    导语:2017年Google IO大会宣布使用Kotlin作为Android的官方开发语言,相比较与典型的面相对象的JAVA语言,Kotlin作为一种新式的函数式编程语言,也有人称之为Android平 ...

  10. Springboot - 自定义错误页面

    Springboot 没找到页面或内部错误时,会访问默认错误页面.这节我们来自定义错误页面. 自定义错误页面 1.在resources 目录下面再建一个 resources 文件夹,里面建一个 err ...