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

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

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

51單片機簡易波形發生器程序 Proteus仿真 模擬正弦波、方波、鋸齒波和三角波 有注釋

[復制鏈接]
跳轉到指定樓層
樓主
可以模擬正弦波、方波、鋸齒波和三角波四種波形。
仿真原理圖如下(proteus仿真工程文件可到本帖附件中下載)


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

  5. /**************定義按鍵的接口*************/
  6. sbit s1=P3^5;         //KEY2 頻率加                                          
  7. sbit s2=P3^6;          //KEY3 頻率減
  8. sbit s3=P3^7;         //KEY1 波形選擇
  9. sbit s4=P3^4;        //KEY4 步進值

  10. sbit led0=P3^0;         //正弦波指示燈
  11. sbit led1=P3^1;  //方波指示燈
  12. sbit led2=P3^2;         //三角波指示燈
  13. sbit led3=P3^3;         //鋸齒波指示燈
  14.                                                                            
  15. sbit lcdrs=P2^7;//液晶讀數據引腳,還有一個控制腳是RW(讀),因為我們只需要向液晶里寫數據系那是就好了,所以,我們直接將RW引腳接地                                          
  16. sbit lcden=P2^6;//使能端

  17. char num,boxing,u;                                           //定義全局變量        
  18. int pinlv=100,bujin=1,bujin1=1;                   //定義pinlv(頻率)的初始值為10hz,可以根據實際情況修改 bujin表示增減頻率按鍵按下一次的增減值 初始值為0.1
  19. uchar code table[]="0123456789";                //定義顯示頻率的數組
  20. uchar code table1[]="Fout=     Wave form:";        //Fout后面顯示頻率值;Wave form后面顯示模擬波形  
  21. unsigned long int m;
  22. int a,b,h,num1;                //a b為定時器初始值變量  

  23. /************自定義字符 用于顯示當前示波器顯示波形的模擬波形 字符集可通過取模軟件獲取****************/
  24. uchar code zifu[]={
  25.                                                 0x0e,0x11,0x11,0x00,0x00,0x00,0x00,0x00,
  26.                                                 0x00,0x00,0x00,0x00,0x11,0x11,0x0e,0x00,          //正弦波0 1兩行

  27.                                                 0x00,0x07,0x04,0x04,0x04,0x04,0x1c,0x00,
  28.                                                 0x00,0x1c,0x04,0x04,0x04,0x04,0x07,0x00,          //矩形波2 3兩行

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

  31.                                                 0x00,0x01,0x03,0x05,0x09,0x11,0x00,0x00,            //鋸齒波6一行
  32. };
  33. uchar code sin[64]={
  34. 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,
  35. 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
  36. };                                          //正弦波取碼,此數組內的數據為,da輸出對應電壓值對應的數字量,0是0V,255是5V
  37. uchar code juxing[64]={
  38. 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,
  39. 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
  40. };                                          //矩形波取碼 一個周期是采樣64個點, 所以數組內是64個數據

  41. uchar code sanjiao[64]={
  42. 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,
  43. 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
  44. };                                                //三角波取碼
  45. uchar code juchi[64]={
  46. 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,
  47. 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
  48. };                                          //鋸齒波取碼

  49. void delay(uint xms)         //延時函數        具體延時時間根據傳遞的x變量決定
  50. {
  51.         int a,b;
  52.         for(a=xms;a>0;a--)
  53.                 for(b=110;b>0;b--);
  54. }

  55. /**********寫入指令函數**********/
  56. void write_com(uchar com)               
  57. {
  58.         lcdrs=0;   //讀端置0
  59.         P0=com;          //往P0口寫入指令 即lcd地址
  60.         delay(1); //延時
  61.         lcden=1; //使能端從1變化到0 lcd執行一次讀命令
  62.         delay(1);//延時                                                                           
  63.         lcden=0; //使能端置0
  64. }

  65. /***********寫入數據函數*********/
  66. void write_date(uchar date)               
  67. {
  68.         lcdrs=1;//讀端口置為1
  69.         P0=date; //數據送入P0口,寫入LCD顯示
  70.         delay(1);  //延時
  71.         lcden=1;   //同上
  72.         delay(1);
  73.         lcden=0;
  74. }

  75. /************自定義字符集 LCD顯示模擬波形**********/
  76. void Lcd_ram()      
  77. {
  78.         uint i,j,k=0;
  79.         uint temp=0x40;//temp為顯示起始地址
  80.         for(i=0;i<7;i++) //將字符集寫入
  81.         {
  82.            for(j=0;j<8;j++)        //寫入8個字節的字符        即zifu[]的一行數據
  83.            {
  84.             write_com(temp+j);
  85.             write_date(zifu[k]);
  86.             k++;
  87.            }
  88.            temp=temp+8;        //下一個字符的起始地址
  89.         }
  90. }

  91. /**********液晶初始化函數************/
  92. void init_lcd()                                       
  93. {
  94.         uchar i;                                                                  
  95.         lcden=0;                                        //默認開始狀態為關使能端,見lcd液晶時序圖  
  96.         Lcd_ram();                                        // 寫入字符集
  97.         write_com(0x0f);                        //
  98.         write_com(0x38);                        //顯示模式設置,默認為0x38,不用變。
  99.         write_com(0x01);                        //顯示清屏,將上次的內容清除,默認為0x01.
  100.         write_com(0x0c);                        //顯示功能設置0x0f為開顯示,顯示光標,光標閃爍;0x0c為開顯示,不顯光標,光標不閃
  101.         write_com(0x06);                        //設置光標狀態默認0x06,為讀一個字符光標加1.
  102.         write_com(0x80);                        //液晶的第一行首地址
  103.         for(i=10;i<20;i++)                           //將table1的wave form:顯示在第一行
  104.         {
  105.                 write_date(table1[i]);
  106.         }
  107.         write_com(0x80+0x40);  // 液晶第二行的起始地址
  108.         for(i=0;i<9;i++)           //顯示初始化 從table1的第1個字符顯示,將table1的Fout顯示在第二行
  109.         {
  110.                 write_date(table1[i]);         
  111.         }
  112.         write_com(0x80+10);                   //初始為正弦波 在wave form后面顯示模擬波形
  113.         write_date(0);
  114.         write_date(1);
  115.         write_date(0);
  116.         write_date(1);
  117.         write_date(0);
  118.         write_date(1);
  119.         write_com(0x80+0x40+0x09); //設置初始頻率為10hz
  120.         write_date(' ');
  121.         write_date('1');
  122.         write_date('0');
  123.         write_date('.');
  124.         write_date('0');
  125.         write_date('H');
  126.         write_date('z');
  127. }

  128. /**********定時器初始化**********/
  129. void initclock()                           
  130. {
  131.         TMOD=0x01;        //定時器工作方式1
  132.         TH0=a;                //賦初值
  133.         TL0=b;
  134.         EA=1;                 //打開總中斷
  135.         ET0=1;                 //打開T0中斷
  136.         TR0=1;                 //啟動定時器
  137. }


  138. /**********顯示函數(顯示模擬波形和頻率以及控制提示小燈)**********/
  139. void display()                                
  140. {
  141.         uchar qian,bai,shi,ge; //只是將頻率分成千位,百位,十位,個位四位顯示,實際并不是1000的格式
  142.         qian=pinlv/1000;                //將頻率值拆成一位的數據,將數據除以1000,得到的商是一位數,賦值給qian
  143.         bai=pinlv%1000/100;                 //將頻率除以1000的余數再除以100就得到了頻率的百位,賦值給bai
  144.         shi=pinlv%1000%100/10;         //得到頻率的十位
  145.         ge=pinlv%1000%100%10;         //得到頻率的個位
  146.         write_com(0x80+0x40+0x09);//顯示頻率數字起始地址 第二行第九個位置

  147.            if(qian==0)                   //千位為0
  148.         write_date(' ');   //顯示空格
  149.         else
  150.         write_date(table[qian]);//查數組顯示千位數字
  151.         if(qian==0&&bai==0)                 //千位和百位都為0
  152.         write_date(' ');                  //百位不顯示
  153.         else
  154.         write_date(table[bai]);         //顯示百位數
  155.         write_date(table[shi]);        //顯示十位數
  156.         write_date('.');                 //顯示小數點
  157.         write_date(table[ge]);         //顯示個位
  158.         write_date('H');
  159.         write_date('z');
  160.         if(boxing==0)  //正弦波
  161.         {
  162.                 write_com(0x80+10);//wave form:后面的地址  顯示正弦波模擬波形
  163.                 write_date(0);
  164.                 write_date(1);
  165.                 write_date(0);
  166.                 write_date(1);
  167.                 write_date(0);
  168.                 write_date(1);
  169.                 led3=1;            
  170.                 led0=0;                           //led0小燈亮         提示正弦波
  171.         }

  172.         if(boxing==1)    //          方波
  173.         {
  174.                 write_com(0x80+10);         //顯示方波模擬波形
  175.                 write_date(2);
  176.                 write_date(3);
  177.                 write_date(2);
  178.                 write_date(3);
  179.                 write_date(2);
  180.                 write_date(3);
  181.                 led0=1;                  
  182.                 led1=0;                  //led1小燈亮         提示方波
  183.         }

  184.         if(boxing==2)  //三角波
  185.         {
  186.                 write_com(0x80+10);         //顯示三角波模擬波形
  187.                 write_date(4);
  188.                 write_date(5);
  189.                 write_date(4);
  190.                 write_date(5);
  191.                 write_date(4);
  192.                 write_date(5);
  193.                 led1=1;
  194.                 led2=0;                        //led2小燈亮  提示三角波
  195.         }

  196.         if(boxing==3)           //鋸齒波  
  197.         {
  198.                 write_com(0x80+10);        //顯示鋸齒波模擬波形
  199.                 write_date(6);
  200.                 write_date(6);
  201.                 write_date(6);
  202.                 write_date(6);
  203.                 write_date(6);
  204.                 write_date(6);
  205.                 led2=1;
  206.                 led3=0;                        //led小燈亮  提示鋸齒波
  207.         }

  208. }

  209. /*********鍵盤檢測函數************/
  210. void keyscan()                                
  211. {
  212.         if(s1==0)        //頻率加按鍵
  213.         {         
  214.                 EA=0;                  //關閉定時器總中斷
  215.                 delay(2);          //延時消除鍵盤抖動
  216.                 if(s1==0)
  217.                 {
  218.                         while(!s1);        //等待按鍵松開
  219.                         pinlv+=bujin;  //根據步進值增加頻率
  220.                         if(pinlv>1000)        //頻率值如果大于100  則重新賦值為100
  221.                         {
  222.                                 pinlv=100;
  223.                         }
  224.                         display();
  225.                         m=65536-(150000/pinlv);/*頻率值最小是10Hz,pinlv的值是100(因為要顯示小數點后一位),150000/100=1500,這個1500就是定時器需要計時的,單位是us,65536-1500得到的是定時器的初值,
  226. 先不管初值,先看定時時間,1500us,一個波形的周期是由64個定時(前面四個波形數組)組成的,所以,一個波形周期就是64*1500us=96000,也就是96ms,約等
  227. 于100ms,也就是10Hz的頻率*/
  228.                         a=m/256;           //將定時器的初值賦值給變量
  229.                         b=m%256;
  230.                         EA=1;                   //開啟總中斷
  231.                 }
  232.         }
  233.         if(s2==0)  //頻率減按鍵
  234.         {         
  235.                
  236.                 delay(5); //延時消抖
  237.                 if(s2==0)
  238.                 {
  239.                         EA=0;        
  240.                         while(!s2);
  241.                         pinlv-=bujin; //根據步進值減頻率
  242.                         if(pinlv<100)
  243.                         {
  244.                                 pinlv=1000;
  245.                         }
  246.                         display();
  247.                         m=65536-(150000/pinlv);        //同上
  248.                         a=m/256;
  249.                         b=m%256;        
  250.                         EA=1;
  251.                 }
  252.         }
  253.         if(s3==0) //調換波形按鍵
  254.         {         
  255.                 delay(5);
  256.                 if(s3==0)
  257.                 {
  258.                         EA=0;
  259.                         while(!s3);
  260.                         boxing++;
  261.                         if(boxing>=4)
  262.                         {
  263.                                 boxing=0;
  264.                         }
  265.                         display();
  266.                         EA=1;
  267.                 }
  268.         }        
  269. }

  270. /*********步進值設置界面顯示函數*********/
  271. void bujindisplay()        
  272. {
  273.         uint bai,shi,ge; //將步進值分百位,十位,個位顯示,但是并不是100的形式
  274.         bai=bujin1/100;         //將步進值除以100得到百位,也就是頻率值的十位,因為有一個小數位
  275.         shi=bujin1%100/10;//將步進值除以100的余數除以十得到十位
  276.         ge=bujin1%100%10;  //取余10后得到個位,也就是頻率步進值的小數點后一位
  277.         write_com(0x80+11);//選中液晶第一行第十一列
  278.         if(bai==0) //百位是否為0
  279.         write_date(' ');//百位不顯示 空格
  280.         else                        //百位不為0
  281.         write_date(table[bai]);         //顯示百位數據
  282.         write_date(table[shi]);          //顯示十位數據
  283.         write_date('.');           //顯示小數點
  284.         write_date(table[ge]); //顯示個位,也就是小數點后一位
  285. }

  286. /**********步進設置按鍵************/
  287. void bujinjiance()
  288. {
  289.         if(s4==0)   //步進設置按鍵按下
  290.         {
  291.                 delay(5); //延時去抖
  292.                 if(s4==0)  //再次判斷按鍵
  293.                 {
  294.                          while(!s4);  //按鍵釋放,按鍵松開才繼續向下執行
  295.                          h++;                 //變量加
  296.                          if(h==1)           //進入設置狀態時
  297.                          {
  298.                                   write_com(0x01);         //液晶清屏
  299.                                  write_com(0x80);         //初始化顯示步進設置界面
  300.                                  write_date('S');delay(1);          //step value(步進值)
  301.                                  write_date('t');delay(1);
  302.                                  write_date('e');delay(1);
  303.                                  write_date('p');delay(1);
  304.                                  write_date(' ');delay(1);
  305.                                  write_date('v');delay(1);
  306.                                  write_date('a');delay(1);
  307.                                  write_date('l');delay(1);
  308.                                  write_date('u');delay(1);
  309.                                  write_date('e');delay(1);
  310.                                  write_date(':');delay(1);
  311.                                  bujin1=bujin;                         //步進值賦值給臨時變量
  312.                                  
  313.                                  bujindisplay();//顯示步進值
  314.                          }
  315.                          if(h==2)                        //退出設置
  316.                          {
  317.                                  h=0;
  318.                                 bujin=bujin1;                //設置好的臨時步進值賦值給步進變量
  319.                                 
  320.                                 init_lcd();                        //初始化液晶顯示
  321.                                 initclock();           //定時器初始化
  322.                                 display();                  //調用顯示程序
  323.                          }
  324.                 }
  325.         }
  326.         if(h==1)                                         //設置步進值時
  327.         {
  328.                 if(s1==0)                                //加按鍵按下(此時是加步進值不是加頻率)
  329.                 {
  330.                         delay(5);                   //延時去抖
  331.                         if(s1==0)                        //再次判斷
  332.                         {
  333.                                 while(!s1);                //按鍵釋放
  334.                                 bujin1++;                  //步進值加1
  335.                                 if(bujin1>=101)          //步進值最大100,也就是10.0Hz
  336.                                 {
  337.                                         bujin1=1;           //超過最大值就恢復到0.1Hz
  338.                                 }
  339.                                 bujindisplay();            //步進值顯示
  340.                         }
  341.                 }
  342.                 if(s2==0)                                 //減按鍵(減步進值),注釋與上面同理
  343.                 {
  344.                         delay(5);
  345.                         if(s2==0)
  346.                         {
  347.                                 while(!s2);
  348.                                 bujin1--;
  349.                                 if(bujin1<=0)
  350.                                 {
  351.                                         bujin1=100;
  352.                                 }
  353.         {
  354. ……………………

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

所有資料51hei提供下載:
簡易波形發生器.zip (109.45 KB, 下載次數: 201)

評分

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

查看全部評分

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

使用道具 舉報

沙發
ID:1037863 發表于 2022-7-1 11:48 | 只看該作者
請問這個里面是程序仿真都有嗎
回復

使用道具 舉報

板凳
ID:1080209 發表于 2023-5-27 22:20 來自手機 | 只看該作者
Whl1314520 發表于 2022-7-1 11:48
請問這個里面是程序仿真都有嗎

都有的,只要解壓就好了
回復

使用道具 舉報

地板
ID:965040 發表于 2024-4-8 09:46 | 只看該作者
已下載測試,仿真和程序都正常,贊
回復

使用道具 舉報

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

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 天天色影视综合 | 欧美区在线观看 | 日韩一区二区三区视频 | 亚洲精品一区二区在线观看 | 国产成人99久久亚洲综合精品 | 一区二区三区视频播放 | 国产精品a一区二区三区网址 | 天天摸天天干 | av免费网站在线观看 | 亚洲精品在线免费观看视频 | av天天看| 韩日一区二区 | 一级特黄网站 | 毛片毛片毛片毛片 | 在线日韩视频 | 中文字幕乱码亚洲精品一区 | 欧美日韩专区 | 欧美激情精品久久久久 | 成人网在线 | 97精品超碰一区二区三区 | 成人影院一区二区三区 | 中文字幕亚洲欧美日韩在线不卡 | 福利影院在线看 | 午夜免费福利影院 | 日韩欧美精品一区 | av黄色免费 | 欧美专区在线观看 | 91久久| 国产精品女人久久久 | 亚洲国产精品va在线看黑人 | 草久久| 激情毛片| 午夜丁香视频在线观看 | 人人干人人草 | 国产高清视频一区二区 | 亚洲日本激情 | av官网在线 | 成人小视频在线观看 | 青青草精品视频 | 久草免费福利 | 成人av在线网站 |