找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 2222|回复: 0
打印 上一主题 下一主题
收起左侧

TCS230颜色识别模块LCD1602显示单片机代码

[复制链接]
跳转到指定楼层
楼主
ID:136393 发表于 2019-4-13 09:38 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
实现功能:实现功能:对颜色传感器输出RGB
使用芯片:AT89S52 或者 STC89C52
晶振:12MHZ
编译环境:Keil

单片机源程序如下:
  1. #include <reg52.h>
  2. #define uchar unsigned char
  3. #define uint  unsigned int
  4. //==============LCD1602接口连接方法=====================
  5. /*-----------------------------------------------------
  6.        |DB0-----P0.0 | DB4-----P0.4 | RW-------P2.3    |
  7.        |DB1-----P0.1 | DB5-----P0.5 | RS-------P2.4    |
  8.        |DB2-----P0.2 | DB6-----P0.6 | E--------P2.2    |
  9.        |DB3-----P0.3 | DB7-----P0.7 |
  10.     ---------------------------------------------------*/
  11. //================================================*/              
  12. #define LCM_Data     P0    //LCD1602数据接口
  13. #define Busy         0x80   //用于检测LCM状态字中的Busy标识
  14. sbit    LCM_RW     = P2^3;  //读写控制输入端,LCD1602的第五脚
  15. sbit    LCM_RS     = P2^4;  //寄存器选择输入端,LCD1602的第四脚
  16. sbit    LCM_E      = P2^2;  //使能信号输入端,LCD1602的第6脚
  17. sbit    LED1=P1^0;
  18. sbit    LED2=P1^1;
  19. sbit    LED3=P1^2;

  20. //=================颜色传感模块连接=====================
  21. /*-----------------------------------------------------
  22.        |EO-----GND
  23.        |S0-----VCC | S2-----P2.0 | OUT-------P3.5
  24.        |S1-----VCC | S3-----P2.1 |
  25.   ---------------------------------------------------*/
  26. sbit    tcs230_s2=P2^0;//TCS230 S2接单片机P2.0
  27. sbit    tcs230_s3=P2^1;//TCS230 S3接单片机P2.1
  28. sbit    tcs230_en=P3^0; //TCS230 EN(E0)接GND
  29. //**************函数声明***************************************
  30. void    WriteDataLCM                (uchar WDLCM);//LCD模块写数据
  31. void    WriteCommandLCM        (uchar WCLCM,BuysC); //LCD模块写指令
  32. uchar   ReadStatusLCM(void);//读LCD模块的忙标
  33. void    DisplayOneChar(uchar X,uchar Y,uchar ASCII);//在第X+1行的第Y+1位置显示一个字符
  34. void    LCMInit(void);//LCD初始
  35. void    DelayMs(uint Ms);//1MS基准延时程序
  36. void    baipingheng();//白平衡子程序
  37. void    celiang();//实际颜色程序
  38. uint    ryz,gyz,byz;//分别定义红色因子 绿色因子 蓝色因子
  39. uint    rb,gb,bb,z;//RGB值
  40. uchar   tab1[]={'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'};
  41. //***********************主程序******************************
  42. main()   
  43. {   
  44. P0=0xff;
  45.    TMOD=0x51;//设定T0以工作方式1定时10毫秒
  46.          LCMInit();//LCD初始
  47.    baipingheng();//上电时先白平衡一次
  48.    while(1)
  49.    {
  50.                          celiang();//颜色测试
  51.                    z = rb+gb+bb;
  52.        DisplayOneChar(0, 0,'R');//??????RGB??????
  53.        DisplayOneChar(0, 1, rb/100+0x30); //??????
  54.        DisplayOneChar(0, 2, rb/10%10+0x30);//??????
  55.        DisplayOneChar(0, 3, rb%10+0x30);//??????
  56.        DisplayOneChar(0, 5,'G');//??????RGB??????
  57.        DisplayOneChar(0, 6, gb/100+0x30); //??????
  58.        DisplayOneChar(0, 7, gb/10%10+0x30);
  59.        DisplayOneChar(0, 8, gb%10+0x30);
  60.        DisplayOneChar(0, 10,'B');//??????RGB??????
  61.        DisplayOneChar(0, 11, bb/100+0x30);
  62.        DisplayOneChar(0, 12, bb/10%10+0x30);
  63.        DisplayOneChar(0, 13, bb%10+0x30);
  64.                    DisplayOneChar(1, 6, 'Y');
  65.                    DisplayOneChar(1, 7, 'u');
  66.                    DisplayOneChar(1, 8, 'a');
  67.                    DisplayOneChar(1, 9, 'n');
  68.        //*****?LCD1602?????16????RGB*******************
  69. //       DisplayOneChar(1, 1, tab1[rb/16]);
  70. //       DisplayOneChar(1, 2, tab1[rb%16]);
  71. //       DisplayOneChar(1, 3, 'H');
  72. //       DisplayOneChar(1, 6, tab1[gb/16]);
  73. //       DisplayOneChar(1, 7, tab1[rb%16]);
  74. //       DisplayOneChar(1, 8, 'H');
  75. //       DisplayOneChar(1, 11,tab1[bb/16]);
  76. //       DisplayOneChar(1, 12,tab1[bb%16]);
  77. //       DisplayOneChar(1, 13,'H');



  78.      if (tab1[rb/16]>=tab1[gb/16])                  //红色
  79.                    {
  80.                  P1=0xfe;
  81.                  DisplayOneChar(1, 3, '1');
  82.                  DisplayOneChar(1, 4, '0');
  83.                  DisplayOneChar(1, 5, '0');


  84.                }

  85.                         if( tab1[gb/16]>=tab1[bb/16])          //绿色
  86.                    {
  87.                  P1=0xfd;
  88.                  DisplayOneChar(1, 3, '5');
  89.                  DisplayOneChar(1, 4, '0');
  90.                  DisplayOneChar(1, 5, ' ');
  91.                }

  92.                     if(tab1[bb/16]>=tab1[rb/16])          //蓝色
  93.                    {
  94.                  P1=0xfb;
  95.                  DisplayOneChar(1, 3, '1');
  96.                  DisplayOneChar(1, 4, '0');
  97.                  DisplayOneChar(1, 5, ' ');
  98.                }

  99.            DelayMs(250);//每隔0.25秒测试一次颜色         
  100.    }
  101. }
  102. //******************************************************
  103. //白平衡子程序
  104. void   celiang()
  105. {
  106.      //*********求R值************************************
  107.      TH0=(65536-10000)/256;
  108.            TL0=(65536-10000)%256;
  109.      TH1=0;
  110.      TL1=0;
  111.      tcs230_s2=0;
  112.      tcs230_s3=0;//选择红色滤光器
  113.      tcs230_en=0;
  114.      TR0=1;//10毫秒开始计时
  115.      TR1=1;//开始计数
  116.      while(TF0==0);//等待定时器溢出
  117.      TF0=0;//清楚定时器0溢出标志
  118.      TR0=0;//关闭定时0
  119.      TR1=0;
  120.      rb=(unsigned long)(TH1*256+TL1)*255/ryz-30;
  121.      if(rb>255)rb=255;//判断RGB值是否合法
  122.      //***********求B值**************************************
  123.      TH0=(65536-10000)/256;
  124.            TL0=(65536-10000)%256;
  125.      TH1=0;
  126.      TL1=0;
  127.      tcs230_s2=0;
  128.      tcs230_s3=1;//选择蓝色滤光器
  129.      TR0=1;//10毫秒开始计时
  130.      TR1=1;//开始计数
  131.      while(TF0==0);//等待定时器溢出
  132.      TF0=0;//清楚定时器0溢出标志
  133.      TR0=0;//关闭定时0
  134.      TR1=0;
  135.      bb=(unsigned long)(TH1*256+TL1)*255/byz-70;
  136.      if(bb>255)bb=255;//判断RGB值是否合法     
  137.      //***********求G值**************************************   
  138.      TH0=(65536-10000)/256;
  139.            TL0=(65536-10000)%256;
  140.      TH1=0;
  141.      TL1=0;
  142.      tcs230_s2=1;
  143.      tcs230_s3=1;//选择绿色滤光器
  144.      TR0=1;//10毫秒开始计时
  145.      TR1=1;//开始计数
  146.      while(TF0==0);//等待定时器溢出
  147.      TF0=0;//清楚定时器0溢出标志
  148.      TR0=0;//关闭定时0
  149.      TR1=0;
  150.      tcs230_en=1;
  151.      gb=((unsigned long)(TH1*256+TL1)*255/gyz)-70;
  152.      if(gb>255)gb=255;//判断RGB值是否合法  
  153. }
  154. //******************************************************
  155. //白平衡子程序
  156. void    baipingheng()
  157. {
  158.      //**************求取红色因子***********************
  159.      TH0=(65536-10000)/256;
  160.            TL0=(65536-10000)%256;
  161.      TH1=0;
  162.      TL1=0;
  163.      tcs230_s2=0;
  164.      tcs230_s3=0;//选择红色滤光器
  165.      tcs230_en=0;
  166.      TR0=1;//10毫秒开始计时
  167.      TR1=1;//开始计数
  168.      while(TF0==0);//等待定时器溢出
  169.      TF0=0;//清楚定时器0溢出标志
  170.      TR0=0;//关闭定时0
  171.      TR1=0;
  172.      ryz=TH1*256+TL1;//其实这里的比例因子应该为255/(TH1*256+TL1)
  173.      //**************求取蓝色因子***********************
  174.      TH0=(65536-10000)/256;
  175.            TL0=(65536-10000)%256;
  176.      TH1=0;
  177.      TL1=0;
  178.      tcs230_s2=0;
  179.      tcs230_s3=1;//选择蓝色滤光器
  180.      TR0=1;//10毫秒开始计时
  181.      TR1=1;//开始计数
  182.      while(TF0==0);//等待定时器溢出
  183.      TF0=0;//清楚定时器0溢出标志
  184.      TR0=0;//关闭定时0
  185.      TR1=0;
  186.      byz=TH1*256+TL1;//其实这里的比例因子应该为255/(TH1*256+TL1)
  187.      //**************求绿红色因子***********************
  188.      TH0=(65536-10000)/256;
  189.            TL0=(65536-10000)%256;
  190.      TH1=0;
  191.      TL1=0;
  192.      tcs230_s2=1;
  193.      tcs230_s3=1;//选择绿色滤光器
  194.      TR0=1;//10毫秒开始计时
  195.      TR1=1;//开始计数
  196.      while(TF0==0);//等待定时器溢出
  197.      TF0=0;//清楚定时器0溢出标志
  198.      TR0=0;//关闭定时0
  199.      TR1=0;
  200.      tcs230_en=1;
  201.      gyz=TH1*256+TL1;//其实这里的比例因子应该为255/(TH1*256+TL1)
  202. }
  203. /*======================================================================
  204. LCM初始化
  205. ======================================================================*/
  206. void LCMInit(void)
  207. {
  208. LCM_Data = 0;
  209. WriteCommandLCM(0x38,0); //三次显示模式设置,不检测忙信号
  210. DelayMs(5);
  211. WriteCommandLCM(0x38,0);
  212. DelayMs(5);
  213. WriteCommandLCM(0x38,0);
  214. DelayMs(5);
  215. WriteCommandLCM(0x38,1); //显示模式设置,开始要求每次检测忙信号
  216. WriteCommandLCM(0x08,1); //关闭显示
  217. WriteCommandLCM(0x01,1); //显示清屏
  218. WriteCommandLCM(0x06,1); // 显示光标移动设置
  219. WriteCommandLCM(0x0C,1); // 显示开及光标设置
  220. DelayMs(100);
  221. }
  222. //==============================LCD1602显示子程序================================================
  223. // 写数据函数: E =高脉冲 RS=1 RW=0
  224. //======================================================================*/
  225. void WriteDataLCM(uchar WDLCM)
  226. {
  227. ReadStatusLCM(); //检测忙
  228. LCM_Data = WDLCM;
  229. LCM_RS = 1;
  230. LCM_RW = 0;
  231. LCM_E = 0; //若晶振速度太高可以在这后加小的延时
  232. LCM_E = 0; //延时
  233. LCM_E = 1;
  234. }
  235. /*====================================================================
  236.   写指令函数: E=高脉冲 RS=0 RW=0
  237. ======================================================================*/
  238. void WriteCommandLCM(uchar WCLCM,BuysC) //BuysC为0时忽略忙检测
  239. {
  240. if (BuysC) ReadStatusLCM(); //根据需要检测忙
  241. LCM_Data = WCLCM;
  242. LCM_RS = 0;
  243. LCM_RW = 0;
  244. LCM_E = 0;
  245. LCM_E = 0;
  246. LCM_E = 1;
  247. }
  248. /*====================================================================
  249.   正常读写操作之前必须检测LCD控制器状态:E=1 RS=0 RW=1;
  250.   DB7: 0 LCD控制器空闲,1 LCD控制器忙。
  251.   读状态
  252. ======================================================================*/
  253. uchar ReadStatusLCM(void)
  254. {
  255. LCM_Data = 0xFF;
  256. LCM_RS = 0;
  257. LCM_RW = 1;
  258. LCM_E = 0;
  259. LCM_E = 0;
  260. LCM_E = 1;
  261. while (LCM_Data & Busy); //检测忙信号  
  262. return(LCM_Data);
  263. }
  264. /*======================================================================
  265. 功 能:     在1602 指定位置显示一个字符:第一行位置0~15,第二行16~31
  266. 说 明:     第 X 行,第 y 列  注意:字符串不能长于16个字符
  267. ======================================================================*/
  268. void DisplayOneChar( uchar X, uchar Y, uchar ASCII)
  269. {
  270.   X &= 0x1;
  271.   Y &= 0xF; //限制Y不能大于15,X不能大于1
  272.   if (X) Y |= 0x40; //当要显示第二行时地址码+0x40;
  273.   Y |= 0x80; // 算出指令码
  274.   WriteCommandLCM(Y, 0); //这里不检测忙信号,发送地址码
  275.   WriteDataLCM(ASCII);
  276. }
  277. /*====================================================================
  278.   设定延时时间:x*1ms
  279. ====================================================================*/
  280. void DelayMs(uint Ms)
  281. {
  282.   uint i,TempCyc;
  283.   for(i=0;i<Ms;i++)
  284.   {
  285.     TempCyc = 250;
  286.     while(TempCyc--);
  287.   }
  288. }
复制代码



分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏2 分享淘帖 顶 踩
回复

使用道具 举报

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

本版积分规则

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

Powered by 单片机教程网

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