[swmm]调用存储数据的“结构指针数组”

Posted kimilin

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[swmm]调用存储数据的“结构指针数组”相关的知识,希望对你有一定的参考价值。

目的:

  swmm读取.inp文件后,会进行一定计算(如计算conduit.slope)并将信息存储在各个指针变量中,如:TConduit* Conduit等。通过在python中调用该指针,能更快捷的编程。

 思路步骤:

  1. 根据swmm5.c,编写Swmm5Extend.c;
    1. 删去读取文件后的模拟函数;
    2. 重写便于python使用的swmm5_extend_run();
    3. 编写将object数量存储在全局变量中的函数count_object();
  2. 将函数与全局变量都放在Swmm5Extend.h中;
  3. 编写swig输入文件swmm.i
  4. 编写swig编译文件setup.py
  5. 在powershell中运行

具体代码:

  Swmm5Extend.c

  1 #include <stdlib.h>
  2 #include <string.h>
  3 #include <math.h>
  4 #include <time.h>
  5 #include <float.h>
  6 #include "swmm5headers.h"//include #include <stdio.h>
  7 #include "swmm5swmm5.h"
  8 #include "Swmm5Extend.h"
  9 
 10 //#define _CRT_SECURE_NO_DEPRECATE
 11 #define  MAX_EXCEPTIONS 100            // max. number of exceptions handled
 12 
 13 
 14 //-----------------------------------------------------------------------------
 15 //  Unit conversion factors
 16 //-----------------------------------------------------------------------------
 17 /*
 18 const double Ucf[10][2] =
 19 {//  US      SI
 20 {43200.0,   1097280.0 },         // RAINFALL (in/hr, mm/hr --> ft/sec)
 21 {12.0,      304.8     },         // RAINDEPTH (in, mm --> ft)
 22 {1036800.0, 26334720.0},         // EVAPRATE (in/day, mm/day --> ft/sec)
 23 {1.0,       0.3048    },         // LENGTH (ft, m --> ft)
 24 {2.2956e-5, 0.92903e-5},         // LANDAREA (ac, ha --> ft2)
 25 {1.0,       0.02832   },         // VOLUME (ft3, m3 --> ft3)
 26 {1.0,       1.608     },         // WINDSPEED (mph, km/hr --> mph)
 27 {1.0,       1.8       },         // TEMPERATURE (deg F, deg C --> deg F)
 28 {2.203e-6,  1.0e-6    },         // MASS (lb, kg --> mg)
 29 {43560.0,   3048.0    }          // GWFLOW (cfs/ac, cms/ha --> ft/sec)
 30 };
 31 const double Qcf[6] =                  // Flow Conversion Factors:
 32 { 1.0,     448.831, 0.64632,     // cfs, gpm, mgd --> cfs
 33   0.02832, 28.317,  2.4466 };    // cms, lps, mld --> cfs
 34 */
 35 //-----------------------------------------------------------------------------
 36 //  Shared variables
 37 //-----------------------------------------------------------------------------
 38 static int  IsOpenFlag;           // TRUE if a project has been opened
 39 static int  IsStartedFlag;        // TRUE if a simulation has been started
 40 static int  SaveResultsFlag;      // TRUE if output to be saved to binary file
 41 static int  ExceptionCount;       // number of exceptions handled
 42 static int  DoRunoff;             // TRUE if runoff is computed
 43 static int  DoRouting;            // TRUE if flow routing is computed
 44 
 45 
 46 int main(){
 47     char *inputFile= "iofiles\\dalizi_rain1.inp";//Kimi: can edit by hand
 48     char *reportFile= "iofiles\\swmm5_extend.rpt";
 49     char *binaryFile;
 50     char blank[] = "";
 51     time_t start;
 52     double runTime;
 53 
 54     // --- initialize flags
 55     //Kimi: put it in swmm5_extend_run()
 56 
 57     // --- check for proper number of command line arguments
 58     start = time(0);
 59     if (0) ;
 60     else
 61     {
 62         binaryFile = blank;
 63         writecon(FMT02);
 64 
 65         // --- run SWMM
 66         swmm5_extend_run(inputFile, reportFile, binaryFile);
 67 
 68         // Display closing status on console
 69         runTime = difftime(time(0), start);
 70         //sprintf(Msg, "

... EPA-SWMM completed in %.2f seconds.", runTime);
 71         writecon(Msg);
 72         if (ErrorCode) writecon(FMT03);
 73         else if (Warnings) writecon(FMT04);                               //(5.1.011)
 74         else                    writecon(FMT05);
 75     }
 76     
 77     //Kimi: close and free memory
 78     swmm_close();
 79     //Kimi: finish
 80     printf("Press ENTER key to Continue
");
 81     getchar();
 82     return 0;
 83 }
 84 
 85 //functions
 86 void swmm5_extend_run(char* f1, char* f2, char* f3) {
 87     IsOpenFlag = FALSE;
 88     IsStartedFlag = FALSE;
 89     SaveResultsFlag = TRUE;
 90     /*
 91     long newHour, oldHour = 0;
 92     long theDay, theHour;
 93     double elapsedTime = 0.0;                                                  //(5.1.011)
 94 */
 95     // --- open the files & read input data
 96     ErrorCode = 0;
 97     swmm_open(f1, f2, f3);
 98 
 99     // --- run the simulation if input data OK
100     /*
101     if (!ErrorCode)
102     {
103         // --- initialize values
104         swmm_start(TRUE);
105 
106         // --- execute each time step until elapsed time is re-set to 0
107         if (!ErrorCode)
108         {
109             writecon("
 o  Simulating day: 0     hour:  0");
110             do
111             {
112                 swmm_step(&elapsedTime);
113                 newHour = (long)(elapsedTime * 24.0);
114                 if (newHour > oldHour)
115                 {
116                     theDay = (long)elapsedTime;
117                     theHour = (long)((elapsedTime - floor(elapsedTime)) * 24.0);
118                     writecon("");
119                     sprintf(Msg, "%-5d hour: %-2d", theDay, theHour);
120                     writecon(Msg);
121                     oldHour = newHour;
122                 }
123             } while (elapsedTime > 0.0 && !ErrorCode);
124             writecon(""
125                 "");
126             writecon("Simulation complete           ");
127         }
128 
129         // --- clean up
130         swmm_end();
131     }
132 
133     // --- report results
134     if (Fout.mode == SCRATCH_FILE) swmm_report();
135     */
136     // --- close the system
137 
138     //conduit_slope=ConduitSlope(Conduit);
139 
140     //swmm_close(); //Kimi: move to main()
141 
142     
143     //-----------------------------------------------------------------------------
144     //Kimi: utilize the information we needed in here
145     count_object();
146     //-----------------------------------------------------------------------------
147 
148     return error_getCode(ErrorCode);                                          
149 }
150 
151 void* count_object() {
152     NumConduit = Nlinks[CONDUIT];//length of array
153     //NumLink=
154 }

  Swmm5Extend.h

 1 #ifndef SWMM5EXTEND_H
 2 #define SWMM5EXTEND_H
 3 
 4 //functions
 5 void swmm5_extend_run(char* f1, char* f2, char* f3);
 6 void* count_object();
 7 
 8 //global varibles
 9 int NumConduit;
10 int NumLink;//useless now
11 #endif // !SWMM5EXTEND_H

  swmm.i(精华所在)

 1 %module SWMM5ReadInpFile
 2 
 3 //Kimi: to covert the pointer to the array type automatically
 4 %pythonappend Conduit %{
 5     # Wrap it automatically
 6     newval = TConduitArray.frompointer(val)
 7     newval.ptr_retain = val
 8     val = newval
 9 %}
10 
11 %{
12 #define SWIG_FILE_WITH_INIT
13 #include "swmm5/consts.h"                    // defined constants
14 #include "swmm5/macros.h"                    // macros used throughout SWMM
15 #include "swmm5/enums.h"                     // enumerated variables
16 #include "swmm5/error.h"                     // error message codes
17 #include "swmm5/datetime.h"                  // date/time functions
18 #include "swmm5/objects.h"                   // definitions of SWMM‘s data objects
19 #include "swmm5/funcs.h"                     // declaration of all global functions
20 #include "swmm5/text.h"                      // listing of all text strings 
21 #define  EXTERN                        // defined as ‘extern‘ in headers.h
22 #include "swmm5/globals.h"                   // declaration of all global variables
23 #include "swmm5/swmm5.h"                     // declaration of exportable functions
24 #include "Swmm5Extend.h"
25 #define  MAX_EXCEPTIONS 100            // max. number of exceptions handled
26 %}
27 
28 //directives
29 %include "Swmm5Extend.h"
30 //global variables
31 #define  EXTERN 
32 %include "swmm5/objects.h"
33 %include "swmm5/globals.h" 
34 
35 //Kimi: to convert TConduit type pointer to list in python
36 %include <carrays.i>
37 %array_class(TConduit, TConduitArray);
38 //"TConduit" is equal to "struct foo"

 

其他过程略去,见其他随笔。

 1 #用python调用swmm数据
 2 
 3 import SWMM5ReadInpFile as swmm
 4 
 5 inputFile=iofiles\\dalizi_rain1.inp
 6 reportFile= "iofiles\\swmm5_extend.rpt"
 7 binaryFile=‘‘
 8 
 9 #run swmm to get input summary
10 swmm.swmm5_extend_run(inputFile, reportFile, binaryFile);
11 
12 pointer=swmm.cvar.Conduit
13 conduit=swmm.TConduitArray.frompointer(pointer)
14 ConduitLen=swmm.cvar.NumConduit
15 print(ConduitLen)
16 for i in range(0,ConduitLen):
17     print(conduit[i].slope)

运行成功:

技术图片

重要参考链接:

https://stackoverflow.com/questions/11998369/swig-python-structure-array

以上是关于[swmm]调用存储数据的“结构指针数组”的主要内容,如果未能解决你的问题,请参考以下文章

如何在两个不同的片段中使用存储库数据

在C中为结构指针数组成员分配地址

当我尝试重新分配()结构指针数组时,为啥我的 C 程序会崩溃?

从指向结构数组的指针中提取元素

为啥保守光栅化无法为某些三角形调用片段着色器?

C:指向结构指针数组的指针(分配/解除分配问题)