找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 10116|回复: 17
收起左侧

基于stc8g1k08a单片机的简易示波器和超声波测距二合一小制作 附程序

  [复制链接]
ID:849913 发表于 2021-4-22 15:20 | 显示全部楼层 |阅读模式
stc8g1k08a是深圳宏晶科技的一款8脚8位兼容51系列的单片机  价格很便宜 但麻雀虽小五脏俱全 且速度还挺快下面就是用它做的小东东 精度不是很高 本人也是刚入门 水平有限 希望大家多多指教共两个按键 :按住周期和启动按键5秒钟后两个程序互相切换  短按示波器是切换扫描周期  超声波测距是启动按键  波形锁定是示波器波形固定用的

制作出来的实物图如下:
51hei图片_20210422143655.jpg 51hei图片_20210422143700.jpg 51hei图片_20210422143705.jpg 51hei图片_20210422143632.jpg

电路原理图如下:
51hei图片_20210422143642.jpg

单片机源程序如下:
  1. #include "reg51.h"
  2. #include "intrins.h"
  3. #include"main.h"
  4. #include<math.h>

  5. unsigned char xdata se[128];//把AD转换后的临时数据存于该数组
  6. unsigned char code sm[8]={0x01,0x02,0x04,0x08,0x10,0x20,0x40,0x80};//用于显示一列中的一点
  7. unsigned int code time[9]={1,40,100,500,1000,2600,5250,10000,25000};
  8. unsigned int code time1[9]={1,3,5,20,40,100,200,400,1000};
  9. unsigned char xdata timd[5];
  10. unsigned char xdata ju_li[5];
  11. unsigned char p,z=0;
  12. unsigned int i=0,q=0;
  13. unsigned int m=5;
  14. //unsigned double q;
  15. sbit k1=P3^1;
  16. sbit trig=P3^3;
  17. sbit echo=P3^2;
  18. void delay_us(unsigned int i)
  19. {
  20.         while(i)
  21.         {
  22.           unsigned char a;
  23.     for(a=5;a>0;a--);
  24.         i--;
  25.         }
  26.        
  27. }
  28. void delay100ms(unsigned int i)   //误差 -0.000000000025us
  29. {
  30.    while(i)
  31.    {
  32.    unsigned char a,b,c;
  33.     for(c=106;c>0;c--)
  34.         for(b=197;b>0;b--)
  35.             for(a=38;a>0;a--);
  36.     _nop_();  //if Keil,require use intrins.h
  37.         i--;
  38.         }
  39. }
  40. void main()
  41. {
  42.         P0M1 = 0;   P0M0 = 0;   //设置为准双向口
  43.     P1M1 = 0;   P1M0 = 0;   //设置为准双向口
  44.     P2M1 = 0;   P2M0 = 0;   //设置为准双向口
  45.     P3M1 = 0;   P3M0 = 0;   //设置为准双向口
  46.     P4M1 = 0;   P4M0 = 0;   //设置为准双向口
  47.     P5M1 = 0;   P5M0 = 0;   //设置为准双向口
  48.     /*        P_SW2 = 0x80;  //特殊寄存器允许位(0x80)
  49.         CKSEL = 0x00; //选择内部 IRC ( 默认 )
  50.         CLKDIV = 0x03; //时钟 3分频
  51.         P_SW2 = 0x00;
  52.         */       
  53.         ad_init();        //ad转换初始化
  54.     OLED_Init(); //OLED初始化
  55.           
  56.         for(z=0;z<128;z++)                             // 起   显示启动示波器文字
  57.         {
  58.                   for(i=0;i<8;i++)   //清屏
  59.                 {
  60.                                  OLED_WrCmd(0xb0+i);
  61.                                 OLED_WrCmd(0x00+(z%16));
  62.                         OLED_WrCmd(0x10+z/16);
  63.                                 OLED_WrDat(0x00);
  64.                 }
  65.         }
  66.                 for (i=0;i<8;i++)
  67.          {
  68.                 OLED_P16x16Ch(i*16,i/8*2+2,i);
  69.          }
  70.          delay100ms(30);
  71.                  for(z=0;z<128;z++)               
  72.         {
  73.                   for(i=0;i<8;i++)   //清屏
  74.                 {
  75.                                  OLED_WrCmd(0xb0+i);
  76.                                 OLED_WrCmd(0x00+(z%16));
  77.                         OLED_WrCmd(0x10+z/16);
  78.                                 OLED_WrDat(0x00);
  79.                 }
  80.         }                                                                                //止


  81.   while(1)
  82.   {
  83.         if(k1==0)
  84.         {
  85.                  i=0;
  86.                 m++;
  87.                  while(!k1)
  88.                  {
  89.                         
  90.                         delay100ms(1);
  91.                         i++;

  92.                        
  93.                         if(i>50)

  94.                         goto bosh;
  95.                        
  96.                  }
  97.                  
  98.                  if(m>8)
  99.                  {
  100.                          m=0;
  101.                  }
  102.                  //i++;
  103.      }
  104. //*******************AD转换将临时数据存于数组se[] ****************************************


  105.                 for(i=0;i<128;i++)
  106.         {
  107.                 se[i]=56-ad_dat()/18;
  108.                 delay(time[m]);

  109.         }
  110.                  timd[0]=time1[m]%10;
  111.                  timd[1]=time1[m]/10%10;
  112.                  timd[2]=time1[m]/100%10;
  113.                  timd[3]=time1[m]/1000%10;
  114.                  timd[4]=time1[m]/10000%10;
  115.                  OLED_P6x8Str(10,7,"Time = ");
  116.          OLED_P6x8Str(10+70,7," ms");

  117.        
  118.          for(i=0;i<5;i++)                                          //显示扫描周期每一位 共五位
  119.          {
  120.          oled_6x8((10+40)+6*i,7,timd[4-i]+16);
  121.                  }
  122.                  


  123. //**********************oled显示****************************
  124. /*    for(i=0;i<7;i++)
  125.        
  126.            {
  127.                                 OLED_WrCmd(0xb0+i);
  128.                                 OLED_WrCmd(0x00+(z%16));
  129.                         OLED_WrCmd(0x10+z/16);
  130.                                 OLED_WrDat(0x00);
  131.            }
  132.        
  133.         */
  134.           
  135.         for(z=0;z<128;z++)
  136.         {
  137.        
  138.                 for(i=0;i<7;i++)
  139.        
  140.            {
  141.                                 OLED_WrCmd(0xb0+i);
  142.                                 OLED_WrCmd(0x00+(z%16));
  143.                         OLED_WrCmd(0x10+z/16);
  144.                                 if((i==3)&&(z%4==0))
  145.                                 OLED_WrDat(0x10);
  146.                                 else if(z==63)
  147.                                 OLED_WrDat(0x44);
  148.                                 else
  149.                                 OLED_WrDat(0x00);
  150.            }
  151.                 OLED_WrCmd(0xb0+se[z]/7);
  152.                 OLED_WrCmd(0x00+(z%16));
  153.                 OLED_WrCmd(0x10+z/16);
  154.                 OLED_WrDat(sm[se[z]%7]);       
  155.        
  156.                
  157.         //        OLED_WrDat(0x00);
  158.                
  159.         }
  160.        
  161.          if(trig==0)
  162.          {
  163.                  delay100ms(10);
  164.                 while(trig);
  165.                 while(!trig);
  166.                
  167.          }  
  168.          
  169.   }
  170.   

  171.         //启动超声波测距程序。。。。。。
  172.         bosh:
  173.                 for(z=0;z<128;z++)
  174.         {
  175.                   for(i=0;i<8;i++)   //清屏
  176.                 {
  177.                                  OLED_WrCmd(0xb0+i);
  178.                                 OLED_WrCmd(0x00+(z%16));
  179.                         OLED_WrCmd(0x10+z/16);
  180.                                 OLED_WrDat(0x00);
  181.                 }
  182.         }
  183.                 for (i=0;i<8;i++)
  184.          {
  185.                 OLED_P16x16Ch(i*16,i/8*2+2,i+8);
  186.          }
  187.                        
  188.        
  189.         delay100ms(30);
  190.                 for(z=0;z<128;z++)
  191.         {
  192.                   for(i=0;i<8;i++)   //清屏
  193.                 {
  194.                                  OLED_WrCmd(0xb0+i);
  195.                                 OLED_WrCmd(0x00+(z%16));
  196.                         OLED_WrCmd(0x10+z/16);
  197.                                 OLED_WrDat(0x00);
  198.                 }
  199.         }

  200.                        
  201.         while(1)
  202.   {
  203.                   if(k1==0)
  204.         {
  205.                  q=0;
  206.                 trig=0;
  207.                 delay_us(1000);
  208.                 trig=1;
  209.        
  210.                 i=10;
  211.                  while(!echo)
  212.                  {
  213.                          delay_us(1);
  214.                         i--;
  215.                 i=0;       
  216.                  }
  217.                  while(echo)
  218.                  {
  219.                   delay_us(1);
  220.                   q++;       
  221.                  }
  222.                  q=q*0.21;
  223.                        
  224.        


  225.                  while(!k1)
  226.                  {
  227.                          i++;
  228.                 delay100ms(1);

  229.                        
  230.                         if(i>50)
  231.                         return;
  232.              }
  233.                         
  234.                  
  235.                

  236.     }
  237.             //q=156;
  238.                 ju_li[0]=q%10;
  239.                 ju_li[1]=q/10%10;
  240.                 ju_li[2]=q/100%10;
  241.                 ju_li[3]=q/1000%10;
  242.                 ju_li[4]=q/10000%10;
  243.          for(i=0;i<5;i++)
  244.          {
  245.          oled1_8x16( 48+8*i,4,ju_li[4-i]);
  246.          }
  247.                  for (i=0;i<6;i++)
  248.          {
  249.                 OLED_P16x16Ch(i*16,i/8*2,19+i);
  250.          }
  251.                         for (i=0;i<3;i++)
  252.          {
  253.                 OLED_P16x16Ch(i*16,i/8*2+2+2,19+6+i);

  254.          }
  255.                         for (i=0;i<2;i++)
  256.          {
  257.                 OLED_P16x16Ch(95+i*16,i/8*2+2+2,19+6+3+i);
  258.          }
  259.          
  260.   }
  261. }
