在使用OpenSees对结构进行分析的过程中,recorder命令是必不可少的,使用recorder命令对结构中重要节点、单元的位移进行记录,同时记录结构的基底反力,就可以绘制结构的滞回曲线。但是在OpenSees使用recorder向文件中添加数据的时候,文件是处于锁定状态的,也就是不能在模型计算的过程中实时的获取记录的位移,因此recorder提供了另一个方法——TCP发送。通过查看recorder的研究可以发现:
1 2 3 4 5 |
recorder Node <-file $fileName> <-xml $fileName> <-binary $fileName> <-tcp $inetAddress $port> <-precision $nSD> <-timeSeries $tsTag> <-time> <-dT $deltaT> <-closeOnWrite> <-node $node1 $node2 ...> <-nodeRange $startNode $endNode> <-region $regionTag> -dof ($dof1 $dof2 ...) $respType' |
其中有一个-tcp的选项,根据帮助文件的描述:
1 2 |
inetAddr ip address, "xx.xx.xx.xx", of remote machine to which data is sent $port port on remote machine awaiting tcp |
需要一个IP地址和一个端口号,那这个很简单了,可以直接使用Python监听一个TCP的端口,那么就可以获得分析的数据了。因此,直接使用Python的socket模块监听一个TCP端口,这里我选择了8099端口,因为这个端口使用的软件比较少:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
# -*- coding: utf-8 -*- """ Created on Sun Nov 20 19:55:39 2016 @author: Orycho """ import socket HOST = '127.0.0.1' PORT = 8099 tcpSerSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) tcpSerSock.bind((HOST, PORT)) tcpSerSock.listen(5) (tcpCliSock, address) = tcpSerSock.accept() datarecv = tcpCliSock.recv(20) print(datarecv) |
这里先期接收20个字节的数据看看是什么,在正式监听之前需要将下面的recorder代码插入到时程分析的OpenSees代码中去:
1 |
recorder Node -tcp 127.0.0.1 8099 -time -node 504 -dof 1 2 3 disp |
这里我输出的是模型中504号节点的三个自由度位移,这个节点号根据自己的需求定义。
执行TCP监听的Python代码,并且运行时程分析的tcl代码,可以直接得到print的数据:b’\x00\x00\x00\x00\x00\x00\x10@’。这里说明一下在进行TCP传输时,OpenSees是发送的字节流,因此还需要对获得的数据进行解码,在Python中对字节流解码可以使用str.decode()的命令,但是还有一个更为方便的就是使用Struct模块:
This module performs conversions between Python values and C structs represented as Python bytes objects. This can be used in handling binary data stored in files or from network connections, among other sources. It uses Format Strings as compact descriptions of the layout of the C structs and the intended conversion to/from Python values.
直接import struct后使用unpack()函数解码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
# -*- coding: utf-8 -*- """ Created on Sun Nov 20 19:55:39 2016 @author: Orycho """ import socket from struct import Struct HOST = '127.0.0.1' PORT = 8099 tcpSerSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) tcpSerSock.bind((HOST, PORT)) tcpSerSock.listen(5) unpacker = Struct('d') (tcpCliSock, address) = tcpSerSock.accept() datarecv = tcpCliSock.recv(unpacker.size) print(unpacker.unpack(datarecv)) |
在构建Struct对象的时候,直接使用结构化代码’d’来定义,这表明传输过来的数据为64为浮点数。当然Struct支持很多种类的字节流解码:
可以根据实际传输需要进行选择,这样可以得到传输过来的第一个数据:(4.0,),其实这是表明后面数据的大小。为了更加清楚的了解传输数据的方式,可以写了一个循环进行数据的获取,看看传输过来的是什么数据:
1 2 3 4 5 |
i = 0 while i<10: datarecv = tcpCliSock.recv(unpacker.size) print(unpacker.unpack(datarecv)) i+=1 |
可以得到下面的数据:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
>>>runfile('C:/Users/Orycho/Orycho.py', wdir='C:/Users/Orycho') (4.0,) (4.0,) (0.001,) (0.02662836286939649,) (0.16768533007884542,) (0.0011595104392798314,) (4.0,) (0.002,) (0.02663361813387128,) (0.16768534734794735,) (0.001159510437405491,) (4.0,) (0.003,) (0.026644128120415136,) (0.16768538434738964,) (0.0011595104332500982,) (4.0,) (0.004,) (0.02665989196147036,) (0.1676854414368055,) |
很明显,我们选择输出的信息为当前的时间以及节点的三个自由度上的位移,因此,输出的数据可以这么理解,第一个数据为数据长度,从第二个数据开始,依次输出数据长度,时间及三个自由度上的位移,根据这样的规则,我们编写一个简单的循环来存储所有的数据:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 |
# -*- coding: utf-8 -*- """ Created on Sun Nov 20 19:55:39 2016 @author: Orycho """ import socket from struct import Struct HOST = '127.0.0.1' PORT = 8099 tcpSerSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) tcpSerSock.bind((HOST, PORT)) tcpSerSock.listen(5) unpacker = Struct('d') (tcpCliSock, address) = tcpSerSock.accept() datarecv = tcpCliSock.recv(unpacker.size) DataSize = unpacker.unpack(datarecv)[0] data = [] i=0 while i<100: datarecv = tcpCliSock.recv(unpacker.size) if len(datarecv) != unpacker.size: continue if unpacker.unpack(datarecv)[0] ==DataSize : if data != []: print(data) data= [] else: data.append(unpacker.unpack(datarecv)[0]) i+=1 |
这样可以得到下面的数据:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
>>>runfile('C:/Users/Orycho/Orycho.py', wdir='C:/Users/Orycho') [0.001, 0.02662836286939649, 0.16768533007884542, 0.0011595104392798314] [0.002, 0.02663361813387128, 0.16768534734794735, 0.001159510437405491] [0.003, 0.026644128120415136, 0.16768538434738964, 0.0011595104332500982] [0.004, 0.02665989196147036, 0.1676854414368055, 0.0011595104267354218] [0.005, 0.026680908754954664, 0.16768551849102625, 0.001159510417892269] [0.006, 0.026707177623336277, 0.1676856156783133, 0.0011595104066926946] [0.007, 0.026738697682774034, 0.16768573301740494, 0.0011595103931363285] [0.008, 0.02677546805855007, 0.16768587062885457, 0.0011595103772075665] [0.009000000000000001, 0.026817487877874878, 0.16768602859199622, 0.0011595103588957188] [0.010000000000000002, 0.026864756272776964, 0.1676862070259561, 0.0011595103381862616] [0.011000000000000003, 0.0269172723793958, 0.16768640604339596, 0.0011595103150641997] [0.012000000000000004, 0.026975035337699335, 0.16768662577665835, 0.0011595102895130339] [0.013000000000000005, 0.027038044292212134, 0.16768686636266025, 0.001159510261515003] [0.014000000000000005, 0.027106298391222153, 0.16768712795137092, 0.0011595102310509792] [0.015000000000000006, 0.027179796787572715, 0.1676874107007538, 0.001159510198100731] [0.016000000000000007, 0.02725853863801898, 0.16768771477951422, 0.0011595101626424838] [0.017000000000000008, 0.027342523103821525, 0.1676880403653035, 0.0011595101246535124] [0.01800000000000001, 0.027431749350301772, 0.16768838764559219, 0.0011595100841094957] [0.01900000000000001, 0.027526216547265208, 0.16768875681695575, 0.00115951004098519] |
到此成功获取recorder传输的TCP数据,由于TCP通信相比于从文件中获取数据来说更加直接,也更加适合与远程操作和分布计算,因此这样的方式更加有效率,根据这样的思路可以编写一个循环监听8099端口的代码,下面的代码是我修改了OpenSees论坛上的代码,用来循环获取数据,这样可以使用Python开启子线程一直监听8099端口,一旦接收到数据,主线程可以进行接下来的工作:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 |
# -*- coding: utf-8 -*- """ Created on Fri Nov 20 21:49:08 2016 @author: orycho """ import socket from struct import Struct HOST = '127.0.0.1' PORT = 8099 tcpSerSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) tcpSerSock.bind((HOST, PORT)) tcpSerSock.listen(5) unpacker = Struct('d') while True: print('waiting for connection...') (tcpCliSock, address) = tcpSerSock.accept() print('Connected from:',address) datarecv = tcpCliSock.recv(unpacker.size) DataSize = unpacker.unpack(datarecv)[0] print('Data Size == {0}'.format(DataSize)) SizeReceived = 0 Data = [] while True: try: datarecv = tcpCliSock.recv(unpacker.size) if SizeReceived > 0: Data.append(unpacker.unpack(datarecv)[0]) if SizeReceived == DataSize: SizeReceived = 0 print(Data) Data = [] else: SizeReceived += 1 except: print('disconnect from:', address) tcpCliSock.close() # 退出 break tcpSerSock.close() |
需要注意的是,在本机测试上面的代码是没有任何问题的,但是在联机测试的过程中,当recorder的节点数量过大(数百个节点自由度)时会产生丢包现象。这样导致的结果就是采用Struct解码的时候会出现错误数据,常见的问题就是数据扩大或者缩小至非正常数据,比如5.8e+312或者1.4e-217这样的数据,所以如果在多台电脑之间使用TCP传输数据,可以将安装OpenSees的A机器编写一个Python脚本接收OpenSees的数据,根据带宽采用合适的数据流传输至B机器。这样可以保证传输比较可靠。
下载本文源代码:点击下载
非常有意思的文章,手动点赞