[转]用PB编写多线程应用程序

开发者在线 Builder.com.cn 更新时间:2007-10-23作者:经乾 来源:Yesky

本文关键词: PB 多线程 应用程序

多线程是指一个程序内部同时执行的多个流程,与单线程相比,它至少有两个方面的优点:第一,它可以更好地利用系统资源,比如CPU,若一个线程因I/O操作受阻,另一个线程仍可利用CPU来执行;第二,它更好地满足了客户的需求,因为挑剔的客户希望你开发的程序在显示动画的同时还能播放音乐、显示文件、下载网络文件等,这是单线程应用程序无法完成的。目前,支持多线程的开发工具主要有:Java、VC、Delphi和C++Builder。

  多线程强调的是一个进程内部有多个流程在同时执行,同时执行的概念相当于PB分布式计算中的异步处理。也就是说,只要我们在一个程序内部实现了异步处理,就相当于实现了多线程。分析至此,下一步该怎么做就不言而喻了:在一个应用程序内部实现分布式计算,使用服务器推送技术,异步执行共享对象中的函数。

  下面就编写一个Demo,介绍在PB中如何利用分布式计算技术来开发多线程应用程序,其程序结构如图1所示。uo—thread1和uo—thread2是类用户对象uo—thread的两个实例,uo—thread的功能是给实例变量li—Count自加5秒,自加的同时向中间对象uo—argv1和uo—argv2发送ue—thread消息。两个中间对象的功能是把ue—thread消息转化为ue—thread1和ue—thread2后发给主窗口w—main。

fangz20705_03.jpg

图1 程序结构

  主窗口接收到ue—thread1消息后,显示uo—thread1中自加变量的值,接收到ue—thread2后,显示uo—thread2中自加变量的值。在单线程中,uo—thread1先执行,5秒钟后uo—thread2开始执行,因此,前5秒内主窗口只能收到ue—thread1消息,后5秒内只能收到ue—thread2消息。在多线程中,uo—thread1和uo—thread2同时执行,因此w—main可以不断地收到ue—thread1和ue—thread2消息。

  1.设计用户对象uo—thread

  新建一个类用户对象,命名为uo—thread,添加如下两个实例变量:

  NonVisualObject inv—arg

  Long li—Count //自加变量

  创建如下三个用户函数:

  ⑴uf—start(),功能是:完成自加5秒并向中间对象发送ue—thread消息。脚本为:

  Time t0

  t0= Now() //获取当前时间

  Do While SecondsAfter(t0,Now())〈=5

  li—Count++//实例变量自加5秒

  inv—arg.TriggerEvent(′ue—thread′)

  //向中间对象发送ue—thread消息

  Loop

  ⑵uf—getcount(),功能是:获取自加变量的瞬间值。其脚本为:

  Return li—Count //返回实例变量

  ⑶uf—setparent(NonVisualObject nv—arg),调用时,使用中间对象为参数。脚本为:

  inv—arg=nv—arg

  //用中间对象给实例变量赋值

  2.设计中间对象uo—argv1和uo—argv2

  新建一个类用户对象,命名为uo—argv1,添加如下实例变量:

  Window win—arg

  创建用户函数uf—setparent(Window w—argv),调用时,用主窗口作为参数。其脚本为:

  win—arg=w—argv

  //用主窗口对象给实例变量赋值

  声明用户事件ue—thread,用于对uo—thread1发出的uo—thread消息进行响应,其脚本为:

  win—arg.TriggerEvent(″ue—thread1″)

  //向主窗口发送ue—thread1消息

  uo—argv2和uo—argv1完全一样,只需将uo—argv1中的ue—thread1改为uo—thread2即可。

  3.设计主窗口w—main

  主窗口外观如图2所示,凹下的三个控件分别为:st—thread1、st—thread2、st—time,分别用于显示uo—thread1和uo—thread2中自加变量的当前瞬间值和系统时间。声明两个用户事件:ue—thread1和ue—thread2,分别用于对中间对象发送来的ue—thread1和ue—thread2消息进行响应。ue—thread1事件处理代码为:

