久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 3683|回復: 1
打印 上一主題 下一主題
收起左側

關于51單片機使用malloc函數的問題

[復制鏈接]
跳轉到指定樓層
樓主
ID:909487 發表于 2021-5-8 16:57 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
我在網上看到51單片機似乎不建議使用malloc函數,但這段程序也是看了論壇一位大佬寫的多級菜單的程序,不知道怎么改
求教怎么才能正確初始化菜單 使用的是stc89c52rc
  1. #include <reg52.h>
  2. #include <string.h>
  3. #include <stdio.h>
  4. #include <stdlib.h>


  5. /********IO引腳定義***********************************************************/
  6. sbit LCD_RS=P1^0;
  7. sbit LCD_RW=P1^1;
  8. sbit LCD_E=P1^2;
  9. sbit PSB =P1^6;                //PSB腳為12864-12系列的串、并通訊功能切換,我們使用8位并行接口,PSB=1
  10. sbit KEY1=P3^7;
  11. sbit KEY2=P3^6;
  12. sbit KEY3=P3^5;
  13. sbit KEY4=P3^4;
  14. sbit KEY5=P3^3;
  15. sbit KEY6=P3^2;   //定義按鍵1~7

  16. /********宏定義***********************************************************/
  17. #define LCD_Data P0
  18. #define Busy    0x80 //用于檢測LCD狀態字中的Busy標識

  19. /********函數聲明*************************************************************/
  20. void WriteDataLCD(unsigned char WDLCD);
  21. void WriteCommandLCD(unsigned char WCLCD,BuysC);
  22. unsigned char ReadDataLCD(void);
  23. unsigned char ReadStatusLCD(void);
  24. void LCDInit(void);
  25. void LCDClear(void);
  26. void LCDFlash(void);
  27. void Delay5Ms(void);
  28. void Delay400Ms(void);
  29. void highlight(unsigned char x,unsigned char y,unsigned char width,unsigned char mode);
  30. unsigned char key_scan(void);
  31. void display(unsigned char line);
  32. void func(void);
  33. void LCDClear(void);
  34. void LCD_0(void);
  35. void LCD_Set_XY(unsigned char x,unsigned char y);
  36. void Lcd_Disp(unsigned char x,unsigned char y,unsigned char *s);


  37. //首先定義一個菜單結構體
  38. typedef struct menu                      //定義一個菜單
  39. {
  40. unsigned char range_from,range_to;     //當前顯示的項開始及結束序號
  41. unsigned char itemCount;               //項目總數
  42. unsigned char selected;                //當前選擇項
  43. unsigned char *menuItems[17];                  //菜單項目
  44. struct menu **subMenus;                //子菜單
  45. struct menu *parent;                   //上級菜單 ,如果是頂級則為null
  46. void (**func)();                       //選擇相應項按確定鍵后執行的函數

  47. }Menu;

  48. Menu MainMenu = {                       //定義主菜單
  49. 0,3,4,0,                               //默認顯示0-3項,總共4項,當前選擇第0項
  50.   {
  51.   "測量表          ",                   //  >表示有下級選項
  52.   "其他設置        ",
  53.   "保護設定        ",
  54.   "查詢           >"
  55.   }
  56. };

  57. Menu searchMenu = {                    //查詢菜單
  58. 0,3,6,0,
  59. {
  60.   "記帳記錄明細   ",
  61.   "未采集記錄數   ",
  62.   "設備機號       ",
  63.   "本機IP地址     ",
  64.   "記錄空間大小   ",
  65.   "軟件版本       "
  66. }
  67. };
  68. Menu *currentMenu;                               //當前的菜單



  69. //初始化菜單:
  70. void initMenu()
  71. {
  72. char mempool[512];
  73. init_mempool (&mempool, sizeof(mempool));
  74. MainMenu.subMenus = malloc(sizeof(&MainMenu)*4);
  75. MainMenu.subMenus[0] = &searchMenu;//第1到3項沒有子菜單置null,選擇后程序會調用func中相應項中的函數
  76. MainMenu.subMenus[1] = NULL;
  77. MainMenu.subMenus[2] = NULL;
  78. MainMenu.subMenus[3] = NULL;//第四項查詢有子菜單
  79.         
  80. MainMenu.func = malloc(sizeof(&func)*4);
  81. MainMenu.func[0] = NULL;
  82. MainMenu.func[1] = NULL;
  83. MainMenu.func[2] = &func;//當選擇了并按了確定,會執行func函數
  84. MainMenu.func[3] = NULL;
  85. MainMenu.parent = NULL;//表示是頂級菜單

  86. searchMenu.subMenus = malloc(sizeof(&searchMenu)*6);
  87. searchMenu.subMenus[0] = searchMenu.subMenus[1] = searchMenu.subMenus[2] = searchMenu.subMenus[3] = searchMenu.subMenus[4] = searchMenu.subMenus[5] = NULL;
  88. searchMenu.func = malloc(sizeof(&printf)*6);
  89. searchMenu.func[0] = searchMenu.func[1] = searchMenu.func[2] = searchMenu.func[3] = searchMenu.func[4] = searchMenu.func[5] = NULL;
  90. searchMenu.parent = &MainMenu;//上一級菜單是MainMenu.進入查詢子菜單后按返回鍵,將會顯示這個菜單項

  91. currentMenu = &MainMenu;

  92. }

  93. //main

  94. int main(void)
  95. {

  96.          Delay400Ms();         //啟動等待,等LCD講入工作狀態
  97.          LCDInit();                 //LCD初始化
  98.          Delay5Ms();         //延時片刻
  99.   initMenu();
  100.   display(currentMenu->selected);
  101.   while(1){
  102. unsigned char key = key_scan();
  103. if(key == 0)
  104.   {
  105.   Delay5Ms();
  106.         Delay5Ms();
  107.    continue;
  108.   }
  109.   switch(key)
  110.   {
  111.    case 3://向上
  112.     if(currentMenu->selected == 0)//到了第一項
  113.      break;
  114.     else
  115.     {
  116.      currentMenu->selected--;
  117.      if(currentMenu->selected < currentMenu->range_from)//更新顯示的區域
  118.      {
  119.       currentMenu->range_from = currentMenu->selected;
  120.       currentMenu->range_to = currentMenu->range_from+3;
  121.      }
  122.      display(currentMenu->selected);
  123.      break;
  124.     }
  125.    case 4://向下
  126.     if(currentMenu->selected == currentMenu->itemCount-1)
  127.      break;
  128.     else
  129.     {
  130.      currentMenu->selected++;
  131.      if(currentMenu->selected>currentMenu->range_to)
  132.      {
  133.       currentMenu->range_to = currentMenu->selected;
  134.       currentMenu->range_from = currentMenu->range_to-3;
  135.      }
  136.      display(currentMenu->selected);
  137.      break;
  138.     }
  139.    case 1://Enter鍵
  140.    {
  141.     if(currentMenu->subMenus[currentMenu->selected] !=NULL)
  142.     {
  143.      currentMenu=currentMenu->subMenus[currentMenu->selected];
  144.      display(0);
  145.     }
  146.     else
  147.     {
  148.      if(currentMenu->func[currentMenu->selected] != NULL)
  149.      {
  150.       currentMenu->func[currentMenu->selected]();//執行相應的函數
  151.                         
  152.       display(currentMenu->selected);//返回后恢復原來的菜單狀態
  153.      }
  154.     }
  155.     break;
  156.    }
  157.    case 6://返回鍵
  158.    {
  159.     if(currentMenu->parent!=NULL)//父菜單不為空,將顯示父菜單
  160.     {
  161.      currentMenu = currentMenu->parent;
  162.      display(currentMenu->selected);
  163.     }
  164.     break;
  165.    }
  166.    default:
  167.      break;
  168.   }
  169.     Delay5Ms();
  170.           Delay5Ms();
  171. }
  172. }


  173. /*************************反白顯示*************************************/
  174. void highlight(unsigned char x,unsigned char y,unsigned char width,unsigned char mode)                //反白,X值為0-7,Y值為0-3,width為行反白格數
  175. {
  176.   unsigned char i,j,flag=0x00;
  177.   if(y>1)
  178.   {
  179.    flag=0x08;
  180.    y=y-2;
  181.   }
  182.   WriteCommandLCD(0x34,1);        //寫數據時,關閉圖形顯示,且打開擴展指令集
  183.         for(i=0;i<16;i++)
  184.         {
  185.                 WriteCommandLCD(0x80+(y<<4)+i,1);
  186.                 WriteCommandLCD(0x80+flag+x,1);
  187.                 for(j=0;j<width;j++)
  188.                 {
  189.                         WriteDataLCD(mode);
  190.                         WriteDataLCD(mode);
  191.                 }
  192.         }
  193.         WriteCommandLCD(0x36,1);       //寫完數據,開圖形顯示
  194. }



  195. /***********寫數據********************************************************/
  196. void WriteDataLCD(unsigned char WDLCD)
  197. {
  198.          ReadStatusLCD(); //檢測忙
  199.          LCD_RS = 1;
  200.          LCD_RW = 0;
  201.          LCD_Data = WDLCD;
  202.          LCD_E = 1;
  203.          LCD_E = 1;
  204.          LCD_E = 0;
  205. }

  206. /***********寫指令********************************************************/
  207. void WriteCommandLCD(unsigned char WCLCD,BuysC) //BuysC為0時忽略忙檢測
  208. {
  209.          if (BuysC) ReadStatusLCD(); //根據需要檢測忙
  210.          LCD_RS = 0;
  211.          LCD_RW = 0;
  212.          LCD_Data = WCLCD;
  213.          LCD_E = 1;
  214.          LCD_E = 1;
  215.          LCD_E = 0;  
  216. }


  217. /***********讀狀態*******************************************************/
  218. unsigned char ReadStatusLCD(void)
  219. {
  220.          LCD_Data = 0xFF;
  221.          LCD_RS = 0;
  222.          LCD_RW = 1;
  223.         LCD_E = 1;
  224.         LCD_E = 1;
  225.          while (LCD_Data & Busy); //檢測忙信號        
  226.          LCD_E = 0;
  227.          return(1);
  228. }

  229. /***********初始化********************************************************/
  230. void LCDInit(void)
  231. {
  232.          WriteCommandLCD(0x30,1); //顯示模式設置,開始要求每次檢測忙信號
  233.          WriteCommandLCD(0x01,1); //顯示清屏
  234.          WriteCommandLCD(0x06,1); // 顯示光標移動設置
  235.          WriteCommandLCD(0x0C,1); // 顯示開及光標設置
  236. }





  237. /***********短延時********************************************************/
  238. void Delay5Ms(void)
  239. {
  240.          unsigned int TempCyc = 11104;
  241.          while(TempCyc--);
  242. }

  243. /***********長延時********************************************************/
  244. void Delay400Ms(void)
  245. {
  246.          unsigned char TempCycA = 5;
  247.          unsigned int TempCycB;
  248.          while(TempCycA--){
  249.                   TempCycB=7269;
  250.                                 while(TempCycB--);
  251.          }
  252. }

  253. void LCD_0        (void){
  254.         unsigned char x,y,i;
  255.         unsigned int tmp=0;
  256.         for(i=0;i<9;){                                                //分兩屏,上半屏和下半屏,因為起始地址不同,需要分開
  257.         for(x=0;x<32;x++){                                        //32行
  258.                 WriteCommandLCD(0x34,1);
  259.                 WriteCommandLCD((0x80+x),1);        //列地址
  260.                 WriteCommandLCD((0x80+i),1);        //行地址,下半屏,即第三行地址0X88
  261.                 WriteCommandLCD(0x30,1);               
  262.                 for(y=0;y<16;y++)        
  263.                         WriteDataLCD(0x00);//讀取數據寫入LCD
  264.                 tmp+=16;               
  265.         }
  266.         i+=8;
  267.         }
  268.         WriteCommandLCD(0x36,1);                        //擴充功能設定
  269.         WriteCommandLCD(0x30,1);
  270. }

  271. //掃描鍵盤并返回按鍵值
  272. unsigned char key_scan()
  273. {
  274.         unsigned char sta = 1;//設置一個狀態,表示按鍵是否按下,1為抬起
  275.         if(sta == 1&&(KEY1 == 0||KEY2 == 0||KEY3 ==0||KEY4 ==0||KEY5 ==0||KEY6 ==0))//如果有一個按鍵按下,則進入下面程序,注意此時使用的邏輯或的關系
  276.         {
  277.                Delay5Ms();//消抖5ms
  278.                 sta = 0;//設置一個狀態,表示按鍵是否按下,0為按下
  279.                
  280.                 if(KEY1== 0)       return 1;
  281.                 else if(KEY2 ==0)  return 2;
  282.                 else if(KEY3 ==0)  return 3;
  283.                 else if(KEY4 ==0)  return 4;         
  284.                 else if(KEY5 ==0)  return 5;                        
  285.                                               else if(KEY6 ==0)  return 6;         
  286.         }
  287.         else if (KEY1 == 1&&KEY2 == 1&&KEY3 ==1&&KEY4 ==1&&KEY5 ==1&&KEY6 ==1) sta = 1;//表示沒有按鍵按下,注意此時使用的是邏輯與的關系;
  288.         return 0;//沒有按鍵按下
  289. }

  290. //用于顯示菜單項
  291. void display(unsigned char line)                    //顯示菜單項并設置選中的項反白
  292. {
  293.          int i;
  294. line = 3-(currentMenu->range_to-line);
  295.    LCD_0();
  296. for(i = 0;i<4;i++)
  297. {
  298.           Lcd_Disp(i+1,0,currentMenu->menuItems[i+currentMenu->range_from]);
  299. }
  300. highlight(0,line,8,255);//反白顯示指定行
  301. }

  302. void func(void)
  303. {

  304. }



  305. void LCD_set_xy( unsigned char x, unsigned char y )
  306. {        //設置LCD顯示的起始位置,X為行,Y為列
  307.     unsigned char address;
  308.         switch(x)
  309.         {
  310.             case 0: address = 0x80 + y; break;   
  311.             case 1: address = 0x80 + y; break;
  312.             case 2: address = 0x90 + y; break;
  313.             case 3: address = 0x88 + y; break;
  314.             case 4: address = 0x98 + y; break;
  315.             default:address = 0x80 + y; break;
  316.         }
  317.     WriteCommandLCD(address, 1);
  318. }

  319. void Lcd_Disp(unsigned char x,unsigned char y,unsigned char *s)
  320. {
  321.         WriteCommandLCD(0x30,1); //進入標準模式
  322.         LCD_Set_XY(x,y);
  323.         while (*s)  
  324.         {
  325.                 WriteDataLCD(*s);
  326.                 s++;
  327.         }
  328.        WriteCommandLCD(0x36,1); //返回圖形模式
  329. }
