找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 3928|回复: 5
收起左侧

单片机波形发生器设计 包含仿真,源程序

[复制链接]
ID:571533 发表于 2020-2-17 20:20 | 显示全部楼层 |阅读模式
波形发生器,可产生方波,正弦波,三角波,锯齿波,步进频率可调,幅值

仿真原理图如下(proteus仿真工程文件可到本帖附件中下载)
51hei.png

单片机源程序如下:
  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.                                                                            
  10. sbit lcdrs=P2^7;                                           //液晶控制引脚
  11. sbit lcden=P2^6;

  12. char num,boxing,u;                                          
  13. int pinlv=100,bujin=1,bujin1=1;                   //频率初始值是10Hz,步进值默认是0.1,显示步进值变量
  14. uchar code table[]="0123456789";                //定义显示的数组
  15. uchar code table1[]="Fout=     Wave form:";        //初始化显示字符
  16. unsigned long int m;                                   //定义长整形变量 m
  17. int a,b,h,num1;                                                        //定义全局变量

  18. //自定义字符
  19. uchar code zifu[]={                   //此数组内数据为液晶上显示波形符号的自定义字符
  20.                                                 0x0e,0x11,0x11,0x00,0x00,0x00,0x00,0x00,
  21.                                                 0x00,0x00,0x00,0x00,0x11,0x11,0x0e,0x00,          //正弦波           0  1

  22.                                                 0x00,0x07,0x04,0x04,0x04,0x04,0x1c,0x00,
  23.                                                 0x00,0x1c,0x04,0x04,0x04,0x04,0x07,0x00,          //矩形波           2  3

  24.                                                 0x00,0x01,0x02,0x04,0x08,0x10,0x00,0x00,
  25.                                                 0x00,0x10,0x08,0x04,0x02,0x01,0x00,0x00,          //三角波           4  5

  26.                                                 0x00,0x01,0x03,0x05,0x09,0x11,0x00,0x00,            //锯齿波           6
  27. };
  28. uchar code sin[64]={           //da输出对应电压值对应的数字量,0是0V,255是5V
  29. 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,
  30. 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
  31. };                                          //正弦波取码
  32. uchar code juxing[64]={                //一个周期是采样64个点, 所以数组内是64个数据
  33. 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,
  34. 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
  35. };                                          //矩形波取码

  36. uchar code sanjiao[64]={
  37. 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,
  38. 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
  39. };                                                //三角波取码
  40. uchar code juchi[64]={
  41. 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,
  42. 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
  43. };                                          //锯齿波取码

  44. void delay(uint xms)         //延时函数
  45. {
  46.         int a,b;
  47.         for(a=xms;a>0;a--)
  48.                 for(b=110;b>0;b--);
  49. }
  50. void write_com(uchar com)                //写命令函数
  51. {
  52.         lcdrs=0;
  53.         P0=com;        
  54.         delay(1);
  55.         lcden=1;
  56.         delay(1);                                                                           
  57.         lcden=0;
  58. }

  59. void write_date(uchar date)                //写数据函数
  60. {
  61.         lcdrs=1;
  62.         P0=date;
  63.         delay(1);
  64.         lcden=1;
  65.         delay(1);
  66.         lcden=0;
  67. }

  68. //自定义字符集
  69. void Lcd_ram()      
  70. {
  71.         uint i,j,k=0,temp=0x40;
  72.         for(i=0;i<7;i++)
  73.         {
  74.            for(j=0;j<8;j++)
  75.            {
  76.             write_com(temp+j);
  77.             write_date(zifu[k]);
  78.             k++;
  79.            }
  80.            temp=temp+8;
  81.         }
  82. }

  83. void init_lcd()                                        //初始化函数
  84. {
  85.         uchar i;                                                                  
  86.         lcden=0;                                        //默认开始状态为关使能端,见时序图  
  87.         Lcd_ram();
  88.         write_com(0x0f);
  89.         write_com(0x38);                        //显示模式设置,默认为0x38,不用变。
  90.         write_com(0x01);                        //显示清屏,将上次的内容清除,默认为0x01.
  91.         write_com(0x0c);                        //显示功能设置0x0f为开显示,显示光标,光标闪烁;0x0c为开显示,不显光标,光标不闪
  92.         write_com(0x06);                        //设置光标状态默认0x06,为读一个字符光标加1.
  93.         write_com(0x80);                        //设置初始化数据指针,是在读指令的操作里进行的
  94.         for(i=10;i<20;i++)                           //显示初始化
  95.         {
  96.                 write_date(table1[i]);                   //显示第一行字符
  97.         }
  98.         write_com(0x80+0x40);                           //选择第二行
  99.         for(i=0;i<9;i++)
  100.         {
  101.                 write_date(table1[i]);                   //显示第二行字符
  102.         }
  103.         write_com(0x80+10);                                   //选择第一行第十个位置
  104.         write_date(0);
  105.         write_date(1);
  106.         write_date(0);
  107.         write_date(1);
  108.         write_date(0);
  109.         write_date(1);                                           //显示自定义的波形图案
  110.         write_com(0x80+0x40+0x09);                   //选择第二行第九个位置
  111.         write_date(' ');
  112.         write_date('1');
  113.         write_date('0');
  114.         write_date('.');
  115.         write_date('0');
  116.         write_date('H');
  117.         write_date('z');                                   //显示初始的频率值
  118. }
  119. void initclock()                           //定时器初始化函数
  120. {
  121.         TMOD=0x01;                                   //定时器的工作方式
  122.         TH0=a;
  123.         TL0=b;                                           //定时器赋初值
  124.         EA=1;                                           //打开中断总开关
  125.         ET0=1;                                           //打开定时器允许中断开关
  126.         TR0=1;                                           //打开定时器定时开关
  127. }
  128. void display()                                   //显示函数
  129. {
  130.         uchar qian,bai,shi,ge;           //定义变量用于显示
  131.         qian=pinlv/1000;                   //将频率值拆成一位的数据,将数据除以1000,得到的商是一位数,赋值给qian
  132.         bai=pinlv%1000/100;                   //将频率除以1000的余数再除以100就得到了频率的百位,赋值给bai
  133.         shi=pinlv%1000%100/10;           //同上,得到频率的十位
  134.         ge=pinlv%1000%100%10;
  135.         write_com(0x80+0x40+0x09); //选中第二行第九个位置

  136.            if(qian==0)                                   //千位如果为0
  137.         write_date(' ');                   //不显示
  138.         else                                           //千位不为0
  139.         write_date(table[qian]);   //正常显示千位
  140.         if(qian==0&&bai==0)                   //千位和百位都为0
  141.         write_date(' ');                   //百位不显示
  142.         else                                           //不都为0
  143.         write_date(table[bai]);           //百位正常显示
  144.         write_date(table[shi]);           //显示十位数
  145.         write_date('.');                   //显示小数点
  146.         write_date(table[ge]);           //显示个位
  147.         write_date('H');                   //显示频率的单位Hz
  148.         write_date('z');
  149.         if(boxing==0)                           //判断波形为正弦波
  150.         {
  151.                 write_com(0x80+10);           //选中一行频率图案位置
  152.                 write_date(0);                   //显示正弦波图案
  153.                 write_date(1);
  154.                 write_date(0);
  155.                 write_date(1);
  156.                 write_date(0);
  157.                 write_date(1);
  158.                                 
  159.         }
  160.         if(boxing==1)                                //注释同上
  161.         {
  162.                 write_com(0x80+10);
  163.                 write_date(2);
  164.                 write_date(3);
  165.                 write_date(2);
  166.                 write_date(3);
  167.                 write_date(2);
  168.                 write_date(3);
  169.         
  170.         }


  171.         if(boxing==2)
  172.         {
  173.                 write_com(0x80+10);
  174.                 write_date(4);
  175.                 write_date(5);
  176.                 write_date(4);
  177.                 write_date(5);
  178.                 write_date(4);
  179.                 write_date(5);
  180.         
  181.         }
  182.         if(boxing==3)
  183.         {
  184.                 write_com(0x80+10);
  185.                 write_date(6);
  186.                 write_date(6);
  187.                 write_date(6);
  188.                 write_date(6);
  189.                 write_date(6);
  190.                 write_date(6);
  191.         
  192.         }
  193. }
  194. void keyscan()                                        //频率调节键盘检测函数
  195. {
  196.         if(s1==0)                                        //加按键是否按下
  197.         {         
  198.                 EA=0;                                        //关闭中断
  199.                 delay(2);                            //延时去抖
  200.                 if(s1==0)                                //再次判断
  201.                 {
  202.                         while(!s1);                        //按键松开
  203.                         pinlv+=bujin;                //频率以步进值加
  204.                         if(pinlv>1000)                //最大加到100Hz
  205.                         {
  206.                                 pinlv=100;                //100Hz
  207.                         }
  208.                         display();                        //显示函数
  209.                         m=65536-(150000/pinlv);//计算频率
  210. /*频率值最小是10Hz,pinlv的值是100(因为要显示小数点后一位),150000/100=1500,这个1500就是定时器需要计时的,单位是us,65536-1500得到的是定时器的初值,
  211. 先不管初值,先看定时时间,1500us,一个波形的周期是由64个定时组成的,所以,一个波形周期就是64*1500us=96000,也就是96ms,约等
  212. 于100ms,也就是10Hz的频率*/
  213.                         a=m/256;                        //将定时器的初值赋值给变量
  214.                         b=m%256;
  215.                         EA=1;                                //打开中断总开关
  216.                 }
  217.         }
  218.         if(s2==0)                                        //减按键按下
  219.         {         
  220.                
  221.                 delay(5);
  222.                 if(s2==0)
  223.                 {
  224.                         EA=0;        
  225.                         while(!s2);
  226.                         pinlv-=bujin;                   //频率以步进值减
  227.                         if(pinlv<100)
  228.                         {
  229.                                 pinlv=1000;
  230.                         }
  231.                         display();
  232.                         m=65536-(150000/pinlv);
  233.                         a=m/256;
  234.                         b=m%256;        
  235.                         EA=1;
  236.                 }
  237.         }
  238.         if(s3==0)                                         //波形切换按键
  239.         {         
  240.                 delay(5);
  241.                 if(s3==0)
  242.                 {
  243.                         EA=0;
  244.                         while(!s3);
  245.                         boxing++;                           //波形切换
  246.                         if(boxing>=4)                   //4种波形
  247.                         {
  248.                                 boxing=0;
  249.                         }
  250.                         display();
  251.                         EA=1;
  252.                 }
  253.         }        
  254. }
  255. void bujindisplay()                                 //步进值设置界面显示程序
  256. {
  257.         uint bai,shi,ge;                         //定义步进值 百十个位
  258.         bai=bujin1/100;                                 //将步进值除以100得到百位,也就是频率值的十位,因为有一个小数位
  259.         shi=bujin1%100/10;                         //将步进值除以100的余数除以十得到十位
  260.         ge=bujin1%100%10;                         //取余10后得到个位,也就是频率步进值的小数点后一位
  261.         write_com(0x80+11);                         //选中液晶第一行第十一列
  262.         if(bai==0)                                         //百位是否为0
  263.         write_date(' ');                         //百位不显示
  264.         else                                                 //百位不为0
  265.         write_date(table[bai]);                 //显示百位数据
  266.         write_date(table[shi]);                 //显示十位数据
  267.         write_date('.');                         //显示小数点
  268.         write_date(table[ge]);                 //显示个位,也就是小数点后一位
  269. }
  270. void bujinjiance()                                 //步进值设置键盘程序
  271. {
  272.         if(s4==0)                                         //步进设置按键按下
  273.         {
  274.                 delay(5);                                 //延时去抖
  275.                 if(s4==0)                                 //再次判断按键
  276.                 {
  277.                          while(!s4);                 //按键释放,按键松开才继续向下执行
  278.                          h++;                                 //变量加
  279.                          if(h==1)                         //进入设置状态时
  280.                          {
  281.                                   write_com(0x01);                          //清屏
  282.                                  write_com(0x80);                          //初始化显示步进设置界面
  283.                                  write_date('S');delay(1);          //step value
  284.                                  write_date('t');delay(1);
  285.                                  write_date('e');delay(1);
  286.                                  write_date('p');delay(1);
  287.                                  write_date(' ');delay(1);
  288.                                  write_date('v');delay(1);
  289.                                  write_date('a');delay(1);
  290.                                  write_date('l');delay(1);
  291.                                  write_date('u');delay(1);
  292.                                  write_date('e');delay(1);
  293.                                  write_date(':');delay(1);
  294.                                  bujin1=bujin;                                  //步进值赋值给临时变量
  295.                                  
  296.                                  bujindisplay();                          //显示步进值
  297.                          }
  298.                          if(h==2)                                                  //退出设置
  299.                          {
  300.                                  h=0;                                                  //清零
  301.                                 bujin=bujin1;                                  //设置好的临时步进值赋值给步进变量
  302.                                 
  303.                                 init_lcd();                                          //初始化液晶显示
  304.                                 initclock();                                  //定时器初始化
  305.                                 display();                                          //调用显示程序
  306.                          }
  307.                 }
  308.         }
  309.         if(h==1)                                                                  //设置步进值时
  310.         {
  311.                 if(s1==0)                                                          //加按键按下
  312.                 {
  313.                         delay(5);                                                  //延时去抖
  314.                         if(s1==0)                                                  //再次判断
  315.                         {
  316.                                 while(!s1);                                          //按键释放
  317.                                 bujin1++;                                          //步进值加1
  318.                                 if(bujin1>=101)                                  //步进值最大100,也就是10.0Hz
  319.                                 {
  320.                                         bujin1=1;                                  //超过最大值就恢复到0.1Hz
  321.                                 }
  322.                                 bujindisplay();                                  //步进显示
  323.                         }
  324.                 }
  325.                 if(s2==0)                                                          //减按键,注释同上
  326.                 {
  327.                         delay(5);
  328.                         if(s2==0)
  329.                         {
  330.                                 while(!s2);
  331.                                 bujin1--;                                          //步进减
  332.                                 if(bujin1<=0)
  333.                                 {
  334.                                         bujin1=100;
  335.                                 }
  336.                                 bujindisplay();
  337.                         }
  338.                 }
  339.         }
  340. }
  341. void main()                                         //主函数
  342. {        
  343.         init_lcd();                                 //调用初始化程序
  344.         m=65536-(150000/pinlv);         //定时器初值
  345.         a=m/256;
  346.         b=m%256;
  347.         initclock();                         //定时器初始化

  348.         while(1)                                 //进入while循环,括号内为1,一直成立,所以也叫死循环,程序不会跳出,一直在内执行
  349.         {
  350.                 if(h==0)                         //正常模式不是步进调节
  351.                 {
  352.                         keyscan();                 //扫描按键
  353.                 //        display();
  354.                 }
  355.                  
  356.                 bujinjiance();                 //扫描步进调节程序

  357.                 switch(boxing)                                                        //选择波形
  358.                 {
  359.                         case 0 : P1=sin[u]; break;                        //正弦波
  360.                         case 1 : P1=juxing[u]; break;                //矩形波
  361.                         case 2 : P1=sanjiao[u]; break;                //三角波
  362.                         case 3 : P1=juchi[u]; break;                //锯齿波
  363.                 }
  364.         }
  365. }
  366. void T0_time()interrupt 1                   //定时器
  367. {
  368.         TH0=a;
  369.         TL0=b;                                                   //根据不同的初值,定时器定时时间不同,达到不同频率的目的
  370.         u++;                                                   //变量加
  371.         if(u>=64)                                           //一个周期采样64个点, 所以加到64就清零
  372.         u=0;                                                   //u清零
  373. }