fangz20705_04.jpg

  图2 主窗口

  st—thread1.Text=String(uo—thread1.uf—getcount())

  ue—thread2事件的处理代码为:

  st—thread2.Text=String(uo—thread2.uf—getcount())

  w—main的Open事件代码为:

  uo—arg1 = Create uo—argv1

  //初始化中间对象的一个实例

  uo—arg2 = Create uo—argv2

  uo—arg1.uf—setParent(This) //给中间对象的实例变量赋值

  uo—arg2.uf—setParent(This)

  Timer(1) //启动定时器

  Timer事件处理代码为:

  st—time.Text=String(Now())//显示当前时间

  在“单线程”的Clicked事件中加入下列代码:

  SharedObjectUnRegister(″object1″)

  //注销先前注册过的共享对象object1

  SharedObjectUnRegister(″object2″)

  //注销先前注册过的共享对象object2

  If IsValid(uo—thread1) Then Destroy uo—thread1  //若uo—thread1已经存在,先删除

  If IsValid(uo—thread2) Then Destroy uo—thread2

  uo—thread1=Create uo—thread

  //初始化uo—thread1

  uo—thread2=Create uo—thread

  uo—thread1.uf—setparent(uo—argv1) //用中间对象给uo—thread1中的实例变量赋值

  uo—thread2.uf—setparent(uo—argv2)

  uo—thread1.Post uf—start() //执行uo—thread1中的uf—start()函数,完成后才执行下一句

  uo—thread2.Post uf—start()

  //上一条语句执行完成后才会执行

  在“多线程”的Clicked事件中加入下列代码:

  SharedObjectRegister(″uo—thread″,″object1″)

  //将uo—thread对象注册为object1

  SharedObjectRegister(″uo—thread″,″object2″)

  //将uo—thread对象注册为object2

  SharedObjectGet(″object1″,uo—thread1)

  //用uo—thread1引用共享对象object1

  SharedObjectGet(″object2″,uo—thread2)

  //用uo_thread2引用共享对象object2

  uo—thread1.uf—setparent(uo—argv1) //用中间对象给uo—thread1中的实例变量赋值

  uo—thread2.uf—setparent(uo—argv2)

  uo—thread1.Post uf—start() //利用服务器推送技术,异步调用共享对象中的uf—start()

  uo—thread2.Post uf—start()//相当于启动线程

  4.执行

  执行程序后,图2就是点击“多线程”后执行的一瞬间。

 

来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/1845770/viewspace-612369/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/1845770/viewspace-612369/

注意事项: 1、主进程传入的对象或变量不管是不是ref线程内操作传入的引用都会影响主进程的对象或变量,且对象只能是nonvisualobject类型的。 2、基础变量如long等等都不能传引用ref会运行会报错 3、SharedObjectUnregister只是把SharedObjectDirectory中的去掉,实际内存不会释放必须destroy 4、主进程不能直接访问线程中的变量和对象,可以通过处理类私有的办法处理。 5、千万注意释放线程的时候一定要把线程里面的资源释放完,不然百分百卡死。比如一个线程里面有一个timing的计时器,如果不先stop(),直接destroy,百分百卡死。如果连接数据库或者其他接口时千万注意了!!!千万要在uf_stop()(此例子中的释放预留方法)里面把所有的资源都释放干净,资源都释放干净,源都释放干净,都释放干净,释放干净,放干净,干净,净…… 大体设计思路: 1、在主进程中建立一个“任务信息类”数组,其中包含“任务线程类”,一个任务对应一个线程。 2、在主进程中建立一个“任务管理类”,负责处理任务信息类。 简单举例: 1、新建1个“任务管理类”,再新建N“任务信息类”,将“任务信息类”赋值完成加入“任务管理类”,并创建一个“任务线程类”,此时线程开始running。 2、“任务线程类”中有一个内部timing类,监控自己是否执行完成,会改标志。“任务管理类”也有一个timing监控“任务信息类”和“任务线程类”的情况,把完成的结束。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值