子进程回收资源两种方式,僵尸进程与孤儿进程,守护进程,进程间数据隔离,进程互斥锁,队列,IPC机制,线程,守护线程,线程互斥锁,线程local作用,线程池,回调函数add_done_callback,TCP服务端实现并发

子进程回收资源两种方式

- 1) join让主进程等待子进程结束,并回收子进程资源,主进程再结束并回收资源。
- 2) 主进程 “正常结束” ,子进程与主进程一并被回收资源。

from multiprocessing import Process
import time

# 任务
def task():
    print('start....')
    time.sleep(2)
    print('end......')

if __name__ == '__main__':
    p = Process(target=task)

    # 告诉操作系统帮你开启子进程
    p.start()
    # p.join()

    time.sleep(3)

    # 主进程结束
    print('主进程结束....')

2.僵尸进程与孤儿进程(了解)

'''
僵尸进程 (有坏处):
    - 在子进程结束后,主进程没有正常结束, 子进程PID不会被回收。

    缺点:
        - 操作系统中的PID号是有限的,如有子进程PID号无法正常回收,则会占用PID号。
        - 资源浪费。
        - 若PID号满了,则无法创建新的进程。

孤儿进程(没有坏处):

    - 在子进程没有结束时,主进程没有“正常结束”, 子进程PID不会被回收。
    - 操作系统优化机制(孤儿院):
        当主进程意外终止,操作系统会检测是否有正在运行的子进程,会他们放入孤儿院中,让操作系统帮你自动回收。

# 进程的其他属性与方法
'''

from multiprocessing import Process
from multiprocessing import current_process
# 在子进程中调用,可以拿到子进程对象.pid可以pid号
# 在主进程中调用,可以拿到主进程对象.pid可以pid号
import os

import time


# 任务
def task():
    print(f'start....{current_process().pid}')
    time.sleep(1000)
    print(f'end......{os.getpid()}')
    print('子进程结束啦啊....~~~')


if __name__ == '__main__':
    p = Process(target=task)

    # 告诉操作系统帮你开启子进程
    p.start()
    # p.join()

    print(f'进入主进程的IO-->{current_process().pid}')
    time.sleep(4)
    print(f'进入主进程的IO-->{os.getpid()}')
    # 主进程结束
    print('主进程结束....')
    print(f'查看主主进程{os.getppid()}')
    f = open('tank.txt')    # 强制让主进程报错

3.守护进程:
当主进程结束时,子进程也必须结束,并回收。

from multiprocessing import Process

import time

# 任务
def demo(name):
    print(f'start....{name}')
    time.sleep(1000)
    print(f'end......{name}')
    print('子进程结束啦啊....~~~')

if __name__ == '__main__':
    p = Process(target=demo, args=('童子军jason1号', ))

    # 守护进程必须在p.start()调用之前设置
    p.daemon = True  # 将子进程p设置为守护进程

    # 告诉操作系统帮你开启子进程
    p.start()
    # p.join()

    time.sleep(1)
    print('皇帝驾崩啦啊~~~')

4.进程间数据是隔离的

from multiprocessing import Process
import time
'''
进程间数据是隔离。
'''
number = 10

def func():
    global number
    number = 100

def func2(number):
    number += 100

if __name__ == '__main__':
    p_obj = Process(target=func)
    p_obj2 = Process(target=func2, args=(number, ))
    p_obj.start()
    p_obj2.start()
    p_obj2.join()
    p_obj.join()
    time.sleep(1)
    print(number)  # 10

5.进程互斥锁

互斥锁是一把锁,用来保证数据读写安全的。
- 抢票例子

from multiprocessing import Process
from multiprocessing import Lock  # ---》 进程互斥锁
import random
import time
import json
# 抢票例子:


# 1.查看余票
def search(name):
    # 1.读取data.json文件中的数据
    with open('data.json', 'r', encoding='utf-8') as f:
        data_dic = json.load(f)
        print(f'用户【{name}】查看余票,余票还剩: {data_dic.get("number")}!')


