基于STC12C5A60S2系列1T 8051单片机的模数芯片ADC0832实现模数转换应用

news2025/2/24 14:38:14

基于STC12C5A60S2系列1T 8051单片的模数芯片ADC0832实现模数转换应用

  • STC12C5A60S2系列1T 8051单片机管脚图
  • STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置
  • STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍
  • 模数芯片ADC0832介绍
    • 通过模数芯片ADC0832把电压模拟量转化为电压数字量

STC12C5A60S2系列1T 8051单片机管脚图

在这里插入图片描述在这里插入图片描述

STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式及配置

在这里插入图片描述

STC12C5A60S2系列1T 8051单片机I/O口各种不同工作模式介绍

在这里插入图片描述在这里插入图片描述

模数芯片ADC0832介绍

在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述
在这里插入图片描述

通过模数芯片ADC0832把电压模拟量转化为电压数字量

#include <stc12c5a60s2.h>
#define uchar unsigned char//自定义无符号字符型为uchar
#define uint unsigned int//自定义无符号整数型为uint 
#define NixieTubeSegmentCode P0//自定义数码管段码为单片机P0组引脚
#define NixieTubeBitCode P2//自定义数码管位码为单片机P2组引脚
//#define KeyPressDeshakeTime 10//自定义按键按下消抖时间为10ms
//#define KeyLongPressDelayTime 500//自定义按键长按延时时间为500ms
//#define KeyLongPressIntervalChangeTime 25//自定义按键长按间隔变化时间为25ms
//uchar AddKeyLockFlag;//声明增加按键锁定标志位变量
//uchar DecKeyLockFlag;//声明减少按键锁定标志位变量
//uchar KeyNumber = 0;//定义按键键值为0
//uchar AddKeyLongPressAddIntervalTime;//声明增加按键长按连增间隔时间变量
//uchar DecKeyLongPressDecIntervalTime;//声明减少按键长按连减间隔时间变量
//uchar NumberValue;//声明数字量变量
//uint AddKeyPressDelayTime;//声明增加按键按下延时时间变量
//uint DecKeyPressDelayTime;//声明减少按键按下延时时间变量
uchar Code NixieTubeBitCodeArray = [0xfe,0xfd,0xfb,0xf7];//定义共阴数码管位码数组变量
uchar NixieTubeDisplayDataArray[0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71,0x40,0x00];//定义共阴数码管显示0~F数据、符号“—”及熄灭数组变量
uchar NixieTubeCacheDataArray[];//定义数码管缓存数据数组变量
float OutPutVoltage;//声明输出电压变量
float AnalogFilterOutPutVoltage;//声明模拟滤波后输出电压变量
//uint Timer0TimeCount;//声明定时器0定时计数变量
//sbit AddKey = P2^0;//位定义增加按键为单片机P2.0引脚
//sbit DecKey = P2^1;//位定义减少按键为单片机P2.1引脚
sbit ADC0832CS = P1^0;//位定义ADC0832片选变量为P1.0端口
sbit ADC0832CLK = P1^1;//位定义ADC0832时钟变量为P1.1端口
sbit ADC0832DI = P1^2;//位定义ADC0832数据输入变量为P1.2端口
sbit ADC0832DO = P1^2;//位定义ADC0832数据输出变量为P1.2端口
 void ADC0832Init()//ADC0832初始化函数
{
  ADC0832CS = 1;//ADC0832片选变量置高电平
  ADC0832CLK = 0;//ADC0832时钟变量置低电平
  ADC0832DI = 1;//ADC0832数据输入变量置高电平
 }
 void ADC0832Change()//ADC0832转化函数
{
  uchar AnalogDataResult1;//声明高位先移模拟数据变量
  uchar AnalogDataResult2;//声明低位先移模拟数据变量
  ADC0832CS = 0;//ADC0832片选端口拉低电平
  ADC0832CLK = 0;//ADC0832时钟变量置低电平
  ADC0832DI = 1;//ADC0832数据输入变量置高电平
  _nop_();//
  ADC0832CLK = 1;//ADC0832时钟变量置高电平 第一个脉冲 启动A/D转换
  _nop_();//
  ADC0832CLK = 0;//ADC0832时钟变量置低电平
  ADC0832DI = 1;//ADC0832数据输入变量置高电平 第二个脉冲输入 表示单端输入
  _nop_();//
  ADC0832CLK = 1;//ADC0832时钟变量置高电平
  _nop_();//
  ADC0832CLK = 0;//ADC0832时钟变量置低电平
  ADC0832DI = 0;//ADC0832数据输入变量置低电平 第三个脉冲输入 表示单端输入通道0
  _nop_();//
  ADC0832CLK = 1;//ADC0832时钟变量置高电平
  _nop_();//
  ADC0832CLK = 0;//ADC0832时钟变量置低电平
  ADC0832DO = 1;//ADC0832数据输出变量置高电平 准备接收数据
  for(i = 0;i < 8;i++)//高位先移模拟数据变量的八位二进制数移位循环取输出模拟数据变量的八位二进制数
 {
   ADC0832CLK = 1;//ADC0832时钟变量置高电平
   _nop_();//
   ADC0832CLK = 0;//ADC0832时钟变量置低电平
   AnalogDataResult1 = AnalogDataResult1 << 1 ;//高位先移模拟数据变量的八位二进制数左移一位 高位先出
   if(ADC0832D0 == 1)//判断ADC0832数据输出变量是否为1
  {
    AnalogDataResult1 = AnalogDataResult1 | 0x01;//高位先移模拟数据变量的八位二进制数中取到二进制数1的运算
   }
  for(i = 0;i < 8;i++)//低位先移模拟数据变量的八位二进制数移位循环取输出模拟数据变量的八位二进制数
 {
   ADC0832CLK = 1;//ADC0832时钟变量置高电平
   _nop_();//
   ADC0832CLK = 0;//ADC0832时钟变量置低电平
   AnalogDataResult2 = AnalogDataResult2 >> 1;//低位先移模拟数据变量的八位二进制数右移一位 低位先出
   if(ADC0832D0 == 1)//判断ADC0832数据输出变量是否为1
  {
    AnalogDataResult2 = AnalogDataResult2 | 0x80;//低位先移模拟数据变量的八位二进制数中取到二进制数1的运算
    ADC0832CLK = 1;//ADC0832时钟变量置高电平
    _nop_();//
    ADC0832CLK = 0;//ADC0832时钟变量置低电平
   } 
  }
  ADC0832CLK = 1;//ADC0832时钟变量置高电平
  ADC0832CS = 1;//ADC0832片选变量置高电平
  ADC0832DI = 1;//ADC0832数据输入变量置高电平
  return (AnalogDataResult1 == AnalogDataResult2) ? AnalogDataResult1 : 0;//判断低位先移模拟数据变量包含的数据是否等于高位先移模拟数据变量包含的数据 若相等 返回低位先移模拟数据变量或高位先移模拟数据变量包含的数据 否则 返回0
 }
 /****
 void KeyScan()//按键扫描函数 该函数放在定时器定时1ms的中断函数中扫描
{
  if(AddKey)//如果增加按键没按下或弹起
 {
   AddKeyLockFlag = 0;//增加按键锁定标志位清0
   AddKeyPressDelayTime = 0;//增加按键按下延时时间清0
  }   
  else if(!AddKeyLockFlag)//如果增加按键锁定标志位置1 即增加按键按下
 {
   AddKeyPressDelayTime++;//增加按键按下延时时间自加
   if(AddKeyPressDelayTime > KeyPressDeshakeTime)//如果增加按键按下延时时间大于按键按下消抖时间
  {
    AddKeyPressDelayTime = 0;//增加按键按下延时时间清0
    KeyNumber = 1;//按键键值置1 此处是单击增加 可赋给swicth()语句中的变量来对数值单击增加
    AddKeyLockFlag = 1;//增加按键锁定标志位置1
   }
  }
  else if(AddKeyPressDelayTime < KeyLongPressDelayTime)//如果增加按键按下延时时间小于按键长按延时时间
 {
   AddKeyPressDelayTime++;//增加按键按下延时时间自加
  }
  else//如果增加按键按下延时时间大于按键长按延时时间
 {
   AddKeyLongPressAddIntervalTime++;//增加按键长按连增间隔时间自加
   if(AddKeyLongPressAddIntervalTime > KeyLongPressIntervalChangeTime)//如果增加按键长按连增间隔时间大于按键长按间隔变化时间
  {
    AddKeyLongPressAddIntervalTime = 0;//增加按键长按连增间隔时间清0
    KeyNumber = 1;//按键键值置1 此处是连击增加 可赋给swicth()语句中的变量来对数值连击增加
   }
  }   
  if(DecKey)//如果减少按键没按下或弹起
 {
   DecKeyLockFlag = 0;//减少按键锁定标志位清0
   DecKeyPressDelayTime = 0;//减少按键按下延时时间清0
  }   
  else if(!DecKeyLockFlag)//如果减少按键锁定标志位置1 即减少按键按下
 {
   DecKeyPressDelayTime++;//减少按键按下延时时间自加
   if(DecKeyPressDelayTime > KeyPressDeshakeTime)//如果减少按键按下延时时间大于按键按下消抖时间
  {
    DecKeyPressDelayTime = 0;//减少按键按下延时时间清0
    KeyNumber = 2;//按键键值置2 此处是单击减少 可赋给swicth()语句中的变量来对数值单击减少
    DecKeyLockFlag = 1;//减少按键锁定标志位置1
   }
  }
  else if(DecKeyPressDelayTime < KeyLongPressDelayTime)//如果减少按键按下延时时间小于按键长按延时时间
 {
   DecKeyPressDelayTime++;//减少按键按下延时时间自加
  }
  else//如果减少按键按下延时时间大于按键长按延时时间
 {
   DecKeyLongPressDecIntervalTime++;//减少按键长按连减间隔时间自加
   if(DecKeyLongPressDecIntervalTime > KeyLongPressIntervalChangeTime)//如果减少按键长按连减间隔时间大于按键长按间隔变化时间
  {
    DecKeyLongPressDecIntervalTime = 0;//减少按键长按连减间隔时间清0
    KeyNumber = 2;//按键键值置2 此处是连击减少 可赋给swicth()语句中的变量来对数值连击减少
    }
   }
 }
 ****/
 /****
 void NumberValueSet()//数字量数值设置函数
{
  switch(KeyNumber)//按键类型筛选位
 {
   case 1 ://增加按键单击、长按触发位
           NumberValue++;//数字量数值自加
           if(NumberValue > 255)//如果数字量数值大于255 为啥数字量数值变量NumberValue取255来比较?由于数字量数值变量NumberValue要计入DAC0832转换器 而DAC0832转换器是八位寄存器 最大只能计入255 因此数字量数值变量NumberValue取255来比较
          {
            NumberValue = 255;//数字量数值等于255
           }
           KeyNumber = 0;//按键键值清0
           break;//跳出
   case 2 ://减少按键单击、长按触发位
           NumberValue--;//数字量数值自减
           if(NumberValue < 0)//如果数字量数值小于0
          {
            NumberValue = 0;//数字量数值清0
           }
           KeyNumber = 0;//按键键值清0
           break;//跳出
   default:break;//跳出
  }
 }
 ****/
 void NixieTubeDisplayDataSplit()//数码管显示数据分解函数
{
  uchar NixieTubeQianWei,NixieTubeBaiWei,NixieTubeShiWei,NixieTubeGewei;//声明数码管千位、百位、十位、个位变量
  NixieTubeQianWei = AnalogFilterOutPutVoltage / 1000 ;//数码管千位分解
  NixieTubeBaiWei = AnalogFilterOutPutVoltage / 100 % 10;//数码管百位分解
  NixieTubeShiWei = AnalogFilterOutPutVoltage / 10 % 10 ;//数码管十位分解
  NixieTubeGeWei = AnalogFilterOutPutVoltage % 10 ;//数码管个位分解
  NixieTubeCacheDataArray[0] = NixieTubeQianWei;//数码管千位显示数据
  NixieTubeCacheDataArray[1] = NixieTubeBaiWei +15;//数码管百位带小数点显示数据
  NixieTubeCacheDataArray[2] = NixieTubeShiWei;//数码管十位显示数据
  NixieTubeCacheDataArray[3] = NixieTubeGeWei;//数码管个位显示数据
 }
 void NixieTubeDisplayData()//数码管显示数据函数  
{  
  static uchar i = 0;//定义静态数码管位变化变量
  switch(i)//数码管位变化筛选
 {
   case 0 ://数码管千位显示
           NixieTubeSegmentCode = 0x00;//数码管段码消影
           NixieTubeSegmentCode = NixieTubeDisplayDataArray[NixieTubeCacheDataArray[0]];//数码管千位的段码显示
           NixieTubeBitCode = NixieTubeBitCodeArray[0];//数码管千位码显示
           i++;//数码管位变化自加1
           break;//跳出
   case 1 ://数码管百位显示
           NixieTubeSegmentCode = 0x00;//数码管段码消影
           NixieTubeSegmentCode = NixieTubeDisplayDataArray[NixieTubeCacheDataArray[1]];//数码管百位的段码显示
           NixieTubeBitCode = NixieTubeBitCodeArray[1];//数码管百位码显示
           i++;//数码管位变化自加1
           break;//跳出 
   case 2 ://数码管十位显示
           NixieTubeSegmentCode = 0x00;//数码管段码消影
           NixieTubeSegmentCode = NixieTubeDisplayDataArray[NixieTubeCacheDataArray[2]];//数码管十位的段码显示
           NixieTubeBitCode = NixieTubeBitCodeArray[2];//数码管十位码显示
           i++;//数码管位变化自加1
           break;//跳出
   case 3 ://数码管个位显示
           NixieTubeSegmentCode = 0x00;//数码管段码消影
           NixieTubeSegmentCode = NixieTubeDisplayDataArray[NixieTubeCacheDataArray[3]];//数码管个位的段码显示
           NixieTubeBitCode = NixieTubeBitCodeArray[3];//数码管个位码显示
           i = 0;//数码管位变化清0
           break;//跳出
   default:break;//跳出
  }
 }  
/*****关于8051系列单片机定时器溢出率和定时器初值(定时计数初值)之间计算的知识点*****/ 
/****
一、定时器溢出率计算公式
1、定时器溢出率:定时器每秒溢出的次数
2、定时器溢出率计算公式表
    定时方式            分频方式                      公式
方式1:16位定时器  12分频(即12T 默认值)   Ft=晶振频率/12/(65536-定时器初值)
方式2:8位定时器   12分频(即12T 默认值)   Ft=晶振频率/12/(256-定时器初值)
方式1:16位定时器      1分频(即1T)       Ft=晶振频率/1/(65536-定时器初值)
方式2:8位定时器       1分频(即1T)       Ft=晶振频率/1/(256-定时器初值)
二、定时器初值(定时计数初值)计算公式
    定时方式            分频方式                      公式
方式1:16位定时器   12分频(即12T 默认值)  定时器初值(定时计数)=65536-晶振频率/12*定时时间
方式2:8位定时器    12分频(即12T 默认值)  定时器初值(定时计数)=256-晶振频率/12*定时时间
方式1:16位定时器       1分频(即1T)      定时器初值(定时计数)=65536-晶振频率*定时时间
方式2:8位定时器        1分频(即1T)      定时器初值(定时计数)=256-晶振频率*定时时间
****/
 void Timer0Init()//定时器0的16位定时模式1用12分频定时1ms初始化函数 晶振为12MHz
{
  AUXR &= 0x7f;//设定定时器/计数器模式为12T
  TMOD &= 0xf0;//设定定时器/计数器工作模式清0
  TMOD |= 0x01;//设定定时器/计数器为定时器 工作模式为16位定时器0模式1
  TH0 = 0xfc;//设定定时器0高8位初值 
  TL0 = 0x18;//设定定时器0低8位初值
  TF0 = 0;//定时器0溢出中断标志位清0
  ET0 = 1;//打开定时器中断开关
  EA = 1;//打开定时器中断总开关
  TR0 = 1//打开定时器0开关
 } 
 void Timer0() interrupt 1//定时器0的16位定时模式1用12分频定时1ms中断函数 晶振为12MHz
{
  TR0 = 0;//关定时器0开关
  /****
  Timer0TimeCount++;//定时器0定时计数自加
  if(Timer0TimeCount >= 10)//10ms时间到
 {
   Timer0TimeCount = 0;//定时器0定时计数清0
   DAC0832Change(NumberValue);//DAC0832转化函数 
  }
  ****/
  //KeyScan();//按键扫描函数
  NixieTubeDisplayData();//数码管显示数据函数  
  TH0 = 0xfc;//设定定时器0高8位初值
  TL0 = 0x18;//设定定时器0低8位初值
  TR0 = 1;//开定时器0开关
 } 
 void main()//主函数
{
  uchar AnalogDataResult;//声明模拟数字结果变量
  uchar AnalogSamplingCount;//声明模拟采样计数变量
  uint AnalogFilterVoltage;//声明模拟滤波电压变量
  Timer0Init();//定时器0的16位定时模式1用12分频定时1ms初始化函数 晶振为12MHz
  ADC0832Init();//ADC0832初始化函数
  //NumberValueSet();//数字量数值设置函数
  while(1)//主循环
 {
   AnalogDataResult = ADC0832Change();//ADC0832转化函数转化的模拟数据赋给模拟数据变量 
   OutPutVoltage = AnalogDataResult*1.0*500/255;//输出电压计算公式 500是基准电压5V转化成500mv 255是模数芯片ADC0832内部八位模拟转换寄存器储存的最大数值
   AnalogFilterVoltage = AnalogFilterVoltage + OutPutVoltage;//模拟滤波电压变量
   AnalogSamplingCount++;//模拟采样计数变量自加1
   if(AnalogSamplingCount >= 8)//模拟采样计数变量计8次
  {
    AnalogFilterOutPutVoltage = AnalogFilterVoltage >> 3;//模拟滤波电压变量右移三位 表示模拟滤波电压变量除以8取平均滤波后的输出电压
    AnalogSamplingCount = 0;//模拟采样计数变量清0
    AnalogFilterVoltage = 0;//模拟滤波电压变量清0  
   }
   NixieTubeDisplayDataSplit()//数码管显示数据分解函数
  }
 } 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1220935.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Linux】重定向|重新理解Linux下一切皆文件

文章目录 一、什么是重定向输出重定向的原理认识一下输出重定向的系统调用输出重定向的另外写法 二、浅谈输入重定向三、重定向和进程替换有冲突吗四、Linux下一切皆文件总结 一、什么是重定向 理解重定向之前&#xff1a;先理解一个叫做文件描述符的具体操作。 文件描述符&a…

C语言求0—7所能组成的奇数个数

完整代码&#xff1a; // 求0—7所能组成的奇数个数 //根据题意&#xff0c;应该是没有重复数字的&#xff0c;所以最大只能为八位数 //如果可以重复的话&#xff0c;那么位数就限制不了&#xff0c;然后奇数的个数就是无穷大了 #include <stdio.h>int main() {int coun…

万宾科技智能井盖传感器,提升市政井盖健康

市政井盖就是城市里不可或缺的基础设施之一&#xff0c;关于它的监测工作可马虎不得。它承载着保护市民的交通安全以及城市正常运转的重要使命。虽然现在城市化的速度很快&#xff0c;但是传统的市政井盖管理方式变得有些力不从心了。井盖的覆盖范围很广&#xff0c;如果单单依…

开启数据库审计 db,extended级别或os级别)并将审计文件存放到/opt/oracle/audit/下

