Python学习笔记-DAY11

序列化

JSON

JSON进阶

进程和线程:

进程和线程:

多进程:

Pool(进程池)

 

序列化:

在程序运行的过程中,所有的变量都是在内存中,我们把变量从内存中变成可存储或传输的过程称之为序列化,在Python中叫pickling,序列化之后,就可以把序列化后的内容写入磁盘,或者通过网络传输到别的机器上。反过来,把变量内容从序列化的对象重新读到内存里称之为反序列化,即unpickling

Python提供两个模块来实现序列化:cPicklepickle。这两个模块功能是一样的,区别在于cPickleC语言写的,速度快,pickle是纯Python写的,速度慢,跟cStringIOStringIO一个道理。用的时候,先尝试导入cPickle,如果失败,再导入pickle

try:
   
import cPickle as pickle
except ImportError:
   
import pickle
d =
dict(name='Bob', age=20, score=88)
print pickle.dumps(d)

pickle.dumps()方法把任意对象序列化成一个str,然后,就可以把这个str写入文件。或者用另一个方法pickle.dump()直接把对象序列化后写入一个file-like Object

f=open('test3.txt','wb')

pickle.dump(d,f)

f.close()

写入的dump.txt文件,一堆乱七八糟的内容,这些都是Python保存的对象内部信息。

 

当我们要把对象从磁盘读到内存时,可以先把内容读到一个str,然后用pickle.loads()方法反序列化出对象,也可以直接用pickle.load()方法从一个file-like Object中直接反序列化出对象。我们打开另一个Python命令行来反序列化刚才保存的对象:

try:

    import cPickle as pickle

except ImportError:

    import pickle

f=open('test3.txt','rb')

d=pickle.load(f)

f.close()

print d

Pickle的问题和所有其他编程语言特有的序列化问题一样,就是它只能用于Python,并且可能不同版本的Python彼此都不兼容,因此,只能用Pickle保存那些不重要的数据,不能成功地反序列化也没关系。

JSON

如果我们要在不同的编程语言之间传递对象,就必须把对象序列化为标准格式,比如XML,但更好的方法是序列化为JSON,因为JSON表示出来就是一个字符串,可以被所有语言读取,也可以方便地存储到磁盘或者通过网络传输。JSON不仅是标准格式,并且比XML更快,而且可以直接在Web页面中读取,非常方便

JSON表示的对象就是标准的JavaScript语言的对象,JSONPython内置的数据类型对应如下:

JSON类型

Python类型

{}

dict

[]

list

"string"

'str'u'unicode'

1234.56

intfloat

true/false

True/False

null

None

Python内置的json模块提供了非常完善的Python对象到JSON格式的转换。我们先看看如何把Python对象变成一个JSON

import json

d=dict(name='Bob',age=20,score=80)

print json.dumps(d)

 

dumps()方法返回一个str,内容就是标准的JSON。类似的,dump()方法可以直接把JSON写入一个file-like Object

要把JSON反序列化为Python对象,用loads()或者对应的load()方法,前者把JSON的字符串反序列化,后者从file-like Object中读取字符串并反序列化:

import json

json_str='{"age": 20, "score": 88, "name": "Bob"}'

print json.loads(json_str)

有一点需要注意,就是反序列化得到的所有字符串对象默认都是unicode而不是str。由于JSON标准规定JSON编码是UTF-8,所以我们总是能正确地在PythonstrunicodeJSON的字符串之间转换。

JSON进阶

Pythondict对象可以直接序列化为JSON{},不过,很多时候,我们更喜欢用class表示对象,比如定义Student类,然后序列化:

import json



class Student(object):

    def __init__(self, name, age, score):

        self.name = name

        self.age = age

        self.score = score



s = Student('Bob', 20, 88)

print(json.dumps(s, default=lambda obj: obj.__dict__))

因为通常class的实例都有一个__dict__属性,它就是一个dict,用来存储实例变量。也有少数例外,比如定义了__slots__class

同样的道理,如果我们要把JSON反序列化为一个Student对象实例,loads()方法首先转换出一个dict对象,然后,我们传入的object_hook函数负责把dict转换为Student实例:

import json



class Student(object):

    def __init__(self, name, age, score):

        self.name = name

        self.age = age

        self.score = score



s = Student('Bob', 20, 88)

def dict2student(d):

    return Student(d['name'], d['age'], d['score'])

json_str = '{"age": 20, "score": 88, "name": "Bob"}'

print(json.loads(json_str, object_hook=dict2student))

Python语言特定的序列化模块是pickle,但如果要把序列化搞得更通用、更符合Web标准,就可以使用json模块。

json模块的dumps()loads()函数是定义得非常好的接口的典范。当我们使用时,只需要传入一个必须的参数。但是,当默认的序列化或反序列机制不满足我们的要求时,我们又可以传入更多的参数来定制序列化或反序列化的规则,既做到了接口简单易用,又做到了充分的扩展性和灵活性。

进程和线程:
一种是启动多个进程,每个进程虽然只有一个线程,但多个进程可以一块执行多个任务。

还有一种方法是启动一个进程,在一个进程内启动多个线程,这样,多个线程也可以一块执行多个任务。