# 2.若有余票,购买成功,票数会减少
def buy(name):  # buy()

    # 网络延时
    with open('data.json', 'r', encoding='utf-8') as f:
        data_dic = json.load(f)

    # 进入这一步证明最先抢到票
    if data_dic.get('number') > 0:
        data_dic['number'] -= 1
        time.sleep(random.randint(1, 3))
        with open('data.json', 'w', encoding='utf-8') as f:
             json.dump(data_dic, f)
        print(f'用户【{name}】, 抢票成功!')

    else:
        print(f'用户【{name}】, 抢票失败!')


def run(name, lock):
    # 1.假设1000个用户过来都可以立马查看余票
    search(name)

    lock.acquire()  # 加锁
    buy(name)
    lock.release()  # 释放锁


if __name__ == '__main__':
    lock = Lock()
    # 开启多进程: 实现并发
    for line in range(10):
        p_obj = Process(target=run, args=(f'jason{line}', lock))
        p_obj.start()

6.队列: (FIFO)先进先出 进----》 [3, 2, 1] ----》 出 1, 2, 3
- 先存放的数据,就先取出来。
相当于一个第三方的管道,可以存放数据。

附:堆栈(FILO)

应用: 让进程之间数据进行交互

from multiprocessing import Queue  # multiprocessing提供队列  先进先出
from multiprocessing import JoinableQueue  # 基于 Queue 封装的队列 先进先出
import queue  # python内置的队列  先进先出

# 第一种
Queue(5) #指的是队列中只能存放5份数据  如果像q=Queue(),括号内没有指定最大接受消息数量,或者数量为负数,代表可接受的消息数量没有上限,知道内存尽头
q_obj1 = Queue(5)  # q_obj1队列对象
# 添加数据到队列中
q_obj1.put('jason')
print('添加1个')
q_obj1.put('hcy')
print('添加1个')
q_obj1.put('hb')
print('添加1个')
q_obj1.put('zsb')
print('添加1个')
q_obj1.put('lh')
print('添加1个')
#
# put: 只要队列满了,会进入阻塞
# q_obj1.put('sean')
# print('sean into ')

# put_nowait: 只要队列满了,就会报错
# q_obj1.put_nowait('sean')

# get: 只要队列中有数据,就能获取数据,若没有则会进入阻塞
print(q_obj1.get())
print(q_obj1.get())
print(q_obj1.get())
print(q_obj1.get())
print(q_obj1.get())
# print(q_obj1.get())
# get_nowait: 若队列中没有数据获取则会报错
# print(q_obj1.get_nowait())

# 第二种
q_obj1 = JoinableQueue(5)  # q_obj1队列对象
# 添加数据到队列中
q_obj1.put('jason')
print('添加1个')
q_obj1.put('hcy')
print('添加1个')
q_obj1.put('hb')
print('添加1个')
q_obj1.put('zsb')
print('添加1个')
q_obj1.put('lh')
print('添加1个')

# put: 只要队列满了,会进入阻塞
# q_obj1.put('sean')
# print('sean into ')

# put_nowait: 只要队列满了,就会报错
# q_obj1.put_nowait('sean')

# get: 只要队列中有数据,就能获取数据,若没有则会进入阻塞
print(q_obj1.get())
print(q_obj1.get())
print(q_obj1.get())
print(q_obj1.get())
print(q_obj1.get())

# 第三种
q_obj1 = queue.Queue(5)  # q_obj1队列对象
# 添加数据到队列中
q_obj1.put('jason')
print('添加1个')
q_obj1.put('hcy')
print('添加1个')
q_obj1.put('hb')
print('添加1个')
q_obj1.put('zsb')
print('添加1个')
q_obj1.put('lh')
print('添加1个')

# put: 只要队列满了,会进入阻塞
# q_obj1.put('sean')
# print('sean into ')

# put_nowait: 只要队列满了,就会报错
# q_obj1.put_nowait('sean')

# get: 只要队列中有数据,就能获取数据,若没有则会进入阻塞
print(q_obj1.get())
print(q_obj1.get())
print(q_obj1.get())
print(q_obj1.get())
print(q_obj1.get())

7.IPC机制 (进程间实现通信)  面试问: 什么是IPC机制?