文章目录 1、登录到数据库2、查看审计状态3、创建审计目录4、启用审计5、设置审计文件路径5、再次查看结果 1、登录到数据库 使用SQL*Plus或者其他Oracle数据库客户端登录到数据库。 sqlplus / as sysdba;2、查看审计状态 show parameter audit;目前是DB状态&#xff0c;并且…

【超好用的工具库】hutool-all工具库的基本使用

简介&#xff08;可不看&#xff09;&#xff1a; hutool-all是一个Java工具库&#xff0c;提供了许多实用的工具类和方法&#xff0c;用于简化Java开发过程中的常见任务。它包含了各种模块&#xff0c;涵盖了字符串操作、日期时间处理、加密解密、文件操作、网络通信、图片处…

计算机毕业设计 基于SpringBoot的医院档案管理系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…

自动驾驶-BEV感知综述

BEV感知综述 随着自动驾驶传感器配置多模态化、多源化&#xff0c;将多源信息在unified View下表达变得更加关键。BEV视角下构建的local map对于多源信息融合及理解更加直观简洁&#xff0c;同时对于后续规划控制模块任务的开展也更为方便。BEV感知的核心问题是&#xff1a; …

【proverif】proverif的语法3-认证协议的验证代码-案例分析

proverif-系列文章目录 【proverif】proverif的下载安装和初使用【proverif】proverif的语法1-解决中间人攻击-代码详解【proverif】proverif的语法2-各种密码原语的编码【proverif】proverif的语法3-认证协议的验证代码-案例分析 (本文) 文章目录 proverif-系列文章目录前言一…