复制代码
51hei.png
所有资料51hei提供下载:
ad_led.rar (66.39 KB, 下载次数: 208)

评分

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

查看全部评分

回复

使用道具 举报

ID:849913 发表于 2021-4-22 18:55 | 显示全部楼层
谢谢版主帮我整理 忘了告诉大家 单片机下载程序是 时钟选择33.1776MZ
回复

使用道具 举报

ID:857072 发表于 2021-4-24 11:15 来自手机 | 显示全部楼层
有意思,下来有时间研究一下
回复

使用道具 举报

ID:756506 发表于 2021-4-24 11:21 | 显示全部楼层
不错,有时间来做一个
回复

使用道具 举报

ID:431787 发表于 2021-5-2 20:10 | 显示全部楼层
keil5好像编译不成功
回复

使用道具 举报

ID:915390 发表于 2021-5-3 22:00 来自手机 | 显示全部楼层
学到了,有空弄一个
回复

使用道具 举报

ID:601214 发表于 2021-5-30 10:00 | 显示全部楼层
这个能做成示波器吗?
回复

使用道具 举报

ID:849913 发表于 2021-5-30 21:41 | 显示全部楼层
zhike 发表于 2021-5-30 10:00
这个能做成示波器吗?

简易的 都是出于自己对示波器工作原理的理解  频率和精度都不高 主要是出于兴趣玩玩
回复

