语音识别芯片LD3320介绍再续

news2025/1/19 11:15:09

语音识别芯片LD3320驱动程序

    1、芯片复位

    复位就是对LD3320芯片的第47腿(RSTB*)发送低电平,然后需要对片选CS做一次拉低→拉  高的操作,以激活内部DSP。按照以下顺序:

    void LD_reset()

    {

     RSTB=1;delay(1);RSTB=0;delay(1);RSTB=1;delay(1);

     CSB=0;delay(1);CSB=1;delay(1);

   }

    delay(1)是为了更稳定地工作。 初始化一般在程序的开始进行,如果有时芯片的反应不太正常,也可用这个方法恢复芯片初始状态。

    (1)通用初始化

    按照以下序列设置寄存器。

    void LD_Init_Common()

    {

     bMp3Play=0;

     LD_ReadReg(0x06);

     LD_WriteReg(0x17,0x35);delay(10);

     LD_ReadReg(0x06);

     LD_WriteReg(0x89,0x03);delay(5);

     LD_WriteReg(0xCF,0x43);delay(5);

     LD_WriteReg(0xCB,0x02);

     /*PLL setting*/

     LD_WriteReg(0x11,LD_PLL_11);

     if(nLD_Mode==LD_MODE_MP3)

     {

      LD_WriteReg(0x1E,0x00);

      //!!注意,下面三个寄存器,会随晶振频率变化而设置不同

      //!!注意,请根据使用的晶振频率修改参考程序中的CLK_IN

      LD_WriteReg(0x19,LD_PLL_MP3_19);

      LD_WriteReg(0x1B,LD_PLL_MP3_1B);

      LD_WriteReg(0x1D,LD_PLL_MP3_1D);

     }

     else

     {

      LD_WriteReg(0x1E,0x00);

      //!!注意,下面三个寄存器,会随晶振频率变化而设置不同

      //!!注意,请根据使用的晶振频率修改参考程序中的CLK_IN

      LD_WriteReg(0x19,LD_PLL_ASR_19);

      LD_WriteReg(0x1B,LD_PLL_ASR_1B);

      LD_WriteReg(0x1D,LD_PLL_ASR_1D);

    }

    LD_WriteReg(0xCD,0x04);LD_WriteReg(0x17,0x4c);delay(5);

    LD_WriteReg(0xB9,0x00);LD_WriteReg(0xCF,0x4f);

    LD_WriteReg(0x6F,0xFF);

  }

   (2)语音识别用初始化

    按照以下序列设置寄存器。

    void LD_Init_ASR()

     {

      nLD_Mode=LD_MODE_ASR_RUN;

      LD_Init_Common();

      LD_WriteReg(0xBD,0x00);LD_WriteReg(0x17,0x48);delay(10);

      LD_WriteReg(0x3C,0x80);LD_WriteReg(0x3E,0x07);

      LD_WriteReg(0x38,0xff);LD_WriteReg(0x3A,0x07);

      LD_WriteReg(0x40,0x00);LD_WriteReg(0x42,0x08);

      LD_WriteReg(0x44,0x00);LD_WriteReg(0x46,0x08);delay(1);

     }

    (3)写入识别列表

    列表的规则是:每个识别条目对应一个特定的编号(1个字节),不同的识别条目的编号可以相同,而且不用连续。本芯片最多支持50个识别条目,每个识别条目是标准普通话的汉语拼音(小写),每2个字(汉语拼音)之间用一个空格间隔。下面是一个简单的例子:

     编号可以相同,可以不连续,但是数值要小于256(00H~FFH)。例子中的“北京”和“首都”对应同一编号,说这两个词会有相同的结果返回。简单流程图如下:

 

    代码如下:

    #define CODE_DEFAULT 0

    #define CODE_BEIJING 1

    #define CODE_SHANGHAI 3

    #define CODE_TIANJIN 7

    #define CODE_CHONGQING 8

    先介绍一个读取0xB2寄存器的函数,如果在以后的ASR命令函数前不能够读取到正确Idle状态,说明芯片内部可能出错了。当使用的电源电压/电流出现不稳定有较大波动时,有小概率会出现这种情况。出现这种情况时,建议Reset LD3320芯片,重新启动设置芯片。

    // Return 1: success.

    uint8 LD_Check_ASRBusyFlag_b2()

    {

     uint8 j;uint8 flag=0;

     for(j=0;j<10;j++)

     {

      if(LD_ReadReg(0xb2)= 0x21)

      {flag=1;break;}

      delay(10);

     }

     return flag;

   }

    // Return 1: success.

    uint8 LD_AsrAddFixed()

    {

     uint8 k, flag;

     uint8 nAsrAddLength;

     const char sRecog[5][13]={"bei jing","shou du","shang hai","tian jin", "chong qing"};

     const uint8 pCode[5]={CODE_BEIJING,CODE_BEIJING,CODE_SHANGHAI,                                                                   CODE_TIANJIN,CODE_CHONGQING};

     flag=1;

     for(k=0;k<5;k++)

     {

      if(LD_Check_ASRBusyFlag_b2()==0)

      {flag=0;break;}

      LD_WriteReg(0xc1, pCode[k] );

      LD_WriteReg(0xc3, 0 );

      LD_WriteReg(0x08, 0x04); Delay(1);

      LD_WriteReg(0x08, 0x00); Delay(1);

      for(nAsrAddLength=0;nAsrAddLength<20;nAsrAddLength++)

      {

       if(sRecog[k][nAsrAddLength]==0) break;

       LD_WriteReg(0x5,sRecog[k][nAsrAddLength]);

      }

      LD_WriteReg(0xb9, nAsrAddLength);

      LD_WriteReg(0xb2, 0xff);

      LD_WriteReg(0x37, 0x04);

    }

    return flag;

   }

