1  复习

2  TCP-UDP协议

3  tcp协议的socket

4  复杂tcp协议的socket

5  带退出的聊天程序

6  时间练习demo

7  粘包现象

1.复习

# 网络编程概念
# 两大开发架构
# C/S B/S

# ip地址
# mac地址
# 内网/外网
# 交换机 : 单播、广播、组播
# 在一个局域网内的通信
# 二层/数据链路层 - osi七层协议
# 数据链路层的协议 :arp协议(通过一台机器的ip地址找到mac地址)
# 数据链路层的网络设备 : 交换机、网卡
# 路由器 :
# 局域网与局域网之间的连接工具
# 三层/网络层 - osi七层协议
# 网络层的协议 :ip协议
# 网络层的设备 :路由器、三层交换机
# 网关
# 一个局域网要想访问局域网外部的其他机器,都要通过统一的网关接口
# 子网掩码
# 判断两台机器是否在同一个局域网内
# 端口来定位某一台机器上的某一个服务
# 如何定位你的网络中能够找到的唯一一台机器+服务 ip + 该服务的端口号

2.tcp-udp协议

# osi七层协议的传输层 : TCP/UDP协议
# 这两个协议标志着我们的数据传递的方式
# tcp协议 :语音通话
# 必须先接通电话,同一时刻只能和一个人聊天,传输比较稳定,慢
# 可以发送任意长度的消息
# udp协议 :发微信
# 不需要接通 只要互相知道ip 端口就可以通信 ,同时可以和多个人聊天,但是传输相对不稳定,传输速度快
# 不能传输过长的消息

# tcp :发送文件(邮件)、下载安装包、上传、下载电影、从网盘上上传、下载文件
# udp :即时通信类的、在线播放

# tcp协议
# 为什么它传输稳定、可靠?
# 先建立连接
# 三次握手、四次挥手
# tcp协议和udp协议各自的特点

3  tcp协议的socket

server

import socket  #套接字

sk = socket.socket()  #先买一部手机sk.bind(('127.0.0.1',9600))  #绑定一个电话卡  PS本地回环地址127.0.0.1sk.listen()  #开机  py3.4后()里可以不加数字conn,addr = sk.accept()  #阻塞,直到有客户端来连接,三次握手建立连接print(addr)  # ('127.0.0.1', 52171)print(conn)  # 包含addr的一些信息print(sk)    # 不包含addr的conn信息conn.send(b'hello')msg = conn.recv(1024)print(msg)conn.close()  #四次挥手,断开连接sk.close()  #关机

client

import socket

sk = socket.socket()sk.connect(('127.0.0.1',9600))msg = sk.recv(1024)print(msg)sk.send(b'byebye')sk.close()

4  复杂tcp协议的socket

    PS:连续相互对话

server

sk = socket.socket()sk.bind(('127.0.0.1',9600))sk.listen()conn,addr = sk.accept()  #conn,addr只是可以换其他代替

while True:    send_msg = input('msg : ')    conn.send(send_msg.encode())    msg = conn.recv(1024).decode()    print(msg)conn.close()sk.close()

client

import socket

sk = socket.socket()sk.connect(('127.0.0.1',9600))

while True:    msg = sk.recv(1024).decode()    print(msg)    send_msg = input('msg : ')    sk.send(send_msg.encode())

sk.close()

5  带退出的聊天程序

server

import socket

sk = socket.socket()sk.bind(('127.0.0.1',9600))sk.listen()

conn,addr = sk.accept()

while True:    send_msg = input('msg : ')   # q    conn.send(send_msg.encode()) # send(q)    if send_msg == 'q':break    msg = conn.recv(1024).decode()    if msg == 'q':break    print(msg)

conn.close()

client

import socket

sk = socket.socket()sk.connect(('127.0.0.1',9600))

while True:    msg = sk.recv(1024).decode()    if msg == 'q':break    print(msg)    send_msg = input('msg : ')    sk.send(send_msg.encode())    if send_msg == 'q':break

sk.close()

6  时间练习demo

server

import socketimport time

sk = socket.socket()sk.bind(('127.0.0.1',9600))sk.listen()

while True:    conn,addr = sk.accept()    fmt = conn.recv(1024)    str_time = time.strftime(fmt.decode())    conn.send(str_time.encode())    conn.close()sk.close()

client

import socket

sk = socket.socket()sk.connect(('127.0.0.1',9600))sk.send(b'%Y/%m/%d %H:%M:%S')msg = sk.recv(1024).decode()print(msg)sk.close()

补充:

# str 字符串数据类型
# bytes 字节数据类型
# 中 :gbk bytes 100100110101
# 中 :utf8 bytes 100111100001
# send(bytes)
# bytes = recv()
# 我们要看 是看str
# 我们在发送数据之前是str,发送的是bytes
# 就需要对str进行编码,str.encode('utf-8')结果是一个bytes
# 我们在接受数据的时候 收到的也是bytes类型
# 我们要想看懂必须把bytes解码,bytes.decode('utf-8')结果就是一个字符串

7  粘包现象

server

import struct
import socket

sk = socket.socket()
sk.bind(('127.0.0.1',9000))
sk.listen()

conn,addr = sk.accept()
send_msg = input('>>>').encode()
bytes_len = struct.pack('i',len(send_msg))
conn.send(bytes_len)
conn.send(send_msg) # 粘包现象
conn.send(b'world')
conn.close()
sk.close()

# 1.发送端的粘包 合包机制 + 缓存区
# 2.接收端的粘包 延迟接受 + 缓存区
# 3.流式传输
# 电流 高低电压
# 所以我们说 tcp协议是无边界的流式传输
# 4.拆包机制

# 粘包现象
# 接收端不知道发送端给我发送了多长的数据

client

import struct
import socket

sk = socket.socket()
sk.connect(('127.0.0.1',9000))
bytes_len = sk.recv(4)
msg_len = struct.unpack('i',bytes_len)[0]
msg = sk.recv(msg_len)
print(msg.decode())
msg2 = sk.recv(5)
print(msg2)
sk.close()

解决方法一

import struct

ret = struct.pack('i',560000)
print(ret,len(ret))
ret1 = struct.pack('i',123)
print(ret1,len(ret1))
ret2 = struct.pack('i',902730757)
print(ret2,len(ret2))

res = struct.unpack('i',ret)
print(res[0])
res = struct.unpack('i',ret1)
print(res[0])
res = struct.unpack('i',ret2)
print(res[0])

day24_python_1124的更多相关文章

随机推荐

  1. 5.list集合添加姓名{张三,李四,王五,二丫,钱六,孙七},将二丫替换为王小丫, 写入到"D:\\stuinfo.txt"

    package cn.it.text; import java.io.FileWriter; import java.io.IOException; import java.util.ArrayLis ...

  2. PHP CURL 账号密码 添加授权Authorization头Header

    <?phpfunction http_request_xml($url,$data = null,$arr_header = null){ $curl = curl_init(); curl_s ...

  3. ionic 实现 应用内(webview中html页面点击) 和 应用外 (浏览器html页面点击) 打开本地安装应用

    应用内(webview中html页面点击) : 应用内打开本地安装应用指的是webview里打开应用,需要2个步骤: 1: 需要下载一个cordova插件:com.lampa.startapp ,也可 ...

  4. 使用AtomicLong,经典银行账户问题

    1.新建Account类,使用AtomicLong定义账户余额,增加和减少金额方法使用getAndAdd方法. package com.xkzhangsan.atomicpack.bank; impo ...

  5. 含服务端,客户端,数据库的注册/登录/聊天/在线/离线查看的聊天demo

    用websocket,mysql,node的写了一个简单聊天的demo 实现了: 注册,登陆功能: 聊天信息广播: 在线/离线状态的查看: 服务端: 主要引用http,fs,mysql,socket. ...

  6. 求最近点对算法分析 closest pair algorithm

    这个帖子讲得非常详细严谨,转一波. http://blog.csdn.net/lishuhuakai/article/details/9133961

  7. 理解bootstrap的列偏移offset 和 推拉push/pull的区别?

    参考: http://www.cnblogs.com/jnslove/p/5430481.html & https://blog.csdn.net/hly_coder/article/deta ...

  8. 【做题】POI2011R1 - Plot——最小圆覆盖&倍增

    原文链接 https://www.cnblogs.com/cly-none/p/loj2159.html 题意:给出\(n\)个点,你需要按编号将其划分成不超过\(m\)段连续的区间,使得所有每个区间 ...

  9. 深入浅出ES6:不定参数和默认参数

    不定参数 我们通常使用可变参函数来构造API,可变参函数可接受任意数量的参数.例如,String.prototype.concat方法就可以接受任意数量的字符串参数.ES6提供了一种编写可变参函数的新 ...

  10. Project Euler 345: Matrix Sum

    题目 思路: 将问题转化成最小费用流 代码: #pragma GCC optimize(2) #pragma GCC optimize(3) #pragma GCC optimize(4) #incl ...