使用道具 举报

ID:414556 发表于 2021-5-30 23:19 | 显示全部楼层
测距精度高不高,有没有精度微调
回复

使用道具 举报

ID:849913 发表于 2021-6-3 15:26 | 显示全部楼层
测距精度可以用软件调节 前提就是做着玩的 目的是对示波器和测试仪构思的实践
回复

使用道具 举报

ID:927469 发表于 2021-7-5 22:09 | 显示全部楼层
lin2453676722 发表于 2021-5-2 20:10
keil5好像编译不成功

把文件头改一下就能编译
#include "reg51.h"
#include "intrins.h"
#include"main.h"
#include<math.h>
改成:
#include <STC8G.h>
#ifndef     __STC8G_H__
#define     __STC8G_H__
回复

使用道具 举报

ID:849913 发表于 2021-7-10 11:42 | 显示全部楼层
kmikmi 发表于 2021-7-5 22:09
把文件头改一下就能编译
#include "reg51.h"
#include "intrins.h"

谢谢指点 我还不知道有这个方法  所以把reg51.h里面加了很多与8G相关的东西
回复

使用道具 举报

ID:953759 发表于 2021-7-17 10:57 | 显示全部楼层
老董 发表于 2021-4-22 18:55
谢谢版主帮我整理 忘了告诉大家 单片机下载程序是 时钟选择33.1776MZ

您好,这个时钟是怎么确定的呢?
回复

使用道具 举报

ID:558833 发表于 2022-2-14 21:29 | 显示全部楼层
面包板插线,直接下载成功
回复

使用道具 举报

ID:185675 发表于 2022-6-8 22:32 | 显示全部楼层
//一个简单的在stc8g1ko8A上可以运行的小demo,可以用于简单的AD测量,define修改一下就行了
#define TIME_3S 30000
#define TIME_2S 30000
#define TIME_1MS 100
#define  TIME_2MS 200
#define  TIME_5MS 500
// 状态机
#define READY_ST 1
#define CAPCHARGE 2
#define MOT_ST 3
#define STOP_ST 4
#define CHARGE_ST 5
#define DELAY_ST 6
// 信号接收
// 状态机
#define keyRun
#define keyMot
// 数值
#define U500 20
#define U6500 500
#define U8000 700

