我正试图从子进程返回值,但不幸的是,这些值是不可访问的。因此,我成功地在threads模块中使用了全局变量,但在使用多处理模块时,无法检索在子流程中完成的更新。我希望我少了点什么。
最后打印的结果始终与给定变量dataDV03
和dataDV04
的初始值相同。子进程正在更新这些全局变量,但这些全局变量在父进程中保持不变。
import multiprocessing
# NOT ABLE to get python to return values in passed variables.
ants = ['DV03', 'DV04']
dataDV03 = ['', '']
dataDV04 = {'driver': '', 'status': ''}
def getDV03CclDrivers(lib): # call global variable
global dataDV03
dataDV03[1] = 1
dataDV03[0] = 0
# eval( 'CCL.' + lib + '.' + lib + '( "DV03" )' ) these are unpicklable instantiations
def getDV04CclDrivers(lib, dataDV04): # pass global variable
dataDV04['driver'] = 0 # eval( 'CCL.' + lib + '.' + lib + '( "DV04" )' )
if __name__ == "__main__":
jobs = []
if 'DV03' in ants:
j = multiprocessing.Process(target=getDV03CclDrivers, args=('LORR',))
jobs.append(j)
if 'DV04' in ants:
j = multiprocessing.Process(target=getDV04CclDrivers, args=('LORR', dataDV04))
jobs.append(j)
for j in jobs:
j.start()
for j in jobs:
j.join()
print 'Results:\n'
print 'DV03', dataDV03
print 'DV04', dataDV04
我不能发表我的问题,所以将尝试编辑原件。
以下是不可拾取的对象:
In [1]: from CCL import LORR
In [2]: lorr=LORR.LORR('DV20', None)
In [3]: lorr
Out[3]: <CCL.LORR.LORR instance at 0x94b188c>
这是使用多处理时返回的错误。池将实例返回到父级:
Thread getCcl (('DV20', 'LORR'),)
Process PoolWorker-1:
Traceback (most recent call last):
File "/alma/ACS-10.1/casa/lib/python2.6/multiprocessing/process.py", line 232, in _bootstrap
self.run()
File "/alma/ACS-10.1/casa/lib/python2.6/multiprocessing/process.py", line 88, in run
self._target(*self._args, **self._kwargs)
File "/alma/ACS-10.1/casa/lib/python2.6/multiprocessing/pool.py", line 71, in worker
put((job, i, result))
File "/alma/ACS-10.1/casa/lib/python2.6/multiprocessing/queues.py", line 366, in put
return send(obj)
UnpickleableError: Cannot pickle <type 'thread.lock'> objects
In [5]: dir(lorr)
Out[5]:
['GET_AMBIENT_TEMPERATURE',
'GET_CAN_ERROR',
'GET_CAN_ERROR_COUNT',
'GET_CHANNEL_NUMBER',
'GET_COUNT_PER_C_OP',
'GET_COUNT_REMAINING_OP',
'GET_DCM_LOCKED',
'GET_EFC_125_MHZ',
'GET_EFC_COMB_LINE_PLL',
'GET_ERROR_CODE_LAST_CAN_ERROR',
'GET_INTERNAL_SLAVE_ERROR_CODE',
'GET_MAGNITUDE_CELSIUS_OP',
'GET_MAJOR_REV_LEVEL',
'GET_MINOR_REV_LEVEL',
'GET_MODULE_CODES_CDAY',
'GET_MODULE_CODES_CMONTH',
'GET_MODULE_CODES_DIG1',
'GET_MODULE_CODES_DIG2',
'GET_MODULE_CODES_DIG4',
'GET_MODULE_CODES_DIG6',
'GET_MODULE_CODES_SERIAL',
'GET_MODULE_CODES_VERSION_MAJOR',
'GET_MODULE_CODES_VERSION_MINOR',
'GET_MODULE_CODES_YEAR',
'GET_NODE_ADDRESS',
'GET_OPTICAL_POWER_OFF',
'GET_OUTPUT_125MHZ_LOCKED',
'GET_OUTPUT_2GHZ_LOCKED',
'GET_PATCH_LEVEL',
'GET_POWER_SUPPLY_12V_NOT_OK',
'GET_POWER_SUPPLY_15V_NOT_OK',
'GET_PROTOCOL_MAJOR_REV_LEVEL',
'GET_PROTOCOL_MINOR_REV_LEVEL',
'GET_PROTOCOL_PATCH_LEVEL',
'GET_PROTOCOL_REV_LEVEL',
'GET_PWR_125_MHZ',
'GET_PWR_25_MHZ',
'GET_PWR_2_GHZ',
'GET_READ_MODULE_CODES',
'GET_RX_OPT_PWR',
'GET_SERIAL_NUMBER',
'GET_SIGN_OP',
'GET_STATUS',
'GET_SW_REV_LEVEL',
'GET_TE_LENGTH',
'GET_TE_LONG_FLAG_SET',
'GET_TE_OFFSET_COUNTER',
'GET_TE_SHORT_FLAG_SET',
'GET_TRANS_NUM',
'GET_VDC_12',
'GET_VDC_15',
'GET_VDC_7',
'GET_VDC_MINUS_7',
'SET_CLEAR_FLAGS',
'SET_FPGA_LOGIC_RESET',
'SET_RESET_AMBSI',
'SET_RESET_DEVICE',
'SET_RESYNC_TE',
'STATUS',
'_HardwareDevice__componentName',
'_HardwareDevice__hw',
'_HardwareDevice__stickyFlag',
'_LORRBase__logger',
'__del__',
'__doc__',
'__init__',
'__module__',
'_devices',
'clearDeviceCommunicationErrorAlarm',
'getControlList',
'getDeviceCommunicationErrorCounter',
'getErrorMessage',
'getHwState',
'getInternalSlaveCanErrorMsg',
'getLastCanErrorMsg',
'getMonitorList',
'hwConfigure',
'hwDiagnostic',
'hwInitialize',
'hwOperational',
'hwSimulation',
'hwStart',
'hwStop',
'inErrorState',
'isMonitoring',
'isSimulated']
In [6]:
当您使用
multiprocessing
打开第二个进程时,会创建一个具有自己全局状态的全新Python实例。该全局状态不共享,因此子进程对全局变量所做的更改对父进程不可见。此外,} provides 的所有对象。相关引文(我强调):
multiprocessing
提供的大多数抽象都使用pickle来传输数据。使用代理must be pickleable;传输的所有数据,包括^{以及(在
Manager
部分中):Queue
也需要可pickle的数据;文档没有这么说,但是快速测试证实了这一点:输出:
如果您真的不能pickle数据,那么可能为您工作的一种方法是找到一种将其存储为
ctype
对象的方法;然后对内存的引用可以是passed to a child process。这对我来说似乎很狡猾,我从来没有做过。但这对你来说可能是个解决办法。考虑到您的更新,您似乎需要更多地了解
LORR
的内部内容。LORR
是类吗?你能从中派生出来吗?是不是别的东西的子类?什么是MRO?(尝试LORR.__mro__
并发布输出(如果它有效的话)如果它是一个纯python对象,就可以对它进行子类化,创建一个__setstate__
和一个__getstate__
来启用pickling。另一种方法可能是找出如何从
LORR
实例中获取相关数据并通过一个简单的字符串传递它。既然你说你真的只想调用对象的方法,为什么不直接使用Queue
来来回发送消息呢?换句话说,类似这样的东西(示意性地):@DBlas给了你一个快速的url和对Manager类的引用,但是我认为它仍然有点模糊,所以我认为这可能有助于你看到它的应用。。。
因为多处理实际上使用单独的进程,所以不能简单地共享全局变量,因为它们将位于内存中完全不同的“空间”中。你在一个进程下对一个全局所做的事情不会反映在另一个进程中。尽管我承认,从你的角度来看,它看起来很混乱,它就生活在同一段代码中,所以“为什么那些方法不应该访问全局呢?”?你很难相信他们会在不同的进程中运行。
Manager class作为数据结构的代理,可以在进程之间来回传递信息。您将要做的是从管理器创建一个特殊的dict和list,将它们传递到您的方法中,并在本地对它们进行操作。
不可pickle的数据
对于specialize LORR对象,您可能需要创建一个代理来表示实例的可拾取状态。
不是非常健壮或者测试过很多,但是给了你这个想法。
使用
multiprocess
时,在进程之间传递对象的唯一方法是使用Queue
或Pipe
;全局变量不共享。对象必须是可pickle的,因此multiprocess
在这里对您没有帮助。相关问题 更多 >
编程相关推荐