復制代碼


液晶顯示器.zip

8.39 KB, 下載次數: 13

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏 分享淘帖 頂 踩
回復

使用道具 舉報

沙發
ID:831047 發表于 2022-3-4 19:21 | 只看該作者
子菜單和執行函數都應該用一級指針吧?
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 中文字幕在线观看国产 | 久久免费小视频 | 国产小视频在线 | 日韩在线中文 | 欧美日韩最新 | 在线观看免费福利 | 国产片侵犯亲女视频播放 | 国产成人免费视频网站高清观看视频 | 亚洲精品专区 | 亚洲乱码一区二区三区在线观看 | 欧美一区二区三区四区五区无卡码 | 91精品久久久 | 国产精品久久久久久238 | 欧美精品一区二区三区四区 在线 | 亚洲精品一区二区 | 欧美日日 | 国产精品日韩高清伦字幕搜索 | 日韩一区二区视频 | 理论片87福利理论电影 | 午夜欧美a级理论片915影院 | 一级看片免费视频囗交动图 | 午夜视频在线视频 | 久久久久亚洲国产| 亚洲高清在线 | 亚洲福利在线观看 | 欧美在线视频一区 | 久久1区 | 日韩高清成人 | 欧美日韩中文国产一区发布 | 久久免费精品视频 | 欧美精品99| 色999视频 | 欧美综合在线观看 | 久久久久国产一区二区三区不卡 | 中文字幕一区二区三区精彩视频 | 在线观看av网站 | 日韩欧美大片在线观看 | 韩日精品一区 | 免费看黄视频网站 | 国产自产21区 | 日本黄色免费大片 |