結構體變量建模之終極解決方案—上篇中提到:
C 代碼中的結構體變量跟模型中的 Bus 信號是對應的;
C 代碼中的結構體類型跟模型中的 Bus 對象相對應。
下面我們來看看結構體數組。
三. 結構體數組的代碼實現
從前面結構體變量和結構體嵌套的兩個例子,我們也可以很清楚的體會到這兩點。所以,結構體數組,對應到模型中,自然也就是多維的 Bus 信號了。
對圖中 Inport 端口的 Data type 設置為 Bus: myBus,Port dimensions 設置為 2;數據字典中和信號 x 對應的信號對象 x 做同樣的設置,即 Datatype為Bus: myBus,Dimensions 為2。Selector 模塊分別選中兩維信號的第一維和第二維。兩組信號經過運算之后,再次通過 Bus Creator 模塊組合成 Bus 信號,再把兩路 Bus 信號使用 Vector Concatenate 模塊連成維數為 2 的 Bus 信號 y。
信號對象 x、y 除了數據類型設置為 Bus: myBus 之外,Storage Class 均設置為 ExportedGlobal。
由此,就有了如下結構體類型的定義:
typedefstruct
{
real_T a;
real_Tb;
real_T c;
} myBus;
和結構體數組的定義:
myBus x[2];
myBus y[2];
或許你已經注意到幾個增益模塊的參數值被設置為 k.a,k.b,k.c,沒錯,既然 myBus 作為結構體類型是一種組合數據類型,同樣也可以使用這個 myBus 設置參數 k 的數據類型,只是,一旦參數對象 k 的數據類型被設置為 myBus,那么參數的初始化就不像以前那么簡單的。本例設置如下:
k.Value = struct(‘a’,2, ‘b’,3, ‘c’,4);
如果參數k的存儲類設置為 ConstVolatile,那么代碼中參數 k 的定義如下:
const volatilemyBus k =
{
2.0,
3.0,
4.0
};
四. For-Each和結構體數組的結合
上例中 Bus 信號的維數為 2,所以我們很輕松的使用了 Selector 模塊,把兩維數據分別取出來實現后續(xù)算法,而現實中,我們可能面臨幾十甚至上百維的 Bus 信號,而后續(xù)的處理算法,對于每一維來講都是一致的,這種情況下怎么辦?我們可以想象得見,代碼中是可以通過一個 For 循環(huán)去實現的,模型當然也可以,這就是 For-Each 子系統(tǒng)。
雙擊上圖中的 For-Each 子系統(tǒng),得到下圖:
這個模型中 Inport 和 x、y 信號對象的 Dimensions 為 100。生成代碼之后,除了 x、y 兩個結構體數組的維數變成 100 之外,算法中有如下代碼:
結合 For-Each 模塊,讓我們的結構體數組建模更為方便。本想就此結束本篇,想到還會有人惦記另外一些結構體相關的話題,就再說一說:
五.位域結構體的代碼實現
Simulink 參數對象和信號對象的存儲類(Storage Class)里面都有 BitField (Custom) 選項,必須要說明的是,如果你的數據類型設置為 boolean,并且存儲類選擇為 BitField,是可以生成位域結構體變量的,只是,正如上一篇微文有網友留言所說,這樣做沒法指定結構體元素的順序,當然也沒有 Bus 與之對應。
如果我們想得到我們期望的結構體變量,比如:
typedef struct
{
unsigned char a:1;
unsigned char a1:1 ;
unsigned char a2:1;
unsigned char a3:1;
unsigned char a4:1;
unsigned char a5:1;
unsigned char a6:2;
}myBitFieldBus;
或者:此結構體類型中,前 6 個元素各占 1 個 bit,而第 7 個元素占 2 個 bit。模型如下:
顯然,Inport 端口的數據類型應該為 myBitFieldBus,信號對象 input 也一樣。
數據字典中定義了 myBitFieldBus 如下:
不難看出:
myBitFieldBus 內的元素 a,a1,a2,a3,a4,a5,均為 boolean 類型,a6 為 uint8;
Bus 對象 myBitFieldBus 的 Data scope 為 Imported,并且定義在頭文件。
mybitfieldstruct.h 文件中,也就是說,結構體類型 myBitFieldBus 不在這個模塊中定義,為了能夠生成代碼,需要提供 mybitfieldstruct.h 文件。
做完上述設置之后,生成代碼,如下:
typedef struct
{
boolean_T Out1;
boolean_T Out2;
boolean_T Out3;
boolean_T Out4;
boolean_T Out5;
boolean_T Out6;
unit8_TOut7;
}ExtY_bitfieldstruct_T;
myBitFieldBus input;
xtY_bitfieldstruct_T bitfieldstruct_Y;
void bitfieldstruct_step(void)
{
bitfieldstruct_Y.Out1 = input.a;
bitfieldstruct_Y.Out1 = input.a1;
bitfieldstruct_Y.Out1 = input.a2;
bitfieldstruct_Y.Out1 = input.a3;
bitfieldstruct_Y.Out1 = input.a4;
bitfieldstruct_Y.Out1 = input.a5;
bitfieldstruct_Y.Out1 = input.a6;
}
還要什么?指向結構體變量的指針?那就把前面幾個例子里的 x、y、input 等信號對象的存儲類設置為 ImportedExternPointer 就可以了,沒什么特別的。
最后:
C 代碼中的結構體變量跟模型中的 Bus 信號是對應的;
C 代碼中的結構體類型跟模型中的 Bus 對象相對應。
-
數據
+關注
關注
8文章
7241瀏覽量
91013 -
結構體
+關注
關注
1文章
130瀏覽量
11033
發(fā)布評論請先 登錄
Techwiz LCD 2D應用:二維LC透鏡建模分析
VirtualLab應用:亞波長結構偏振光柵的深入分析
程序設計與數據結構
Adams多體動力學仿真解決方案全面解析
VirtualLab:系統(tǒng)建模分析器
VirtualLab Fusion:系統(tǒng)建模分析器
仿真分析誤差來源及減少建模誤差的方法

評論