找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開始

搜索
查看: 3097|回復(fù): 9
收起左側(cè)

關(guān)于C語言結(jié)構(gòu)體的使用及調(diào)用

  [復(fù)制鏈接]
ID:986472 發(fā)表于 2022-2-6 11:22 | 顯示全部樓層 |閱讀模式
當(dāng)前學(xué)習(xí),書上關(guān)于結(jié)構(gòu)的說明不是很詳細(xì)。雖然解決了問題,共享出來,看看是不是有更好的解決辦法。
在RTC.h里定義結(jié)構(gòu):
typedef struct
{
        u16        year;
        u8        month;
        u8        day;
        u8        hour;
        u8        minute;
        u8        second;
}timer_TypeDef;

extern  timer_TypeDef xdata clock;

在RTC.C里加入此語句,
timer_TypeDef xdata clock;
因main文件里include  RTC.h, 可直接調(diào)用。 我試過刪除xdata,編譯就出錯(cuò)。看起來只有這樣的操作才用結(jié)構(gòu)。
我曾經(jīng)在main.c里,全局變量里用 傳統(tǒng)的struct  rtc{} clock;  來定義結(jié)構(gòu),一直沒有成功。
大家有其它方式使用結(jié)構(gòu)嗎?
回復(fù)

使用道具 舉報(bào)

ID:161164 發(fā)表于 2022-2-6 23:18 | 顯示全部樓層
按你的步驟打一遍代碼
xxx.h:timer_TypeDef xdata clock;
xxx.c:timer_TypeDef xdata clock;
編譯成功

xxx.h:timer_TypeDef clock;
xxx.c:timer_TypeDef clock;
編譯成功

xxx.h:timer_TypeDef clock;
xxx.c:timer_TypeDef xdata clock;
編譯失敗

xxx.h:struct  rtc{}; extern struct  rtc clock;
xxx.c:struct  rtc clock;
編譯成功
回復(fù)

使用道具 舉報(bào)

ID:883242 發(fā)表于 2022-2-7 14:52 | 顯示全部樓層
如果必須加xdata才能通過編譯的話,那么就跟什么結(jié)構(gòu)體完全無關(guān)了,是你的RAM全用光了。
回復(fù)

使用道具 舉報(bào)

ID:986472 發(fā)表于 2022-2-7 15:41 | 顯示全部樓層
RTC.h里定義結(jié)構(gòu),RTC.c里使用此結(jié)構(gòu), 在main.c里還會(huì)調(diào)用此結(jié)構(gòu),編譯成功的三個(gè),可以這樣嗎?
多謝!
回復(fù)

使用道具 舉報(bào)

ID:1004486 發(fā)表于 2022-2-9 10:19 | 顯示全部樓層
本帖最后由 dtlhjnmg 于 2022-2-9 10:29 編輯

1. xdata只是申明變量使用外部擴(kuò)展存儲(chǔ)器,和struct定義無關(guān)
2. 用到了extern,再修改源變量定義的同時(shí),需要修改extern申明保持一致,且并不推介使用extern,可以通過函數(shù)封裝獲取。
3. 至于struct的定義,有多種類型,就你上面的typedef而言,可以如下用法:
typedef struct test_s {
   ......
}test_t;
1> 通過strcut定義: struct test_s test; (使用這種, 上面的typedef 和 test_t 可省略去掉)
2> 通過test_t定義: test_t test;            (使用這種, 上面的test_s 可以省略去掉)
這屬于c基本語法,看你上述的表達(dá)應(yīng)該是想使用如下方法:
struct test_s{
  ......
} test;
這樣也可以定義test結(jié)構(gòu)體成員,但實(shí)際使用中基本不存在這種用法,都是上面示例的用法。
至于你說的編譯不過,可以在定義前加入xdata嘗試:例如: xdata test_t test;   或者xdata struct test_s test;另外C基礎(chǔ)語法的學(xué)習(xí)測(cè)試,建議先在電腦端編譯測(cè)試運(yùn)行

說這么多,只為進(jìn)來拿3分下載一個(gè)資源 ,好心人就施舍我?guī)追职?...



評(píng)分

參與人數(shù) 1黑幣 +5 收起 理由
AUG + 5 淡定

查看全部評(píng)分

回復(fù)

使用道具 舉報(bào)

ID:130230 發(fā)表于 2022-2-9 11:58 | 顯示全部樓層
ram用光了是正解
回復(fù)

使用道具 舉報(bào)

ID:161164 發(fā)表于 2022-2-9 15:19 | 顯示全部樓層
上班摸魚中,只能用Proteus 仿真一下
Proteus 的AT89沒有xdata, 用idata代替一下,編譯成功 2022-02-09_151350.png