(4)开始识别

    设置几个相关的寄存器,就可以控制LD3320芯片开始语音识别。 值得注意:单片机程序中,一般会用一个全局变量控制当前状态,(例如:LD_ASR_RUNING状态或者LD_ASR_FOUNDOK状态),在编程时一定要把对该状态的设置放在正式LD3320芯片开始识别以前,例如下面例程中的语句nAsrStatus=LD_ASR_RUNING;便是设置控制变量。需要把这句语句放置在LD3320正式开始识别之前调用:

    {

     。。。

     nAsrStatus=LD_ASR_RUNING;LD_AsrRun();

    }

    参考代码如下:

    // Return 1: success.

    uint8 LD_AsrRun()

    {

     nAsrStatus=LD_ASR_RUNING;

     LD_WriteReg(0x35,MIC_VOL);

     LD_WriteReg(0x1C,0x09);

     LD_WriteReg(0xBD,0x20);

     LD_WriteReg(0x08,0x01);delay(1);

     LD_WriteReg(0x08,0x00);delay(1);

     if(LD_Check_ASRBusyFlag_b2()==0)

     {return 0;}

     LD_WriteReg(0xB2,0xff);

     LD_WriteReg(0x37,0x06);delay(5);

     LD_WriteReg(0x1C,0x0b);

     LD_WriteReg(0x29,0x10);

     LD_WriteReg(0xBD,0x00);

     EX0=1;

     return 1;

    }

    综上所述:语音识别的流程可以总结成如下的参考代码,可以以此为参考,根据使用流程进行合理的改动。

    uint8 RunASR()

    {

     uint8 i=0;uint8 asrflag=0;

     for(i=0;i<5;i++)

     {

      LD_Init_ASR();delay(100);

      if(LD_AsrAddFixed()==0)

      {LD_reset();delay(100);continue;}

      delay(10);

      if(LD_AsrRun()==0)

      {LD_reset();delay(100);continue;}

      asrflag=1;

      break;

     }

     return asrflag;

   }

    对上述函数的调用参考代码:

    文件main.c 函数main()

    case LD_ASR_NONE:

    {

     nAsrStatus=LD_ASR_RUNING;

     if (RunASR()==0)

     {nAsrStatus=LD_ASR_ERROR;LED1=0;LED2=0;}

     break;

    }

    (5)响应中断

    如果麦克风采集到声音,不管是否识别出正常结果,都会产生一个中断信号。而中断程序要根据寄存器的值分析结果。

    读取BA寄存器的值,可以知道有几个候选答案,而C5寄存器里的答案是得分最高、最可能正确的答案。

    例如发音为“上海”并被成功识别(无其他候选),那么BA寄存器里的数值是1,而C5寄存器里的值是对应的编码3。

    以下为简单流程图:

    程序代码如下:

    void ExtInt0Handler(void) interrupt 0 {ProcessInt0();}

    void ProcessInt0()

    {

     uint8 nAsrResCount=0;EX0=0;ET0=0;

     ucRegVal=LD_ReadReg(0x2B);

     ucHighInt=LD_ReadReg(0x29);

     LD_WriteReg(0x29,0);

     ucLowInt=LD_ReadReg(0x02);

     LD_WriteReg(0x02,0);

     if(nLD_Mode==LD_MODE_ASR_RUN)

     {

      if((ucRegVal&0x10)&&LD_ReadReg(0xbf)==0x35&&LD_ReadReg(0xb2)==0x21)

      {

       nAsrResCount=LD_ReadReg(0xba);

       if(nAsrResCount>0&&nAsrResCount<4) {nAsrStatus=LD_ASR_FOUNDOK;}

       else{nAsrStatus=LD_ASR_FOUNDZERO;}

      }

      else{nAsrStatus=LD_ASR_FOUNDZERO;}

      LD_WriteReg(0x2b,0);

      LD_WriteReg(0x1C,0);

      ET0=1;

      return;

     }

   }

    uint8 LD_GetResult() {return LD_ReadReg(0xc5);}

    值得注意:获取识别结果 LD_ReadReg(0xba); 多少条候选识别结果,值1~4说明是有正确的识别结果。

    4个候选结果的读取:根据0xba决定读取几个识别结果。

    LD_ReadReg(0xc5);

    LD_ReadReg(0xc7);

    LD_ReadReg(0xc9);

    LD_ReadReg(0xcb);

    在目前的Demo程序中,只读取了最优候选。在其他使用场合,如果需要读取其他候选,用户可以自己编程实现。

    3、声音播放

    播放声音的操作顺序是:

    MP3播放用初始化(包括通用初始化)→调节播放音量→开始播放声音, 并准备好中断响应函数,打开中断允许位。

    (1)通用初始化

     和语音识别部分一样,按指定序列设置寄存器。

    (2)声音播放用初始化 请参照源代码设置寄存器。

    void LD_Init_MP3()

    {

     nLD_Mode = LD_MODE_MP3;

     LD_Init_Common();

     LD_WriteReg(0xBD,0x02);

     LD_WriteReg(0x17, 0x48); delay(10);

     LD_WriteReg(0x85,0x52);

     LD_WriteReg(0x8F,0x00);

     LD_WriteReg(0x81,0x00);

     LD_WriteReg(0x83,0x00);

     LD_WriteReg(0x8E,0xff);

     LD_WriteReg(0x8D,0xff); delay(1);

     LD_WriteReg(0x87,0xff);

     LD_WriteReg(0x89,0xff); delay(1);

     LD_WriteReg(0x22,0x00);

     LD_WriteReg(0x23,0x00);

     LD_WriteReg(0x20,0xef);

     LD_WriteReg(0x21,0x07);

     LD_WriteReg(0x24, 0x77);

     LD_WriteReg(0x25,0x03);

     LD_WriteReg(0x26, 0xbb);

     LD_WriteReg(0x27,0x01);

    }

    (3)调节播放音量

    这里需要修改寄存器8E。 音量分为16级,用4位二进制表示,范围是0-15。 设置音量时,将(15-音量值) 设给寄存器8E的第2-5位。

    源代码如下:

    void LD_AdjustMIX2SPVolume(uint8 val)

    {

     uint8 ucTmp;

     ucSPVol=val;

     val=((15-val)&0x0f)<<2;

     LD_WriteReg(0x8E,val|0xc3);

    }

    这个函数只调节喇叭(Speaker)的音量,和耳机等其他输出无关。而且实验板上只有喇叭连接了输出。

    耳机音量是可以调节的,使用寄存器81(左耳音量)和83(右耳音量)。 2个寄存器都是只有第1-5位有效,共32级,而且这5位都为0(00000B)的时候是音量最大的,都为1(11111B)的时候音量最小。

    (4)开始播放声音

    开始播放位置清零(自定义变量Mp3Pos=0)

    寄存器1B的第3位设为1(按位或0x08)

        循环执行:

        while(【播放条件】=true)

        {

         顺序将MP3数据放入寄存器01(每次一个字节);

         Mp3Pos增加1

        }

  【播放条件】为下面条件都成立,有一个不满足就跳出循环:

        读取寄存器06,第3位=0

        Mp3Pos < MP3文件的总长度。

   修改BA 、17等寄存器。(参照源代码)

   开放中断允许。例如,EX0=1。

   开始播放的简单流程图如下:

 

    源代码如下:

    void LD_play()

    {

     nMp3Pos=0;bMp3Play=1;

     if(nMp3Pos>=nMp3Size)return ;

     LD_ReloadMp3Data();

     LD_WriteReg(0xBA,0x00);

     LD_WriteReg(0x17,0x48);

     LD_WriteReg(0x33,0x01);

     LD_WriteReg(0x29,0x04);

     LD_WriteReg(0x02,0x01);

     LD_WriteReg(0x85,0x5A);

     EX0=1;

    }

    void LD_ReloadMp3Data()

    {

     uint32 nCurMp3Pos;uint8 val;uint8 k;

     nCurMp3Pos=nMp3StartPos+nMp3Pos;

     FLASH_CS=1;FLASH_CLK=0;FLASH_CS=0;

     IO_Send_Byte(W25P_FastReadData);                   /* read command */

     IO_Send_Byte(((nCurMp3Pos&0xFFFFFF)>>16)); /* send 3 address bytes */

     IO_Send_Byte(((nCurMp3Pos&0xFFFF)>>8));

     IO_Send_Byte(nCurMp3Pos&0xFF);

     IO_Send_Byte(0xFF);

     ucStatus=LD_ReadReg(0x06);

     while(!(ucStatus&MASK_FIFO_STATUS_AFULL)&&(nMp3Pos<nMp3Size))

     {

      val=0;

      for(k=0;k<8;k++){FLASH_CLK=0;val<<=1;FLASH_CLK=1;val|=FLASH_DO;}     

      LD_WriteReg(0x01,val);nMp3Pos++;ucStatus=LD_ReadReg(0x06);

     }

    FLASH_CS=1;FLASH_CLK=0;

   }

    LD_ReloadMp3Data()函数的功能是送入数据,不同的硬件结构可能需要改写这一部分。例如有的系统可能使用大容量的RAM,取数据就会很方便。这里是根据串行FLASH存储器的接口写的函数,使用的是SPI协议。

     (5)中断响应。

    开始播放可以把声音数据的最初部分送入芯片,等到芯片播放这一段后会发出中断请求。而中断函数里会不断的送入数据,直到FIFO_DATA装满或声音数据结束。这一段程序和开始播放比较类似,都是通过LD_ReloadMp3Data()函数送入数据。

    由于LD3320芯片只有一只管脚负责中断请求输出,所以一般情况下用一个中断响应函数处理2种中断。这里为了简明,将中断函数分开书写。

    中断处理函数里,播放声音部分流程图如下:

    要注意的是,寄存器操作2)和寄存器操作3) 并不会恢复寄存器02和29的内容。因为这时播放已经结束了,应该让02,29清零。 而数据重载结束后(达到了继续播放的条件),需要恢复寄存器02,29。 以便于继续出中断,连续播放。 

    源代码是:

    void ExtInt0Handler(void) interrupt 0{ProcessInt0();}

    void ProcessInt0()

    {

     uint8 nAsrResCount=0;

     EX0=0;ET0=0;

     ucRegVal=LD_ReadReg(0x2B);

     ucHighInt=LD_ReadReg(0x29);

     ucLowInt=LD_ReadReg(0x02);

     LD_WriteReg(0x29,0);

     LD_WriteReg(0x02,0);

     if(nLD_Mode==LD_MODE_MP3)

     {

      if(LD_ReadReg(0xBA)&CAUSE_MP3_SONG_END)

      {

       LD_WriteReg(0x2B,0);

       LD_WriteReg(0xBA,0x00);

       LD_WriteReg(0xBC,0x00);

       bMp3Play=0;

       LD_WriteReg(0x08,0x01);

       LD_WriteReg(0x08,0x00);

       LD_WriteReg(0x33,0x00);

       return ;

     }

    if(nMp3Pos>=nMp3Size)

    {

     LD_WriteReg(0xBC,0x01);

     LD_WriteReg(0x29,0x10);

     EX0=1;

     return;

    }

    LD_ReloadMp3Data();

    LD_WriteReg(0x29,ucHighInt);

    LD_WriteReg(0x02,ucLowInt) ;

    EX0=1;

    }

  }

    补充说明

    1、此芯片的特色是兼有语音识别和MP3播放的两项功能,但是由于这两项功能会使用一些公用的资源,所以为了使芯片稳定地工作,在功能切换的时候,必须从最“通用初始化”开始,对芯片进行一系列的设置。

    2、当芯片长时间没有响应时,可能是应用程序的设置不合理或者是电源的电压/电流有比较大的波动造成。这时应使用芯片复位的功能(对芯片的RTSB*发送低电平),使芯片重新开始工作。

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

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