unsigned int STATUS
unsigned int ADS
// 计数器
unsigned int keyRunCounter
unsigned int keyShutDownCounter
unsigned int shortCurrentCounter
unsigned int delaySTCounter
unsigned int capChargeCounter
unsigned int OVMonitCounter_1
unsigned int OVMonitCounter_2


mian()
{
case Status
    Run:
        Gpio.init;//keyRun 为推挽
    Mot:
        Gpio.init2;//keyRun 为input
    default:
        Gpio.init;//初始化为Run
}
void ISR()
{
    /*滤波器*/

    /*保护*/
    Protect();
    /*按键检测*/
    keyMonitor();

    /*状态与动作*/
    switch STATUS:
    case READY_ST:
        setGPIO1;
    case CHARGE_ST:
        setGPIO1();
    case DELAY_ST:
        setGPIO1();
    case MOT_ST:
        setGPIO1();
    case STOP_ST:
        setGPIO1();   
    default:   
}


/******************************
*执行保护逻辑
****************************/
void Protect()
{
    // Short State monitor
    if (STATUS != READY_ST) && (ADS < U500)//在待机时不检测,其他时段都不放过
    {
        shortCurrentCounter++;
    }
    else
    {
        shortCurrentCounter = 0;
    }
    if (shortCurrentCounter > TIME_1MS)
    {
        shortCurrentCounter = 0;
        STATUS = STOP_ST;
        return;
    }
    // OverVoltage Monitor 1
    if (STATUS != READY_ST) && (ADS > U500)//
    {
        OVMonitCounter_1++;
    }
    else
    {
        OVMonitCounter_1 = 0;
    }
    if (OVMonitCounter_1 > OVMonitCounter_1Debounce)
    {
        OVMonitCounter_1 = 0;
        STATUS = DELAY_ST;
    }
    // OverVoltage Monitor 2
    if (STATUS != READY_ST) && (ADS > U7500)//
    {
        OVMonitCounter_2++;
    }
    else
    {
        OVMonitCounter_2 = 0;
    }
    if (OVMonitCounter_2 > OVMonitCounter_2Debounce)
    {
        OVMonitCounter_2 = 0;
        STATUS = DELAY_ST;
    }
}
/******************************
*按键检测逻辑
****************************/
void keyMonitor()
{
    //待机状态,长按*秒开机
    if (STATUS == READY_ST) && (keyRun == 1)//待机状态,长按3秒开机
    {
        keyRunCounter++;
    }
    else
    {
        keyRunCounter = 0;
    }
    if (keyRunCounter > TIME_3S)
    {
        keyRunCounter = 0;
        STATUS = CHARGE_ST;
    }
    //运行状态,信号消失1ms关机
    if ((STATUS == CHARGE_ST) || (STATUS == DELAY_ST) || (STATUS == CAPCHARGE)  )&& (keyRun == 0)
    {
        keyShutDownCounter++;
    }
    else
    {
        keyShutDownCounter = 0;
    }
    if (keyShutDownCounter > TIME_3S)
    {
        keyShutDownCounter = 0;
        STATUS = READY_ST;
    }
   
    /*母线大电容充电2秒,开始工作*/
    if (STATUS == CAPCHARGE)  
    {
        if  (capChargeCounter < TIME_2S)
        {
            capChargeCounter++;
        }
        else
        {
            capChargeCounter = 0;
            STATUS = CHARGE_ST;
        }
    }
    //delay状态,1ms后使能检测
    if ((STATUS == DELAY_ST) ) //&& (ADS < U500) delay 之后1ms再次检测,不够再加,够了就再次进入delay
    {
        delaySTCounter++;
    }
    else
    {
        delaySTCounter = 0;
    }
    if (delaySTCounter > TIME_2MS)
    {
        delaySTCounter = 0;
        STATUS = CHARGE_ST;
    }
}

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

ID:1042937 发表于 2022-8-27 15:48 | 显示全部楼层
有时间研究一下
回复

使用道具 举报

ID:476058 发表于 2022-10-15 22:33 | 显示全部楼层
共同学习,共同进步,向大佬学习!
回复

使用道具 举报

ID:681572 发表于 2023-4-14 09:17 | 显示全部楼层
请问这个是用keil4还是keil5建立的工程啊??建立工程时选择哪个器件啊?为什么我的keil4 C51里面没有这个stc8g1k08a器件啊???
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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