51单片机应用从零开始(六)·逻辑运算

51单片机应用从零开始&#xff08;一&#xff09;-CSDN博客 51单片机应用从零开始&#xff08;二&#xff09;-CSDN博客 51单片机应用从零开始&#xff08;三&#xff09;-CSDN博客 51单片机应用从零开始&#xff08;四&#xff09;-CSDN博客 51单片机应用从零开始&#xff08;…

【入门篇】1.1 redis 基础数据类型详解和示例

文章目录 1. 简介2. Redis基础数据类型2.1 String类型场景示例常用命令示例 2.2 List类型场景示例 2.3 Set类型场景示例 2.4 Hash类型场景示例 2.5 Sorted Set类型 3. 使用Redis存储数据的注意事项1. 内存管理2. 数据持久化3. 高并发下的性能考量 4. 参考资料 1. 简介 Redis概…

springMvc中的拦截器【巩固】

先实现下想要的拦截器功能 package com.hmdp.utils;import com.hmdp.entity.User; import org.springframework.web.servlet.HandlerInterceptor;import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import javax.servlet.http.Ht…

【SpringMvc】SpringMvc +MyBatis整理

&#x1f384;欢迎来到边境矢梦的csdn博文&#x1f384; &#x1f384;本文主要梳理 Java 框架 中 SpringMVC的知识点和值得注意的地方 &#x1f384; &#x1f308;我是边境矢梦&#xff0c;一个正在为秋招和算法竞赛做准备的学生&#x1f308; &#x1f386;喜欢的朋友可以关…