相关文章

TencentOS 3.1下安装zabbix 5.0.30

TencentOS是使用官方镜像文件安装的虚拟机。 虚拟机为virtualBox 6.1 zabbix 使用zabbix官方安装包编译安装。 下载地址: Download Zabbix sources zabbix软件包解包,本次安装解包在/opt下 zabbix需要nginx、php、mysql等软件支持,因此先安装它们。 安装mysql如下: yu…

Usaco Training刷怪旅 第二层第二题:Transformations

usaco training 关注我持续更新usaco training A square pattern of size N x N (1 < N < 10) black and white square tiles is transformed into another square pattern. Write a program that will recognize the minimum transformation that has been applied to t…

QCon直击|闲鱼推荐大规模应用背后的工程实践

讲师介绍闲鱼技术部 | 吴白万小勇&#xff08;吴白&#xff09;&#xff0c;闲鱼服务端专家。毕业于南京大学计算机系&#xff0c;目前负责闲鱼技术推荐架构。推荐在闲鱼的应用不同于搜索的确定性&#xff0c;推荐场景面临的问题往往是不确定的。但是正是因为这种不确定&#x…

Kafka无法对外暴露端口的相关解决方案

1、kafka开放外部访问 1.1、修改kafka configMap 修改内容如下 # 修改EXTERNAL_ACCESS_IP变量&#xff0c;值为真实的外网IP export EXTERNAL_ACCESS_IP${真实的外网IP} 1.2、重启kafka # 重启kafka kubectl rollout restart statefulset kafka -n default # 查看kafka状态 k…