进程间数据是相互隔离的,若想实现进程间通信,可以利用队列.

from multiprocessing import Process
from multiprocessing import JoinableQueue
import time

def task1(q):
    x = 100
    q.put(x)
    print('添加数据')

    time.sleep(3)
    print('p1获取数据:',q.get())

def task2(q):
    # 想要在task2中获取task1的x
    res = q.get()
    print(f'p2获取的数据是{res}')
    q.put(9527)

if __name__ == '__main__':
    # 产生队列
    q=JoinableQueue(10)
    # 产生两个不同的子进程
    p1 = Process(target=task1,args=(q,))
    p2 = Process(target=task2,args=(q,))
    p1.start()
    p2.start()

8.生产者与消费者
- 生产者: 生产数据的
- 消费者: 使用数据的

- 生产油条的有人总比吃油条的人少 ---> 生产数据跟不上 使用数据的人 ---》 供需不平衡
- 吃油条的人比生产的油条要少 ---> 使用数据的速度 跟不上 生产数据的速度

- 通过队列来实现,解决供需不平衡问题

from multiprocessing import JoinableQueue
from multiprocessing import Process
import time

# 生产者: 生产数据 ---》 队列
def producer(name, food, q):
    msg = f'{name} 生产了 {food} 食物'
    # 生产一个食物,添加到队列中
    q.put(food)
    print(msg)

# 消费者: 使用数据 《---  列队
def customer(name, q):
    while True:
        try:
            time.sleep(0.5)
            # 若报错,则跳出循环
            food = q.get_nowait()
            msg = f'{name} 吃了 {food} 食物!'
            print(msg)

        except Exception:
            break

if __name__ == '__main__':
    q = JoinableQueue()

    # 创建两个生产者
    for line in range(10):
        p1 = Process(target=producer, args=('tank1', f'Pig饲料{line}', q))
        p1.start()

    # 创建两个消费者
    c1 = Process(target=customer, args=('jason', q))
    c2 = Process(target=customer, args=('sean', q))
    c1.start()
    c2.start()

9.线程

线程不需要像进程,使用if __name__ == '__main__':

10.守护线程

'''
线程:
    1.什么是线程?
        进程: 资源单位。
        线程: 执行单位。

        线程与进程都是虚拟的概念,为了更好表达某种事物。

        注意: 开启一个进程,一定会自带一个线程,线程才是真正的执行者。


    2.为什么要使用线程?
        节省资源的占用。

        - 开启进程:
            - 1) 会产生一个内存空间,申请一块资源。
            - 2) 会自带一个主线程
            - 3) 开启子进程的速度要比开启子线程的速度慢

        - 开启线程
            - 1) 一个进程内可以开启多个线程,从进程的内存空间中申请执行单位。
            - 2) 节省资源。

        - 开启三个进程:
            - 占用三份内存资源

        - 开启三个线程:
            - 从一个内存资源中,申请三个小的执行单位

        - IO密集型用(多核): 多线程
            - IO(时间由用户定):
                - 阻塞: 切换 + 保存状态

        - 计算密集型用(多核): 多进程
            - 计算(时间由操作系统定):
                - 计算时间很长 ---> 切换 + 保存状态


    注意: 进程与进程之间数据是隔离的,线程与线程之间的数据是共享的。

    3.怎么使用线程?


    - 守护线程

'''
from threading import Thread
import time

number = 1000

# 启动线程的方式一:
# 任务1:
def task():
    global number
    number = 100
    print('start...')
    time.sleep(1)
    print('end...')

if __name__ == '__main__':
    # 开启一个子线程
    t = Thread(target=task)
    t.start()
    # t.join()
    print('主进程(主线程)...')
    print(number)

# 启动线程的方式二:
class MyThread(Thread):
    def run(self):
        print('start...')
        time.sleep(1)
        print('end...')

if __name__ == '__main__':
    # 开启一个子线程
    t = MyThread()
    t.start()
    # t.join()
    print('主进程(主线程)...')