python_主动调用其他类的成员

# 主动调用其他类的成员 # 方式一: class Base(object):def f1(self):print("5个功能") class Foo(object):def f1(self):print("3个功能")# Base.实例方法(自己传self),与继承无关Base.f1(self)obj Foo() obj.f1()print("#"*20)# 方式二:按照类…

Rockdb简介

背景 最近在使用flink的过程中&#xff0c;由于要存储的状态很大&#xff0c;所以使用到了rockdb作为flink的后端存储&#xff0c;本文就来简单看下rockdb的架构设计 Rockdb设计 Rockdb采用了LSM的结构&#xff0c;它和hbase很像&#xff0c;不过严格的说&#xff0c;基于LS…

希亦ACE和小吉内衣洗衣机选哪个?两款洗衣机对比

内衣洗衣机可以称得上是实现幸福的小家电&#xff0c;它不仅懒人的福音还是我们打工人的福音&#xff0c;在每天下班之后可以有时间休息了&#xff0c;洗完澡还有要手洗内衣裤&#xff0c;真的很痛苦&#xff0c;拥有了内衣洗衣机简直是一件非常幸福的事情&#xff0c;但现在市…

wpf devexpress Property Grid创建属性定义

WPF Property Grid控件使用属性定义定义如何做和显示 本教程示范如何绑定WP Property Grid控件到数据和创建属性定义。 执行如下步骤 第一步-创建属性定义 添加PropertyGridControl组件到项目。 打开工具箱在vs&#xff0c;定位到DX.23.1: Data 面板&#xff0c;选择Prope…