main.c

  1. #include <STC89C5xRC.H>//Code:8k        Ram:512        xdata:256        EEProm:5k        Addr:0x2000
  2. #include "struct_Test.h"

  3. typedef         unsigned char        u8;  //0 to 255
  4. typedef         unsigned int        u16;  //0 to 65535
  5. typedef         unsigned long        u32;  //0 to 4294967295

  6. DateTime idata PLC_Clock = {2022, 2, 9, 15, 12, 0};

  7. u8 TMR_Clock_ACC;
  8. u8 TMR_Disp_ACC;
  9. u8 Disp_Buff[8];
  10. u8 code smgduan[] = {0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x07, 0x7f, 0x6f, 0x40}; //共陰數(shù)碼管
  11.                                                                                 //0*****1*****2*****3*****4*****5*****6*****7*****8*****9*****-**
  12. void Timer0Init(void)
  13. {
  14.     TMOD &= 0xF0;                //設(shè)置定時(shí)器模式
  15.     TMOD |= 0x01;                //設(shè)置定時(shí)器模式
  16.     TL0 = 0x18;                //設(shè)置定時(shí)初始值1ms@12.000MHz 12T
  17.     TH0 = 0xFC;                //設(shè)置定時(shí)初始值1ms@12.000MHz 12T
  18.     TF0 = 0;                //清除TF0標(biāo)志
  19.     TR0 = 1;                //定時(shí)器0開始計(jì)時(shí)
  20. }
  21. void timer0_int (void) interrupt 1
  22. {
  23.     static u8 Sys10ms;
  24.     TL0 = 0x18;                //設(shè)置定時(shí)初始值1ms@12.000MHz 12T
  25.     TH0 = 0xFC;                //設(shè)置定時(shí)初始值1ms@12.000MHz 12T
  26.     if(TMR_Disp_ACC)TMR_Disp_ACC--;
  27.     if(++Sys10ms == 10)
  28.     {
  29.         Sys10ms = 0;
  30.         //Timer 10ms
  31.         if(TMR_Clock_ACC)TMR_Clock_ACC--;
  32.     }
  33. }
  34. void Sys_Init(void)
  35. {
  36.     ET0 = 1;                        //Timer0 interrupt Enable
  37.     EA = 1;                                //All interrupt Enable
  38. }
  39. void main()
  40. {
  41.     u8 Sta = 0;
  42.     Timer0Init();
  43.     Sys_Init();
  44.     while (1)
  45.     {
  46.         if(!TMR_Clock_ACC)
  47.         {   TMR_Clock_ACC = 100;
  48.             Clock_Calc();
  49.             Disp_Buff[0] = PLC_Clock.SEC % 10;
  50.             Disp_Buff[1] = PLC_Clock.SEC / 10;
  51.             Disp_Buff[3] = PLC_Clock.MIN % 10;
  52.             Disp_Buff[4] = PLC_Clock.MIN / 10;
  53.             Disp_Buff[6] = PLC_Clock.HR % 10;
  54.             Disp_Buff[7] = PLC_Clock.HR / 10;
  55.         }
  56.         if(!TMR_Disp_ACC)
  57.         {   static u8 i = 0;
  58.             TMR_Disp_ACC = 2;
  59.             P2 = 0xff;
  60.             if(i == 2 || i == 5)
  61.                 P0 = 0x40;
  62.             else
  63.                 P0 = smgduan[Disp_Buff[i]];
  64.             P2 = ~(0x01 << i);
  65.             i = ++i % 8;
  66.         }
  67.     }
  68. }
復(fù)制代碼


struct_Test.h

  1. #ifndef _struct_Test_H_
  2. #define _struct_Test_H_

  3. typedef struct {
  4.     unsigned int YY;
  5.     unsigned char MM;
  6.     unsigned char DD;
  7.     unsigned char HR;
  8.     unsigned char MIN;
  9.     unsigned char SEC;
  10. } DateTime;

  11. extern DateTime idata PLC_Clock;

  12. void Clock_Calc();

  13. #endif
復(fù)制代碼


struct_Test.c

  1. #ifndef _struct_Test_H_
  2. #include "struct_Test.h"
  3. #endif

  4. void Clock_Calc()
  5. {
  6.     if(++PLC_Clock.SEC >= 60)
  7.     {
  8.         PLC_Clock.SEC = 0;
  9.         if(++PLC_Clock.MIN >= 60)
  10.         {
  11.             PLC_Clock.MIN = 0;
  12.             if(++PLC_Clock.HR >= 24)
  13.             {
  14.                 PLC_Clock.HR = 0;
  15.                 if(++PLC_Clock.DD >= 30)
  16.                 {
  17.                     PLC_Clock.DD = 0;
  18.                     if(++PLC_Clock.MM >= 13)
  19.                     {
  20.                         PLC_Clock.MM = 1;
  21.                         PLC_Clock.YY++;
  22.                     }
  23.                 }
  24.             }
  25.         }
  26.     }
  27. }
復(fù)制代碼

回復(fù)

使用道具 舉報(bào)

ID:251368 發(fā)表于 2022-2-9 15:23 | 顯示全部樓層
xdata只是數(shù)據(jù)定義在了外部的ram區(qū),這個(gè)ram可能不是真的外部ram,一般stc在內(nèi)部集成了片外的ram區(qū),所以xdata只是定義了變量的位置,和程序沒有什么關(guān)系,報(bào)錯(cuò)大部分就是ram不夠用了,根據(jù)你選擇的芯片決定你的ram區(qū)大小,像89c51只有128字節(jié),內(nèi)存可經(jīng)不起你這么折騰
回復(fù)

使用道具 舉報(bào)

ID:434018 發(fā)表于 2022-2-12 15:42 | 顯示全部樓層
51 have 3 type memory
data : address 0x00~ 0x7F (internal memory)
idata: address 0x00~ 0xFF (internal memory)
xdata: address 0x0000~ 0xFFFF (external memory 外部)
回復(fù)

使用道具 舉報(bào)

ID:401564 發(fā)表于 2022-2-13 01:48 | 顯示全部樓層
這樣寫就不會(huì)出錯(cuò)了
extern struct                                                                                                          
{
成員1;
成員2;
.......
}xdata GPS_Buffer0,GPS_Buffer1;       
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表