Python入门教学——多进程和多线程

目录

一、线程和进程

二、创建多个线程

三、锁

四、线程间通讯

五、创建多个进程

六、进程间通讯

七、总结


一、线程和进程

1、线程和进程的基本概念
  • 进程是资源分配的最小单位,线程是CPU调度的最小单位。
  • 进程:相当于是公司的老板,他自己本身不干活,但为干活的员工提供资源、平台。
  • 线程:相当于是公司的员工,真正干活的人就是这些员工。
2、线程和进程的关系
  • 一个线程只能属于一个进程,一个进程可以有多个进程,每个进程至少有一个线程(主线程)。
  • 资源分配给进程,一个进程中的所有线程将共享该进程中所有资源。
  • 线程在执行过程中,需要协作同步,不同进程间的线程需要利用消息向步机制来实现通讯。
  • 真正执行任务的是线程,它是进程内可调度的实体。
  • 创建进程要比创建线程消耗更多的计算机资源。
  • 进程间不会相互影响,而相同进程内部如果有一个线程出现异常,则其他所有线程将全部被阻塞。
  • 同一进程内不同线程使用同一个资源时需要加锁。
3、串行、并行和并发
  • 串行:做完一件事再完成另一件事。完成任务时间是所有单个任务完成时间的总和。
  • 并行:两个或多个事件在同一时刻发生,相当于多个人同时完成多件事,是物理意义上的并行。
  • 并发:两个或多个事件在同一时间间隔内发生相当于一个人同时做多件事,是逻辑意义上的并行。

二、创建多个线程

1、线程相关的模块
  • _thread:该模块是在python3之前thread模块的重命名,它是比较底层的模块,所以一般不会在代码中直接使用。
  • threading:python3 之后的线程模块,编程中一般都使用这个模块来创建线程。
2、创建线程
2.1、通过Thread类构造器来创建新线程
  • 使用threading.Thread直接生成一个线程,并且定义一个函数传给target,其中包含了该线程要完成的任务。
  • 创建两个线程,分别去访问百度和搜狐的网站。
    • import threading
      import urllib.request
      # 访问网站
      def get_web(url):  # 访问网站
              user_agent = '在从自己的浏览器里找'
              headers = {
                      "User-Agent": user_agent
              }
              # 构造Request对象,以便我们设置http请求中的header信息
              req = urllib.request.Request(url, headers=headers)
              resp = urllib.request.urlopen(req)
              print(resp.read().decode()[:50])
      if __name__ == '__main__':
              t1=threading.Thread(target=get_web,args=('https://www.baidu.com/',))  # target是需要做的事情,利用args传参(元组形式)
              t2=threading.Thread(target=get_web,args=('https://www.sohu.com/',))
              t1.start()  # 启动线程,就会执行target的内容
              t2.start()
              t1.join()  # 等待子线程结束后,才执行主线程
              t2.join()
              print("程序运行结束!")  # 主线程的代码
    • 【注】join方法:让子线程阻塞主线程的执行过程,也就是说使用了join之后,主线程会等待使用了join方法的子线程结束后再往下执行。
  • 运行结果:
2.2、通过继承于Thread类来创建新线程
  • 通过继承Thread类,重写__init__和run方法,__init__中可以添加许多自定义属性,run中是该线程要完成的任务。线程启动后,会自动执行__init__和run方法。
  • import threading
    import urllib.request
    # 新的线程类   
    class MyThread(threading.Thread):  # 继承Thread
            def __init__(self,name,url):
                    super().__init__()  # 调用父类的初始化方法
                    self.name=name  # 线程的名字
                    self.url=url
            # 启动线程后,run会被自动调用
            def run(self):  # 重写父类的run方法,定义在新的线程类里面要完成的任务
                    print(f"我是{self.name}")
                    user_agent = '在从自己的浏览器里找'
                    headers = {
                            "User-Agent": user_agent
                    }
                    # 构造Request对象,以便我们设置http请求中的header信息
                    req = urllib.request.Request(self.url, headers=headers)
                    resp = urllib.request.urlopen(req)
                    print(resp.read().decode()[:50])
    if __name__ == '__main__':
            t1=MyThread("线程1","https://www.baidu.com/")
            t2=MyThread("线程2","https://www.sohu.com/")
            t1.start()  # 启动线程
            t2.start()
            t1.join()  # 等待子线程结束后,才执行主线程
            t2.join()
            print("程序运行结束!")  # 主线程的代码
  •  运行结果:

三、锁

1.1、锁的概念
  • 锁是多线程模块里的一个对象,只有拥有这个对象(锁),才能对共享资源进行操作。
  • 作用:为了避免多个线程使用同一资源时产生错误。
1.2、GIL锁
  • GIL锁:全局解释器锁,在同一个进程中只要有一个线程获取了全局解释器(cpu)的使用权限,那么其他的线程就必须等待该线程的全局解释器(cpu)使用权消失后才能使用全局解释器(cpu),即使多个线程直接不会相互影响在同一个进程下也只有一个线程使用cpu。
  • GIL锁释放条件:
    • 当前的执行线程在执行IO操作时,会主动放弃GIL。
    • 当前执行的线程执行了100条字节码的时候,会自动释放GIL锁。
  • 【注】GIL全局解释器锁是粗粒度的锁,必须配合线程模块中的锁才能对每个原子操作(不可再拆分的操作)进行锁定。
1.3、如何加锁
  • import threading
    num=0
    lock=threading.Lock()  # 创建锁
    def deposit():
            for i in range(1000000):
                    lock.acquire()  # 获取锁
                    global num
                    num+=1
                    lock.release()  # 释放锁
    def withdraw():
            for i in range(1000000):
                    with lock: # 自动获取锁和释放锁
                            global num
                            num-=1
    if __name__ == '__main__':
            t1=threading.Thread(target=deposit)
            t2=threading.Thread(target=withdraw)
            t1.start()
            t2.start()
            t1.join()
            t2.join()
            print(num)
  • 【注】获取锁和释放锁必须成对出现。

四、线程间通讯

  • 线程间的通信方式有很多,最常用的是利用消息队列进行通讯。 
1.1、消息队列
  • 消息队列是在消息的传输过程中保存消息的容器,主要用于不同线程间任意类型数据的共享。
  • 消息队列最经典的用法就是消费者和生成者之间通过消息管道来传递消息,消费者和生成者是不同的线程。生产者往管道中写消息,消费者从管道中读消息,且一次只允许一个线程访问管道。
1.2、常用接口
  • from queue import Queue
    q =Queue(maxsize=0)  # 初始化,maxsize=0表示队列的消息个数不受限制;maxsize>0表示存放限制
    q.get()  # 提取消息,如果队列为空会阻塞程序,等待队列消息
    q.get(timeout=1)  # 阻塞程序,设置超时时间
    q.put()  # 发送消息,将消息放入队列
1.3、演示
  • 使用生产者和消费者的案例进行演示。 
    • from queue import Queue
      import threading
      import time
      def product(q):  # 生产者
              kind = ('猪肉','白菜','豆沙')
              for i in range(3):
                      print(threading.current_thread().name,"生产者开始生产包子")
                      time.sleep(1)
                      q.put(kind[i%3])  # 放入包子
                      print(threading.current_thread().name,"生产者的包子做完了")
      def consumer(q):  # 消费者
              while True:
                      print(threading.current_thread().name,"消费者准备吃包子")
                      time.sleep(1)
                      t=q.get()  # 拿出包子
                      print("消费者吃了一个{}包子".format(t))
      if __name__=='__main__':
              q=Queue(maxsize=1)
              # 启动两个生产者线程
              threading.Thread(target=product,args=(q, )).start()
              threading.Thread(target=product,args=(q, )).start()
              # 启动一个消费者线程
              threading.Thread(target=consumer,args=(q, )).start()
  • 运行结果:

五、创建多个进程

1、以指定函数作为参数创建进程
  • 与通过Thread类构造器来创建新线程的方法类似,只不过使用的模块不同。
    • import multiprocessing  # 引入多进程模块
      import urllib.request
      # 访问网站
      def get_web(url):  # 访问网站
              user_agent = '在自己的浏览器里找'
              headers = {
                      "User-Agent": user_agent
              }
              # 构造Request对象,以便我们设置http请求中的header信息
              req = urllib.request.Request(url, headers=headers)
              resp = urllib.request.urlopen(req)
              print(resp.read().decode()[:50])
      if __name__ == '__main__':
              p1=multiprocessing.Process(target=get_web,args=('https://www.baidu.com/',))
              p2=multiprocessing.Process(target=get_web,args=('https://www.sohu.com/',))
              p1.start()  # 启动进程
              p2.start()
              p1.join()  # 等待子进程结束后,才执行主进程
              p2.join()
              print("进程操作全部执行完毕!")  # 主进程的代码
  • 运行结果:
    •  
2、继承Process类创建进程
  • 与通过继承于Thread类来创建新线程的方法类似。通过继承Process类,重写__init__和run方法,__init__中可以添加许多自定义属性,run中是该进程要完成的任务。进程启动后,会自动执行__init__和run方法。
    • import multiprocessing
      import urllib.request
      import os  # 拿到进程id
      # 新的进程类   
      class MyProcess(multiprocessing.Process):  # 继承Process
              def __init__(self,name,url):
                      super().__init__()  # 调用父类的初始化方法
                      self.name=name  # 进程的名字
                      self.url=url
              # 启动进程后,run会被自动调用
              def run(self):  # 重写父类的run方法,定义在新的进程类里面要完成的任务
                      print(f"我是{self.name}","当前进程ID为:",os.getpid(),"我的父进程ID为:",os.getppid())
                      user_agent = '在自己的浏览器里找'
                      headers = {
                              "User-Agent": user_agent
                      }
                      # 构造Request对象,以便我们设置http请求中的header信息
                      req = urllib.request.Request(self.url, headers=headers)
                      resp = urllib.request.urlopen(req)
                      print(resp.read().decode()[:50])
      if __name__ == '__main__':
              print("当前主进程的ID为:",os.getpid())
              p1=MyProcess("子进程1","https://www.baidu.com/")
              p2=MyProcess("子进程2","https://www.sohu.com/")
              p1.start()  # 启动进程
              p2.start()
              p1.join()  # 等待子进程结束后,才执行主进程
              p2.join()
              print("进程操作全部执行完毕!")  # 主进程的代码
  •  运行结果:

六、进程间通讯

  • 进程间也可以使用消息队列进行通讯。
    • import multiprocessing
      import time
      def product(q):  # 生产者
              kind = ('猪肉','白菜','豆沙')
              for i in range(3):
                      print(multiprocessing.current_process().name,"生产者开始生产包子")
                      time.sleep(1)
                      q.put(kind[i%3])  # 放入包子
                      print(multiprocessing.current_process().name,"生产者的包子做完了")
      def consumer(q):  # 消费者
              while True:
                      print(multiprocessing.current_process().name,"消费者准备吃包子")
                      time.sleep(1)
                      t=q.get()  # 拿出包子
                      print("消费者吃了一个{}包子".format(t))
      if __name__=='__main__':
              q=multiprocessing.Queue(maxsize=1)  # 创建多进程队列对象
              #  启动两个生产者进程
              p1=multiprocessing.Process(target=product,args=(q,))
              p2=multiprocessing.Process(target=product,args=(q,))
              p1.start()
              p2.start()
              #  启动一个消费者进程
              p3=multiprocessing.Process(target=consumer,args=(q,))
              p3.start()
  • 运行结果:

七、总结