当然还有第三种方法,就是启动多个进程,每个进程再启动多个线程,这样同时执行的任务就更多了,当然这种模型更复杂,实际很少采用。

总结一下就是,多任务的实现有3种方式:

  • 多进程模式;
  • 多线程模式;
  • 多进程+多线程模式

多进程:

Unix/Linux操作系统提供了一个fork()系统调用,它非常特殊。普通的函数调用,调用一次,返回一次,但是fork()调用一次,返回两次,因为操作系统自动把当前进程(称为父进程)复制了一份(称为子进程),然后,分别在父进程和子进程内返回。

子进程永远返回0,而父进程返回子进程的ID。这样做的理由是,一个父进程可以fork出很多子进程,所以,父进程要记下每个子进程的ID,而子进程只需要调用getppid()就可以拿到父进程的ID

Pythonos模块封装了常见的系统调用,其中就包括fork,可以在Python程序中轻松创建子进程:

multiprocessing模块提供了一个Process类来代表一个进程对象,下面的例子演示了启动一个子进程并等待其结束:

# -*- coding: UTF-8 -*-
from multiprocessing import Process
import os

# 子进程要执行的代码
def run_proc(name):
   
print 'Run child process %s (%s)...' % (name, os.getpid())

if __name__=='__main__':
   
print 'Parent process %s.' % os.getpid()
    p = Process(
target=run_proc, args=('test',))
   
print 'Process will start.'
   
p.start()
    p.join()
   
print 'Process end.'

创建子进程时,只需要传入一个执行函数和函数的参数,创建一个Process实例,用start()方法启动,join()方法可以等待子进程结束后再继续往下运行,通常用于进程间的同步。

Pool(进程池)

from multiprocessing import Pool
import os, time, random

def long_time_task(name):
   
print 'Run task %s (%s)...' % (name, os.getpid())
    start = time.time()
    time.sleep(random.random() *
3)
    end = time.time()
   
print 'Task %s runs %0.2f seconds.' % (name, (end - start))

if __name__=='__main__':
   
print 'Parent process %s.' % os.getpid()
    p = Pool()
   
for i in range(5):
        p.apply_async(long_time_task
, args=(i,))
   
print 'Waiting for all subprocesses done...'
   
p.close()
    p.join()
   
print 'All subprocesses done.'

Pool对象调用join()方法会等待所有子进程执行完毕,调用join()之前必须先调用close(),调用close()之后就不能继续添加新的Process了。

进程间通信

Process之间需要通信,操作系统提供了很多机制来实现进程间的通信。Pythonmultiprocessing模块包装了底层的机制,提供了QueuePipes等多种方式来交换数据。

# -*-coding:UTF-8-*-
from multiprocessing import Process, Queue
import os, time, random

# 写数据进程执行的代码:
def write(q):
   
for value in ['A', 'B', 'C']:
       
print 'Put %s to queue...' % value
        q.put(value)
        time.sleep(random.random())


# 读数据进程执行的代码:
def read(q):
   
while True:
        value = q.get(
True)
       
print 'Get %s from queue.' % value

if __name__=='__main__':
   
# 父进程创建Queue,并传给各个子进程:
   
q = Queue()
    pw = Process(
target=write, args=(q,))
    pr = Process(
target=read, args=(q,))
   
# 启动子进程pw,写入:
   
pw.start()
   
# 启动子进程pr,读取:
   
pr.start()
   
# 等待pw结束:
   
pw.join()
   
# pr进程里是死循环,无法等待其结束,只能强行终止:
   
pr.terminate()

多线程:

多任务可以由多进程完成,也可以由一个进程内的多线程完成。

线程是操作系统直接支持的执行单元,因此,高级语言通常都内置多线程的支持,Python也不例外,并且,Python的线程是真正的Posix Thread,而不是模拟出来的线程。

Python的标准库提供了两个模块:threadthreadingthread是低级模块,threading是高级模块,对thread进行了封装。绝大多数情况下,我们只需要使用threading这个高级模块

启动一个线程就是把一个函数传入并创建Thread实例,然后调用start()开始执行:

#-*-coding=UTF-8-*_
import time, threading

# 新线程执行的代码:
def loop():
   
print 'thread %s is running...' % threading.current_thread().name
    n =
0
   
while n < 5:
        n = n +
1
       
print 'thread %s >>> %s' % (threading.current_thread().name, n)
        time.sleep(
1)
   
print 'thread %s ended.' % threading.current_thread().name

print 'thread %s is running...' % threading.current_thread().name
t = threading.Thread(
target=loop, name='LoopThread')
t.start()
t.join()

print 'thread %s ended.' % threading.current_thread().name

由于任何进程默认就会启动一个线程,我们把该线程称为主线程,主线程又可以启动新的线程,Pythonthreading模块有个current_thread()函数,它永远返回当前线程的实例。主线程实例的名字叫MainThread,子线程的名字在创建时指定,我们用LoopThread命名子线程。名字仅仅在打印时用来显示,完全没有其他意义,如果不起名字Python就自动给线程命名为Thread-1Thread-2……

Lock

 

猜你喜欢

转载自blog.csdn.net/yige__cxy/article/details/81170199