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

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 6346|回復: 4
收起左側

51單片機頻率幅度可調波形發生器代碼 三角波、鋸齒波、正弦波、方波

  [復制鏈接]
ID:74293 發表于 2021-4-28 21:59 | 顯示全部樓層 |閱讀模式
使用51單片機產生三角波、鋸齒波、正弦波、方波,頻率幅度可調。帶仿真,可以在protues里觀察波形情況。
仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載)

仿真

仿真


單片機源程序如下:
  1. #include<reg52.h>                                           //包含頭文件
  2. #include<intrins.h>
  3. #define uchar unsigned char                           //宏定義
  4. #define uint unsigned int

  5. sbit s1=P3^5;                                                    //定義按鍵的接口
  6. sbit s2=P3^6;
  7. sbit s3=P3^7;
  8. sbit s4=P3^4;

  9. sbit led0=P3^0;                                                        //定義四個LED,分別表示不同的波形
  10. sbit led1=P3^1;
  11. sbit led2=P3^2;
  12. sbit led3=P3^3;
  13.                                                                           
  14. sbit lcdrs=P2^7;                                           //液晶控制引腳,還有一個控制腳是RW,因為我們只需要向液晶里寫數據系那是就好了,所以,我們直接將RW引腳接地
  15. sbit lcden=P2^6;

  16. char num,boxing,u;                                           //定義全局變量
  17. uchar pinlv=100,bujin=1,bujin1=1;                   //頻率初始值是10Hz,步進值默認是0.1,顯示步進值變量
  18. uchar code table[]="0123456789";                //定義顯示的數組
  19. uchar code table1[]="Fout=     Wave form:";        //初始化顯示字符
  20. unsigned int m;                                   //定義變量 m
  21. int a,b,h,num1;                                                        //定義全局變量

  22. //自定義字符
  23. uchar code zifu[]={                   //此數組內數據為液晶上顯示波形符號的自定義字符
  24.                                                 0x0e,0x11,0x11,0x00,0x00,0x00,0x00,0x00,
  25.                                                 0x00,0x00,0x00,0x00,0x11,0x11,0x0e,0x00,          //正弦波           0  1

  26.                                                 0x00,0x07,0x04,0x04,0x04,0x04,0x1c,0x00,
  27.                                                 0x00,0x1c,0x04,0x04,0x04,0x04,0x07,0x00,          //矩形波           2  3

  28.                                                 0x00,0x01,0x02,0x04,0x08,0x10,0x00,0x00,
  29.                                                 0x00,0x10,0x08,0x04,0x02,0x01,0x00,0x00,          //三角波           4  5

  30.                                                 0x00,0x01,0x03,0x05,0x09,0x11,0x00,0x00,            //鋸齒波           6
  31. };
  32. uchar code sin[64]={           //此數組內的數據為,da輸出對應電壓值對應的數字量,0是0V,255是5V
  33. 135,145,158,167,176,188,199,209,218,226,234,240,245,249,252,254,254,253,251,247,243,237,230,222,213,204,193,182,170,158,
  34. 146,133,121,108,96,84,72,61,50,41,32,24,17,11,7,3,1,0,0,2,5,9,14,20,28,36,45,55,66,78,90,102,114,128
  35. };                                          //正弦波取碼
  36. uchar code juxing[64]={                //一個周期是采樣64個點, 所以數組內是64個數據
  37. 255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,255,
  38. 255,255,255,255,255,255,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
  39. };                                          //矩形波取碼

  40. uchar code sanjiao[64]={
  41. 0,8,16,24,32,40,48,56,64,72,80,88,96,104,112,120,128,136,144,152,160,168,176,184,192,200,208,216,224,232,240,248,
  42. 248,240,232,224,216,208,200,192,184,176,168,160,152,144,136,128,120,112,104,96,88,80,72,64,56,48,40,32,24,16,8,0
  43. };                                                //三角波取碼
  44. uchar code juchi[64]={
  45. 0,4,8,12,16,20,24,28,32,36,40,45,49,53,57,61,65,69,73,77,81,85,89,93,97,101,105,109,113,117,121,125,130,134,138,142,
  46. 146,150,154,158,162,166,170,174,178,182,186,190,194,198,202,206,210,215,219,223,227,231,235,239,243,247,251,255
  47. };                                          //鋸齒波取碼

  48. void delay(uint xms)         //延時函數
  49. {
  50.         int a,b;
  51.         for(a=xms;a>0;a--)
  52.                 for(b=110;b>0;b--);
  53. }
  54. void write_com(uchar com)                //寫命令函數
  55. {
  56.         lcdrs=0;
  57.         P0=com;       
  58.         delay(1);
  59.         lcden=0;
  60.         delay(1);                                                                           
  61.         lcden=1;
  62. }

  63. void write_date(uchar date)                //寫數據函數
  64. {
  65.         lcdrs=0;
  66.         P0=date;
  67.         delay(1);
  68.         lcden=0;
  69.         delay(1);
  70.         lcden=1;
  71. }

  72. //自定義字符集
  73. void Lcd_ram()      
  74. {
  75.         uint i,j,k=0,temp=0x04;
  76.         for(i=0;i<7;i++)
  77.         {
  78.            for(j=0;j<8;j++)
  79.            {
  80.             write_com(temp+j);
  81.             write_date(zifu[k]);
  82.             k++;
  83.            }
  84.            temp=temp+8;
  85.         }
  86. }

  87. void init_lcd()                                        //初始化函數
  88. {
  89.         uchar i;                                                                  
  90.         lcden=0;                                        //默認開始狀態為關使能端,見時序圖  
  91.         Lcd_ram();
  92.         write_com(0x01);                        //顯示清屏,將上次的內容清除,默認為0x01.
  93.         write_com(0x0f);
  94.         write_com(0x38);                        //顯示模式設置,默認為0x38,不用變。
  95.         write_com(0x0c);                        //顯示功能設置0x0f為開顯示,顯示光標,光標閃爍;0x0c為開顯示,不顯光標,光標不閃
  96.         write_com(0x06);                        //設置光標狀態默認0x06,為讀一個字符光標加1.
  97.         write_com(0x80);                        //設置初始化數據指針,是在讀指令的操作里進行的
  98.         for(i=10;i<20;i++)                           //顯示初始化
  99.         {
  100.                 write_date(table1[i]);                   //顯示第一行字符
  101.         }
  102.         write_com(0x80+40);                           //選擇第二行
  103.         for(i=0;i<9;i++)
  104.         {
  105.                 write_date(table1[i]);                   //顯示第二行字符
  106.         }
  107.         write_com(0x80+10);                                   //選擇第一行第十個位置
  108.         write_date(0);
  109.         write_date(1);
  110.         write_date(0);
  111.         write_date(1);
  112.         write_date(0);
  113.         write_date(1);                                           //顯示自定義的波形圖案
  114.         write_com(0x80+40+9);                   //選擇第二行第九個位置
  115.         write_date(' ');
  116.         write_date('1');
  117.         write_date('0');
  118.         write_date('.');
  119.         write_date('0');
  120.         write_date('H');
  121.         write_date('z');                                   //顯示初始的頻率值
  122. }
  123. void initclock()                           //定時器初始化函數
  124. {
  125.         TMOD=0x10;                                   //定時器的工作方式
  126.         TH0=a;
  127.         TL0=b;                                           //定時器賦初值
  128.         EA=1;                                           //打開中斷總開關
  129.         ET0=1;                                           //打開定時器允許中斷開關
  130.         TR0=1;                                           //打開定時器定時開關
  131. }
  132. void display()                                   //顯示函數
  133. {
  134.         uchar qian,bai,shi,ge;           //定義變量用于顯示
  135.         qian=pinlv/1000;                   //將頻率值拆成一位的數據,將數據除以1000,得到的商是一位數,賦值給qian
  136.         bai=pinlv%1000/100;                   //將頻率除以1000的余數再除以100就得到了頻率的百位,賦值給bai
  137.         shi=pinlv%1000%100/10;           //同上,得到頻率的十位
  138.         ge=pinlv%1000/100%10;
  139.         write_com(0x80+40+9); //選中第二行第九個位置

  140.            if(qian==0)                                   //千位如果為0
  141.         write_date(' ');                   //不顯示
  142.         else                                           //千位不為0
  143.         write_date(table[qian]);   //正常顯示千位
  144.         if(qian==0&&bai==0)                   //千位和百位都為0
  145.         write_date(' ');                   //百位不顯示
  146.         else                                           //不都為0
  147.         write_date(table[bai]);           //百位正常顯示
  148.         write_date(table[shi]);           //顯示十位數
  149.         write_date('.');                   //顯示小數點
  150.         write_date(table[ge]);           //顯示個位
  151.         write_date('H');                   //顯示頻率的單位Hz
  152.         write_date('z');
  153.         if(boxing==0)                           //判斷波形為正弦波
  154.         {
  155.                 write_com(0x80+10);           //選中一行頻率圖案位置
  156.                 write_date(0);                   //顯示正弦波圖案
  157.                 write_date(1);
  158.                 write_date(0);
  159.                 write_date(1);
  160.                 write_date(0);
  161.                 write_date(1);
  162.                 led3=1;
  163.                 led0=0;                                        //點亮正弦波指示燈
  164.         }
  165.         if(boxing==1)                                //注釋同上
  166.         {
  167.                 write_com(0x80+10);
  168.                 write_date(2);
  169.                 write_date(3);
  170.                 write_date(2);
  171.                 write_date(3);
  172.                 write_date(2);
  173.                 write_date(3);
  174.                 led0=1;
  175.                 led1=0;
  176.         }


  177.         if(boxing==2)
  178.         {
  179.                 write_com(0x80+10);
  180.                 write_date(4);
  181.                 write_date(5);
  182.                 write_date(4);
  183.                 write_date(5);
  184.                 write_date(4);
  185.                 write_date(5);
  186.                 led1=1;
  187.                 led2=0;
  188.         }
  189.         if(boxing==3)
  190.         {
  191.                 write_com(0x80+10);
  192.                 write_date(6);
  193.                 write_date(6);
  194.                 write_date(6);
  195.                 write_date(6);
  196.                 write_date(6);
  197.                 write_date(6);
  198.                 led2=1;
  199.                 led3=0;
  200.         }
  201. }
  202. void keyscan()                                        //頻率調節鍵盤檢測函數
  203. {
  204.         if(s1==0)                                        //加按鍵是否按下
  205.         {         
  206.                 EA=0;                                        //關閉中斷
  207.                 delay(2);                            //延時去抖
  208.                 if(s1==0)                                //再次判斷
  209.                 {
  210.                         while(!s1);                        //按鍵松開
  211.                         pinlv+=bujin;                //頻率以步進值加
  212.                         if(pinlv>1000)                //最大加到100Hz
  213.                         {
  214.                                 pinlv=100;                //100Hz
  215.                         }
  216.                         display();                        //顯示函數
  217.                         m=65536-(15000/pinlv);//計算頻率
  218. /*頻率值最小是10Hz,pinlv的值是100(因為要顯示小數點后一位),150000/100=1500,這個1500就是定時器需要計時的,單位是us,65536-1500得到的是定時器的初值,
  219. 先不管初值,先看定時時間,1500us,一個波形的周期是由64個定時組成的,所以,一個波形周期就是64*1500us=96000,也就是96ms,約等
  220. 于100ms,也就是10Hz的頻率*/
  221.                         a=m/256;                        //將定時器的初值賦值給變量
  222.                         b=m%256;
  223.                         EA=1;                                //打開中斷總開關
  224.                 }
  225.         }
  226.         if(s2==0)                                        //減按鍵按下
  227.         {         
  228.                
  229.                 delay(5);
  230.                 if(s2==0)
  231.                 {
  232.                         EA=0;       
  233.                         while(!s2);
  234.                         pinlv-=bujin;                   //頻率以步進值減
  235.                         if(pinlv<100)
  236.                         {
  237.                                 pinlv=100;
  238.                         }
  239.                         display();
  240.                         m=65536-(15000/pinlv);
  241.                         a=m/256;
  242.                         b=m%256;       
  243.                         EA=1;
  244.                 }
  245.         }
  246.         if(s3==0)                                         //波形切換按鍵
  247.         {         
  248.                 delay(5);
  249.                 if(s3==0)
  250.                 {
  251.                         EA=0;
  252.                         while(!s3);
  253.                         boxing++;                           //波形切換
  254.                         if(boxing>=4)                   //4種波形
  255.                         {
  256.                                 boxing=0;
  257.                         }
  258.                         display();
  259.                         EA=1;
  260.                 }
  261.         }       
  262. }
  263. void bujindisplay()                                 //步進值設置界面顯示程序
  264. {
  265.         uint bai,shi,ge;                         //定義步進值 百十個位
  266.         bai=bujin1/100;                                 //將步進值除以100得到百位,也就是頻率值的十位,因為有一個小數位
  267.         shi=bujin1%100/10;                         //將步進值除以100的余數除以十得到十位
  268.         ge=bujin1%100%10;                         //取余10后得到個位,也就是頻率步進值的小數點后一位
  269.         write_com(0x80+11);                         //選中液晶第一行第十一列
  270.         if(bai==0)                                         //百位是否為0
  271.         write_date(' ');                         //百位不顯示
  272.         else                                                 //百位不為0
  273.         write_date(table[bai]);                 //顯示百位數據
  274.         write_date(table[shi]);                 //顯示十位數據
  275.         write_date('.');                         //顯示小數點
  276.         write_date(table[ge]);                 //顯示個位,也就是小數點后一位
  277. }
  278. void bujinjiance()                                 //步進值設置鍵盤程序
  279. {
  280.         if(s4==0)                                         //步進設置按鍵按下
  281.         {
  282.                 delay(5);                                 //延時去抖
  283.                 if(s4==0)                                 //再次判斷按鍵
  284.                 {
  285.                          while(!s4);                 //按鍵釋放,按鍵松開才繼續向下執行
  286.                          h++;                                 //變量加
  287.                          if(h==1)                         //進入設置狀態時
  288.                          {
  289.                                   write_com(0x01);                          //清屏
  290.                                  write_com(0x80);                          //初始化顯示步進設置界面
  291.                                  write_date('S');delay(1);          //step value
  292.                                  write_date('t');delay(1);
  293.                                  write_date('e');delay(1);
  294.                                  write_date('p');delay(1);
  295.                                  write_date(' ');delay(1);
  296.                                  write_date('v');delay(1);
  297.                                  write_date('a');delay(1);
  298.                                  write_date('l');delay(1);
  299.                                  write_date('u');delay(1);
  300.                                  write_date('e');delay(1);
  301.                                  write_date(':');delay(1);
  302.                                  bujin1=bujin;                                  //步進值賦值給臨時變量
  303.                                  
  304.                                  bujindisplay();                          //顯示步進值
  305.                          }
  306.                          if(h==2)                                                  //退出設置
  307.                          {
  308.                                  h=0;                                                  //清零
  309.                                 bujin=bujin1;                                  //設置好的臨時步進值賦值給步進變量
  310.                                
  311.                                 init_lcd();                                          //初始化液晶顯示
  312.                                 initclock();                                  //定時器初始化
  313.                                 display();                                          //調用顯示程序
  314.                          }
  315.                 }
  316.         }
  317.         if(h==1)                                                                  //設置步進值時
  318.         {
  319.                 if(s1==0)                                                          //加按鍵按下
  320.                 {
  321.                         delay(5);                                                  //延時去抖
  322.                         if(s1==0)                                                  //再次判斷
  323.                         {
  324.                                 while(!s1);                                          //按鍵釋放
  325.                                 bujin1++;                                          //步進值加1
  326.                                 if(bujin1>=101)                                  //步進值最大100,也就是10.0Hz
  327.                                 {
  328.                                         bujin1=1;                                  //超過最大值就恢復到0.1Hz
  329.                                 }
  330.                                 bujindisplay();                                  //步進顯示
  331.                         }
  332.                 }
  333.                 if(s2==0)                                                          //減按鍵,注釋同上
  334.                 {
  335.                         delay(5);
  336.                         if(s2==0)
  337.                         {
  338.                                 while(!s2);
  339.                                 bujin1--;                                          //步進減
  340.                                 if(bujin1<=0)
  341. ……………………

  342. …………限于本文篇幅 余下代碼請從51黑下載附件…………
復制代碼

所有資料51hei提供下載:
仿真.rar (44.18 KB, 下載次數: 128)
程序.rar (50.16 KB, 下載次數: 122)



評分

參與人數 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

回復

使用道具 舉報

ID:537530 發表于 2024-1-5 15:27 | 顯示全部樓層
已測試,基本正常。
就是顯示程序復雜,波形數據點增多的話,步進有問題,且不能正常顯示。
回復

使用道具 舉報

ID:965040 發表于 2024-4-8 09:16 | 顯示全部樓層
已下載,HEX文件可正常仿真,程序重新編譯顯示不正常,新手,不知道哪有問題
回復

使用道具 舉報

ID:102702 發表于 2024-6-2 00:59 | 顯示全部樓層
user2402167 發表于 2024-1-5 15:27
已測試,基本正常。
就是顯示程序復雜,波形數據點增多的話,步進有問題,且不能正常顯示。

做成實物后,用仿真的HEX文件燒錄進單片機無法顯示,請問是哪里出的問題?您做成實物了?
回復

使用道具 舉報

ID:102702 發表于 2024-6-2 01:03 | 顯示全部樓層
user2402167 發表于 2024-1-5 15:27
已測試,基本正常。
就是顯示程序復雜,波形數據點增多的話,步進有問題,且不能正常顯示。

補充一下,我做的實物按波形選擇時,LED指示燈是能正常切換的
回復

使用道具 舉報

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

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 国产伦精品一区二区三区高清 | www.色婷婷| 欧美白人做受xxxx视频 | 久久久久久国产精品免费免费男同 | 国产精品自产av一区二区三区 | 日韩精品一区二区三区免费观看 | 国产天天操 | 伊人春色在线观看 | 国产男人的天堂 | 亚洲成av | 天堂一区二区三区 | 日韩毛片免费看 | 九九热精| 不卡av电影在线播放 | 欧美一区二区三区在线观看视频 | 午夜天堂精品久久久久 | 在线区 | 免费黄色的视频 | 91超碰在线 | 国产精品久久一区二区三区 | 国产成人精品区一区二区不卡 | www成人免费 | 欧美久久综合 | 在线观看免费观看在线91 | 欧美影院 | 男女免费网站 | 日本不卡一区二区三区在线观看 | 99精品久久久久久 | 亚洲字幕在线观看 | 中文字幕一区二区三区在线观看 | 99婷婷| 国产一区二区精品在线观看 | 欧美日韩高清在线一区 | 麻豆视频在线看 | 欧美日韩国产一区二区 | 亚洲 中文 欧美 日韩 在线观看 | 亚洲国产欧美日韩 | 日日操日日干 | 最新国产精品精品视频 | 国产精品国产成人国产三级 | 中文字幕中文字幕 |