盛邦安全将于12月27日上会,2022年前三季度收入1亿元

12月20日&#xff0c;上海证券交易所披露的科创板上市委2022年第112次审议会议公告显示&#xff0c;远江盛邦&#xff08;北京&#xff09;网络安全科技股份有限公司&#xff08;下称“盛邦安全”&#xff09;将于2022年12月27日上会。 目前&#xff0c;盛邦安全已经提交了招股…

在tushare量化平台可以获取哪些数据?

沪深股票数据是Tushare量化最传统、最具历史意义的数据服务项目从一开始就为投资者特别是定量投资者提供了稳定、方便的接口。 1.基础数据 提供交易和回测所需要的基础信息&#xff0c;目前主要提供的是上市公司股票列表和交易日历等 2.行情数据 行情数据目前还在继续规则和…

“史上最贵”卡塔尔世界杯,有哪些炫酷的“黑科技”?

2022年卡塔尔世界杯投入超过2200亿美元&#xff0c;堪称“史上最贵世界杯”。这些投入不仅用在了场馆、道路等基础设施建设中&#xff0c;也体现在让人震撼的芯片传感器、人工智能、先进技术等“黑科技”上。今年的卡塔尔世界杯有什么“科技感十足”的先进技术&#xff1f;让我…

坐标系相关整理