HarmonyOS 实现底部导航栏

该功能实现需要Tabs、TabsController、TabContent、Column等组件 Tabs相当于Android中的BottomNavigationView TabContent相当于Android中的fragment TabBuilder内相当于每个Item Entry Component struct Main {public tabsController : object new TabsController()State c…

后端接口性能优化分析-多线程优化

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱吃芝士的土豆倪&#xff0c;24届校招生Java选手&#xff0c;很高兴认识大家&#x1f4d5;系列专栏&#xff1a;Spring源码、JUC源码&#x1f525;如果感觉博主的文章还不错的话&#xff0c;请&#x1f44d;三连支持&…

微软 Gradle 强强联手,Gradle 构建服务器正式开源!

作者&#xff1a;Nick Zhu - Senior Program Manager, Developer Division At Microsoft 排版&#xff1a;Alan Wang Gradle 构建服务器 (Build Server for Gradle) 在九月份&#xff0c;我们宣布 Microsoft 和 Gradle 联手探索了一种基于 Build Server Protocol&#xff08;B…

用户运营:如何搭建用户分析体系

在运营的工作范畴中&#xff0c;用户运营是很重要的一个环节&#xff0c;甚至有公司会设置专门的“用户运营”岗位。 用户运营的价值体现在多个方面&#xff0c;不仅可以帮助引流、吸引更多用户使用产品&#xff0c;在用户正式使用产品之后的运营则更为重要。通过日常用户运营&…