# 守护线程
from threading import current_thread number = 1000 def task(): global number number = 100 print(f'start...{current_thread().name}') time.sleep(3) print(f'end...{current_thread().name}') if __name__ == '__main__': # 开启一个子线程 for line in range(10): t = Thread(target=task) # 加上守护线程: 主进程结束,代表主线程也结束,子线程有可能未被回收。 t.daemon = True t.start() # t.join() print(f'主进程(主线程)...{current_thread().name}') print(number)

 

 线程互斥锁

from threading import Lock
from threading import Thread
import time
lock = Lock()

# 开启10个线程,对一个数据进行修改
number = 100

def task():
    global number

    # lock.acquire()
    number2 = number
    # time.sleep(1)
    number = number2 - 1
    # lock.release()

if __name__ == '__main__':

    list1 = []
    for line in range(100000000):
        t = Thread(target=task)
        t.start()
        list1.append(t)

    for t in list1:
        t.join()

    print(number)  # 90

 

threading.local

# 为每个线程创建一个独立的空间,是的线程对自己的空间中的数据进行操作(数据隔离)
import threading
from threading import local
import time

obj = local()

def task(i):
    obj.xxxxx = i
    time.sleep(2)
    print(obj.xxxxx)

for i in range(10): # 开启10个线程
    t = threading.Thread(target=task,args=(i,))
    t.start()

 

11.线程池
- 线程池是用来限制 创建的线程数

from concurrent.futures import ThreadPoolExecutor
import time

# pool只能创建10个线程
pool = ThreadPoolExecutor(10)  # 只能同时运行10个线程


def task(line):
    print(line)
    time.sleep(10)


if __name__ == '__main__':
    for line in range(100):
        pool.submit(task, line)

回调函数 add_done_callback

from concurrent.futures import ThreadPoolExecutor
# pool线程池对象
pool = ThreadPoolExecutor(50)

# 并发执行的任务
def task():
return '刺客伍六七'

# 自定义的回调函数
def callback(res_obj):  # res_obj  {'result()': '刺客伍六七'}
result = res_obj.result()  # 刺客伍六七


# submit(函数名, 函数接收的参数1, 参数2...)  =====  Thread(target=函数名, args=(函数接收的参数1, 参数2...))
# pool.submit(函数名).add_done_callback(回调函数的名字)

pool.submit(task).add_done_callback(callback)

 

# add_done_callback
from concurrent.futures import ThreadPoolExecutor
import time
# 池子对象: 内部可以帮你提交50个启动进程的任务
p_pool = ThreadPoolExecutor(50)


def task1(n):
    print(f'from task1...{n}')
    time.sleep(5)
    return 'tank'


def get_result(obj):
    # print(obj.__dict__)
    # print(obj._result)
    result = obj.result()
    print(result)


if __name__ == '__main__':
    n = 1
    while True:
        # 参数1: 函数名
        # 参数2: 函数的参数1
        # 参数3: 函数的参数2
        # submit(参数1, 参数2, 参数3)
        # add_done_callback(参数1),会将submit提交的task1执行的结果,传给get_result中的第一个参数,第一个参数是一个对象。
        p_pool.submit(task1, n).add_done_callback(get_result)
        n += 1

 

TCP服务端实现并发

#server
import socket
from concurrent.futures import ThreadPoolExecutor

server = socket.socket()

server.bind(
    ('127.0.0.1', 9000)
)

server.listen(5)


# 1.封装成一个函数
def run(conn):
    while True:
        try:
            data = conn.recv(1024)
            if len(data) == 0:
                break
            print(data.decode('utf-8'))
            conn.send('111'.encode('utf-8'))

        except Exception as e:
            break

    conn.close()


if __name__ == '__main__':
    print('Server is run....')
    pool = ThreadPoolExecutor(50)
    while True:
        conn, addr = server.accept()
        print(addr)
        pool.submit(run, conn)
#client
import socket

client = socket.socket()

client.connect(
    ('127.0.0.1', 9000)
)

print('Client is run....')
while True:
    msg = input('客户端>>:').encode('utf-8')
    client.send(msg)

    data = client.recv(1024)
    print(data)

 

posted @ 2019-12-08 18:04  战斗小人  阅读(833)  评论(0编辑  收藏  举报