参考url:https://www.jianshu.com/p/06890af3d780 Dicom坐标系 x轴&#xff1a;从右手到左手 y轴&#xff1a;从前胸到后背 z轴&#xff1a;从脚到头 解刨学坐标系 医学人体三解剖面&#xff0c;医疗影像三维图&#xff0c;主要是针对人体来说。解剖学上的坐标体系&#xff0…

常见垃圾回收器

1.Serial垃圾回收器 Serial&#xff08;串行&#xff09;收集器是最基本、历史最悠久的垃圾收集器了&#xff0c;是一个单线程收集器 它只会使用一个垃圾收集线程去完成垃圾收集工作 它在进行垃圾收集工作的时候必须暂停其他所有的工作线程&#xff08; "Stop The Worl…

C++——STL之vector详解

C——STL之vector详解&#x1f3d0;1.什么是vector&#x1f3d0;2.vector的使用&#x1f3c0;2.1vector的实例化&#x1f3c0;2.2访问遍历vector⚽2.2.1**下标[]**⚽2.2.2**迭代器**⚽2.2.3**范围for**&#x1f3c0;2.3.vector容量问题⚽2.3.1size和capacity⚽2.3.2reserve和re…

二叉树先、中、后遍历递归+非递归

文章目录前言思路设计思想非递归前序遍历的思路非递归中序遍历的思路非递归后序遍历的思路层序遍历的思路完整代码MyBinaryTree.hMyBinaryTree.cppMain.cpp效果展示前言 作者水平有限&#xff0c;全部的代码是学习前人部分原创不要搬代码&#xff0c;一定要借鉴学习&#xff0…