1、多线程和多进程的优缺点
  • 多进程的优点:独立运行,互不影响。
  • 多进程缺点:创建进程的代价非常大。
  • 多线程优点:效率比较高,不会耗费大量资源。
  • 多线程缺点:稳定性较差,一个崩溃后会影响整个进程。
2、使用场景
  • 多进程适用场景:适合计算密集型任务。
  • 多线程适用场景:适合 IO 密集型任务,如文件读取以及爬虫等操作。
  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
第Ⅰ部分 初步了解Python 第1章 编程基础和字符串 3 1.1 编程与使用计算机的区别 3 1.1.1 编程的一致性 3 1.1.2 编程的可控性 4 1.1.3 程序要应对变化 4 1.1.4 小结 4 1.2 准备工作 4 1.2.1 在非Windows系统上安装 Python 3.1 5 1.2.2 使用Python Shell 5 1.3 开始使用Python——字符串 6 1.3.1 字符串概述 6 1.3.2 为什么需要引号 6 1.3.3 为什么有3种类型的引号 7 1.3.4 使用print()函数 7 1.3.5 理解不同的引号 8 1.4 串联两个字符串 10 1.5 用不同的方法串联字符串 11 1.6 本章小结 12 1.7 习题 13 第2章 数值与运算符 15 2.1 不同类型的数值 15 2.2 程序文件 17 2.2.1 使用不同的类型 18 2.2.2 基本算术 20 2.2.3 一些惊喜 22 2.3 使用数值 23 2.3.1 求值顺序 23 2.3.2 数值格式 24 2.3.3 可能会发生的错误 25 2.3.4 几个不寻常的例子 26 2.4 本章小结 26 2.5 习题 27 第3章 变量 29 3.1 引用数据——使用数据的 名称 29 3.1.1 使用名称修改数据 30 3.1.2 复制数据 31 3.1.3 禁用的名称以及一些规则 31 3.2 使用更多的内置类型 31 3.2.1 元组——不可更改的数据 序列 32 3.2.2 列表——可以更改的数据 序列 35 3.2.3 字典——以名称索引的分组 数据 36 3.2.4 像列表一样处理字符串 38 3.2.5 特殊类型 39 3.3 序列的其他共有属性 40 3.3.1 引用最后一个元素 40 3.3.2 序列的范围 41 3.3.3 通过附加序列增长列表 41 3.3.4 使用列表临时存储数据 42 3.3.5 处理集合 43 3.4 本章小结 44 3.5 习题 44 第Ⅱ部分 Python语言和标准库 第4章 做出决策 49 4.1 比较两个值是否相等 49 4.2 比较两个值是否不相等 51 4.3 比较两个值的大小 51 4.4 对真值和假值取反 53 4.5 观察多个比较运算的结果 54 4.6 循环 57 4.6.1 重复执行操作 57 4.6.2 终止循环 59 4.7 处理错误 62 4.8 本章小结 64 4.9 习题 65 第5章 函数 67 5.1 将程序放在单独的文件中 67 5.2 函数:在一个名称下聚集 代码 69 5.2.1 选择名称 70 5.2.2 在函数中描述函数 70 5.2.3 不同的位置相同的名称 72 5.2.4 添加注释 73 5.2.5 要求函数使用提供的值 73 5.2.6 检查参数 75 5.2.7 为参数设置默认值 77 5.2.8 在函数中调用其他函数 78 5.2.9 函数嵌套函数 80 5.2.10 用自己的词语标记错误 81 5.3 函数的层次 81 5.4 本章小结 82 5.5 习题 83 第6章 类与对象 85 6.1 考虑编程 85 6.1.1 对象的含义 85 6.1.2 已经了解的对象 85 6.1.3 展望:如何使用对象 87 6.2 定义类 87 6.2.1 如何创建对象 87 6.2.2 对象和它们的作用域 95 6.3 本章小结 98 6.4 习题 99 第7章 组织程序 101 7.1 模块 102 7.1.1 导入可用模块 102 7.1.2 通过已有模块创建新模块 102 7.1.3 从命令行开始使用模块 104 7.1.4 改变导入方式 106 7.2 包 106 7.3 模块和包 108 7.3.1 将所有内容引入当前 作用域 108 7.3.2 重新导入模块和包 109 7.4 测试模块和包的基础知识 111 7.5 本章小结 112 7.6 习题 112 第8章 文件和目录 115 8.1 文件对象 115 8.1.1 编写文本文件 116 8.1.2 向文件中追加文本 116 8.1.3 读文本文件 117 8.1.4 文件异常 119 8.2 路径和目录 119 8.3 os中的异常 119 8.3.1 路径 120 8.3.2 目录内容 122 8.3.3 获取文件信息 123 8.3.4 重命名、移动、复制和删除 文件 125 8.3.5 示例:轮换文件 126 8.3.6 创建和删除目录 127 8.3.7 通配 128 8.4 本章小结 129 8.5 习题 129 第9章 Python语言的其他特性 131 9.1 lambda和filter:简单匿名 函数 131 9.2 Map:短路循环 132 9.3 在列表中做出决策—— 列表解析 133 9.4 为循环生成迭代器 133 9.5 使用字典的特殊字符串 替换 135 9.6 重要模块 137 9.6.1 getopt——从命令行中得到 选项 137 9.6.2 使用一个以上的进程 139 9.6.3 线程——在相同的进程中 完成多个工作 141 9.7 本章小结 143 9.8 习题 144 第10章 创建模块 145 10.1 研究模块 145 10.1.1 导入模块 147 10.1.2 查找模块 147 10.1.3 理解模块 148 10.2 创建模块和包 149 10.3 使用类 150 10.3.1 定义面向对象编程 151 10.3.2 创建类 151 10.3.3 扩展已有的类 152 10.4 完成模块 153 10.4.1 定义模块特定的错误 154 10.4.2 选择导出哪些内容 154 10.4.3 为模块建立文档 155 10.4.4 测试模块 164 10.4.5 将模块作为程序运行 165 10.5 创建一个完整的模块 166 10.6 安装模块 171 10.7 本章小结 174 10.8 习题 175 第11章 文本处理 177 11.1 文本处理的用途 177 11.1.1 搜索文件 178 11.1.2 日志剪辑 179 11.1.3 邮件筛选 179 11.2 使用os模块导航文件 系统 179 11.3 使用正则表达式和re 模块 185 11.4 本章小结 189 11.5 习题 189 第Ⅲ部分 开始使用Python 第12章 测试 193 12.1 断言 193 12.2 测试用例和测试套件 195 12.3 测试装置 198 12.4 用极限编程整合 201 12.4.1 用Python实现搜索实用 程序 202 12.4.2 一个更加强大的Python 搜索 207 12.5 软件生命周期中的正规 测试 210 12.6 本章小结 210 第13章 使用Python编写GUI 213 13.1 Python的GUI编程工具箱 213 13.2 Tkinter简介 215 13.3 用Tkinter创建GUI 小组件 215 13.3.1 改变小组件的尺寸 215 13.3.2 配置小组件选项 216 13.3.3 使用小组件 217 13.3.4 创建布局 217 13.3.5 填充顺序 218 13.3.6 控制小组件的外观 219 13.3.7 单选按钮和复选框 220 13.3.8 对话框 221 13.3.9 其他小组件类型 222 13.4 本章小结 222 13.5 习题 223 第14章 访问数据库 225 14.1 使用dbm持久字典 226 14.1.1 选择dbm模块 226 14.1.2 创建持久字典 227 14.1.3 访问持久字典 228 14.1.4 dbm与关系数据库的适用 场合 230 14.2 使用关系数据库 231 14.2.1 编写SQL语句 232 14.2.2 定义表 234 14.2.3 建立数据库 235 14.3 使用Python的数据库API 237 14.3.1 下载各个模块 237 14.3.2 创建连接 238 14.3.3 使用游标 238 14.3.4 使用事务并提交结果 245 14.3.5 检查模块的功能和元 数据 246 14.3.6 处理错误 246 14.4 本章小结 247 14.5 习题 248 第15章 使用Python处理XML 249 15.1 XML的含义 249 15.1.1 层次标记语言 249 15.1.2 一组标准 251 15.2 模式/DTD 251 15.2.1 文档模型的用途 251 15.2.2 是否需要文档模型 252 15.3 文档类型定义 252 15.3.1 DTD示例 252 15.3.2 DTD不完全是XML 253 15.3.3 DTD的局限性 253 15.4 模式 254 15.4.1 示例模式 254 15.4.2 模式是纯粹的XML 254 15.4.3 模式具有层次 255 15.4.4 模式的其他优点 255 15.5 XPath 255 15.6 HTML是XML的子集 256 15.6.1 HTML DTD 256 15.6.2 HTMLParser 256 15.7 Python中可用的XML库 257 15.8 SAX的含义 257 15.8.1 基于流 258 15.8.2 事件驱动 258 15.8.3 DOM的含义 258 15.8.4 内存中访问 258 15.9 使用SAX或者DOM的 理由 259 15.9.1 能力权衡 259 15.9.2 内存考虑 259 15.9.3 速度考虑 259 15.10 Python中可用的SAX 和DOM解析器 259 15.10.1 xml.sax 259 15.10.2 xml.dom.minidom 260 15.11 XSLT简介 262 15.11.1 XSLT是XML 262 15.11.2 转换和格式语言 263 15.11.3 函数式、模板驱动 263 15.12 lxml简介 263 15.13 元素类 263 15.14 使用lxml解析 266 15.15 本章小结 267 15.16 习题 267 第16章 网络编程 269 16.1 理解协议 271 16.1.1 比较协议和程序语言 271 16.1.2 Internet协议栈 272 16.1.3 Internet协议简介 273 16.2 发送电子邮件 274 16.2.1 E-mail文件格式 275 16.2.2 MIME消息 276 16.2.3 使用SMTP和smtplib发送 邮件 284 16.3 检索Internet邮件 286 16.3.1 使用mailbox解析本地 邮筒 286 16.3.2 使用poplib从POP3 服务器获取邮件 288 16.3.3 使用imaplib从IMAP 服务器获取邮件 290 16.3.4 安全的POP3和IMAP 294 16.3.5 Webmail应用程序不是 E-mail应用程序 294 16.4 套接字编程 294 16.4.1 套接字简介 295 16.4.2 绑定到外部主机名 297 16.4.3 镜像服务器 298 16.4.4 镜像客户端 299 16.4.5 套接字服务器 300 16.4.6 多线程服务器 302 16.4.7 Python聊天服务器 303 16.4.8 设计Python聊天 服务器 303 16.4.9 Python聊天服务器协议 304 16.4.10 Python聊天客户端 309 16.4.11 基于select的单线程 多任务 312 16.5 其他主题 313 16.5.1 协议设计的多种考虑 313 16.5.2 对等架构 314 16.6 本章小结 314 16.7 习题 315 第17章 用C扩展编程 317 17.1 扩展模块概述 318 17.2 构建并安装扩展模块 320 17.3 从Python向C传递参数 322 17.4 从C向Python返回值 325 17.5 LAME项目 326 17.6 LAME扩展模块 330 17.7 在C代码中使用Python 对象 342 17.8 本章小结 345 17.9 习题 346 第18章 数值编程 347 18.1 Python语言中的数值 347 18.1.1 整数 348 18.1.2 长整数 348 18.1.3 浮点数 349 18.1.4 格式化数值 350 18.1.5 作为数值的字符 352 18.2 数学 353 18.2.1 算术运算 354 18.2.2 内置数学函数 355 18.3 复数 357 18.4 数组 359 18.5 本章小结 363 18.6 习题 363 第19章 Django简介 365 19.1 框架的定义以及使用框架的 理由 365 19.2 Web框架的其他功能 366 19.3 Django发展史 367 19.4 理解Django的架构 368 19.4.1 项目初始设置 368 19.4.2 创建视图 371 19.5 使用模板 373 19.6 使用模板和视图 375 19.6.1 模型 378 19.6.2 创建模型的第一步—— 配置数据库设置 379 19.7 创建模型:创建一个应用 程序 380 19.8 本章小结 382 19.9 习题 383 第20章 Web应用程序与Web 服务 385 20.1 REST:Web架构 386 20.1.1 REST的特性 386 20.1.2 REST操作 388 20.2 HTTP:REST的实际应用 388 20.2.1 可见的Web服务器 390 20.2.2 HTTP请求 393 20.2.3 HTTP响应 393 20.3 CGI:将脚本变为Web应用 程序 395 20.3.1 Web服务器与CGI脚本的 协议 397 20.3.2 CGI的特殊环境变量 397 20.3.3 通过HTML表单接收用户 输入 399 20.4 HTML表单的有限词汇 400 20.5 访问表单值的安全性 401 20.6 构建wiki 405 20.6.1 BittyWiki核心库 406 20.6.2 BittyWiki的Web接口 409 20.7 Web服务 418 20.8 REST Web 服务 419 20.8.1 REST快速入门: 在Amazon.com上寻找 廉价商品 419 20.8.2 WishListBargainFinder 简介 422 20.8.3 向BittyWiki提供 REST API 425 20.8.4 使用REST Web服务 实现搜索和替换功能 428 20.9 XML-RPC 432 20.9.1 XML-RPC请求 434 20.9.2 XML-RPC中的数据 表示 434 20.9.3 XML-RPC响应 435 20.9.4 错误处理机制 436 20.9.5 通过XML-RPC展示 BittyWiki API 437 20.9.6 基于XML-RPC Web服务 的wiki搜索和替换 440 20.10 SOAP 442 20.10.1 SOAP快速入门 442 20.10.2 SOAP请求 443 20.10.3 SOAP响应 444 20.10.4 错误处理机制 444 20.10.5 展示一个BittyWiki的 SOAP接口 445 20.10.6 基于SOAP Web服务的 wiki搜索和替换 447 20.11 为Web服务API创建 文档 449 20.11.1 人类可读的API 文档 449 20.11.2 XML-RPC自省API 450 20.11.3 WSDL 451 20.12 选择Web服务标准 455 20.13 Web服务礼仪 456 20.13.1 Web服务的使用者 礼仪 456 20.13.2 Web服务的创建者 礼仪 456 20.13.3 将Web应用程序作为 Web服务使用 456 20.14 本章小结 456 20.15 习题 457 第21章 集成Java与Python 459 21.1 在Java应用程序中编写 脚本 460 21.2 比较各种Python实现 461 21.3 安装Jython 461 21.4 运行Jython 461 21.4.1 交互地运行Jython 461 21.4.2 运行Jython脚本 463 21.4.3 控制jython脚本 463 21.4.4 创建可执行命令 464 21.5 独立运行Jython 465 21.6 打包基于Jython的应用 程序 466 21.7 集成Java和Jython 466 21.7.1 在Jython中使用 Java类 466 21.7.2 从Jython访问数据库 471 21.7.3 通过Jython编写 Java EE servlet 477 21.7.4 选择Jython开发工具 483 21.8 使用Jython进行测试 483 21.9 嵌入Jython解释器 484 21.10 处理C-Python和Jython 之间的差异 487 21.11 本章小结 488 21.12 习题 488 第Ⅳ部分 附 录 附录A 习题答案 491 附录B 在线资源 519 附录C Python 3.1的新特性 521 附录D 术语表 527

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

恣睢s

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值