复制代码

可调
全部资料51hei下载地址:
仿真.zip (122.49 KB, 下载次数: 144)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:817203 发表于 2020-9-11 09:28 来自手机 | 显示全部楼层
谁能画一下PCB分享出来吗?
回复

使用道具 举报

ID:980857 发表于 2021-12-3 10:31 | 显示全部楼层
没有PCB??
回复

使用道具 举报

ID:262 发表于 2021-12-3 18:33 | 显示全部楼层

这个是没有,刚帮你找了一下其他的 这里有2个朋友上传了PCB,这种作品很常见论坛里面应该还有很多 你可以自己找一下 搜索框 输入"波形 PCB"
http://www.51hei.com/bbs/dpj-162263-1.html
http://www.51hei.com/bbs/dpj-198756-1.html
回复

使用道具 举报

ID:986539 发表于 2021-12-15 14:09 | 显示全部楼层
为什么显示正弦波图案那里只用写入0101就行了?上面定义的字符有什么作用?
回复

使用道具 举报

ID:986539 发表于 2021-12-15 18:36 | 显示全部楼层
heicad 发表于 2021-12-3 18:33
这个是没有,刚帮你找了一下其他的 这里有2个朋友上传了PCB,这种作品很常见论坛里面应该还有很多 你可以自 ...


为什么显示正弦波图案那里只用写入0101就行了?上面定义的字符有什么作用?
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表