接口测试的痛点和解决办法

在做接口测试时&#xff0c;以下几个测试痛点&#xff0c;一定要仔细琢磨下&#xff1a; 痛点①&#xff1a;由于测试环境数据被改动&#xff0c;导致接口测试失败 这个问题&#xff0c;最好的办法就是重新调用数据库中的最新数据。在做测试用例时&#xff0c;也要考虑到实时调…

汽车服务行业概况与供应商评估方案介绍,数商云SRM系统助力多维考核供应商

近年来&#xff0c;汽车保有量不断增长&#xff0c;中国汽车市场连续多年成为全球产销第一&#xff0c;伴随物联网技术的不断渗透&#xff0c;车联网开始兴起&#xff0c;带给线上线下汽车相关服务行业新的发展机遇。 当前汽车服务行业整体概况 1、市场需求增加&#xff1a;市…

Vue基础超详细

目录 一、Vue的简介 1、什么是vue 2、vue 的特性&#xff08;数据驱动视图、双向数据绑定&#xff09; 3、MVVM及其工作原理 二、Vue的基本实用 1、基本使用步骤 2、配置Vue的调试工具 3、指令与过滤器 3.1内容渲染指令 4、属性绑定指令 5、使用Javascript表达式 6、事件绑…

“ 念旧真的是一件很无趣的事 “

想要的都拥有 得不到的都释然 My Jinji音频&#xff1a;00:0006:40 | 01 | 想通就释然 想不通就茫然 每天不停的循环 | 02 | 终于理解你曾经说的 “有些人不能做朋友” 现在我和你一样惨 | 03 | 村上春树说过&#xff1a; “如果一直想见谁 肯定迟早会见到” 但是他还…

android入门之创建service

1. 前言 Service意为&#xff1a;服务&#xff0c;是一种运行时用户不可见的活动机制。可以理解为它是一个没有布局的Activity。 典型的场景&#xff1a;音乐后台播放、后台下载。 Service不同于子线程&#xff0c;Service是运行在主线程中的&#xff0c;因此不能进行耗时操作。…

电线电缆行业MES系统,帮助企业快速应对订单变化,减少资金占用

电线电缆行业生产管理现状 电线电缆行业是典型的重资产行业&#xff0c;原材料成本高、产品价值高、资金占用大、产品规格型号多达数万种&#xff0c;BOM管理繁杂&#xff0c;现场管理粗放&#xff0c;订单经常会合并或拆分生产&#xff0c;对排程要求高。 1、计划制定不准确…

无分类编址CIDR

无分类编址CIDR 构成超网 将多个子网聚合成一个较大的子网&#xff0c;叫做构成超网&#xff0c;或路由聚合。 方法&#xff1a;将网络前缀缩短&#xff08;所有网络地址取交集&#xff09;。 例题 某路由表中有转发接口相同的4条路由表项&#xff0c;其目的网络地址分别为…

Unicode编码的理解

1、Unicode 这个单词可以拆解为两部分&#xff0c;一个是Uni ,即英文单词unique的意思&#xff0c;也就是唯一的意思。code就是编码的意思。 GBK编码的理解_sgmcy的博客-CSDN博客 在上节博客里面&#xff0c;介绍了ASCII编码、ASCII编码表的扩展以及我们国家汉字的GBK编码。 …

详解OpenCV的椭圆曲线点坐标近似计算函数ellipse2Poly()

详解OpenCV的椭圆曲线点坐标近似计算函数ellipse2Poly() 函数ellipse2Poly()可用于近似计算椭圆曲线的像素坐标。 而前面介绍过的函数ellipse()则是直接在图像中绘制椭圆&#xff0c;详情见 https://www.hhai.cc/thread-174-1-1.html 函数ellipse2Poly()的C原型如下&#xff…