關(guān)于正交解碼,我先解釋何為正交解碼,,,,其實名字挺高大上的,,,,還是先說編碼器吧
看一下我用過的一種編碼器
編碼器的線數(shù) ,是說編碼器轉(zhuǎn)一圈輸出多少個脈沖,,,如果一個編碼器是500線,,,說明這個編碼器轉(zhuǎn)一圈對應(yīng)的信號線會輸出500個脈沖,,,,為什么說是對應(yīng)的,,,對于上面的編碼器而言是這樣的,,,,但是有的編碼器里面有好多線,,,有的信號線是轉(zhuǎn)一圈就輸出對應(yīng)多少線的脈沖,,,有的信號線是轉(zhuǎn)一圈就輸出一個脈沖,,,,這就要看編碼器的資料了......
看我上面用過的這個編碼器,,使用A,B,VCC,GND這四根線,,
編碼器轉(zhuǎn)一圈A信號線會輸出500個脈沖,,,,B信號線也會輸出500個脈沖,,不過
正轉(zhuǎn)的時候
反轉(zhuǎn)的時候
正轉(zhuǎn)的時候信號線A先輸出信號,B信號線后輸出
反轉(zhuǎn)的時候信號線B先輸出信號,A信號線后輸出
假設(shè)只是單純的測正轉(zhuǎn)脈沖或者單純的測反轉(zhuǎn)脈沖,那么只需要用單片機隨意選擇一個信號線就行了,,然后就是脈沖計數(shù),,,,
如果說要是一個電機既有正轉(zhuǎn)又有反轉(zhuǎn),,,,我想知道這個電機絕對正轉(zhuǎn)了多少圈
那么就需要用正轉(zhuǎn)的圈數(shù)減去反轉(zhuǎn)的圈數(shù)了,,,,,,那么問題來了,,,怎么測正轉(zhuǎn)圈數(shù)和反轉(zhuǎn)圈數(shù)
其實傳統(tǒng)的做法
關(guān)于D觸發(fā)器,,,,當clk引腳來一個上升沿,D引腳是什么電平,,那么Q就輸出什么電平,,Q非,,與Q相反
現(xiàn)在如果說電機正轉(zhuǎn)
可以看到每當B來上升沿的時候,A信號總為高電平,,,所以Q會輸出高電平
設(shè)置A為上升沿進入其,,,中斷函數(shù)
然后判斷一下Q是否為高電平,如果為高電平正轉(zhuǎn)變量自加一
如果電機反轉(zhuǎn)
可以看到每當B來上升沿的時候,A信號總為低電平,,,所以Q會輸出低電平
設(shè)置B為上升沿進入其,,,中斷函數(shù)
然后判斷一下Q是否為低電平,如果為低電平反轉(zhuǎn)變量自加一
然后正轉(zhuǎn)變量減去反轉(zhuǎn)變量就能得到電機到底正轉(zhuǎn)了多少圈......這樣就會得到一個相對的變量,,這個變量呢,就是單片機正交解碼的值
說白了單片機的正交解碼功能就是得到一個正反轉(zhuǎn),,,相對的變量,,,,這個變量呢,,你正轉(zhuǎn)的時候他會加,你反轉(zhuǎn)的時候他會減,
不過呢,,,,STM32的這個變量有點坑,,,是太坑了,,,他內(nèi)部定義的這個變量的類型是無符號整形,,,也就是說0---65535,,,,,,,,,,,,,
如果直接利用STM32的正交解碼功能,,直接讀這個值,,這樣呢就會出現(xiàn)很麻煩的問題,,,,
假設(shè)你正轉(zhuǎn)轉(zhuǎn)到10,然后反轉(zhuǎn)轉(zhuǎn)到0之后還是反轉(zhuǎn),,他就會開始從65535開始往下減了,,,,假設(shè)轉(zhuǎn)到了60000,,你又開始正轉(zhuǎn)了,一直越過65535之后,他又開始從0開始往上加了,,,,要是你的電機就是這樣的反反復(fù)復(fù),而且正轉(zhuǎn)與反轉(zhuǎn)次數(shù)每次都在變化,,,,,,,你如何寫程序來得到到底正轉(zhuǎn)了多少圈??????????所以說STM32真是太坑了,,,害的我和我同學(xué)寫算法測試算法耽擱了兩天,,,,,,,最后還是會出現(xiàn)BUG,,,,,,,,網(wǎng)上也有大神寫類似的算法來計算出到底轉(zhuǎn)的圈數(shù),,,,,,,,,,,,,,
最總我決定用最本質(zhì)的方法也就是利用D觸發(fā)器,,,,,,,不過因為手頭并沒有D觸發(fā)器,,,但是STM32正交解碼內(nèi)部實質(zhì)就是用的D觸發(fā)器來做判斷的,,,,,
而且他提供了一個寄存器來判斷是正轉(zhuǎn)還是反轉(zhuǎn)
if((TIM3->CR1& TIM_CounterMode_Down) == TIM_CounterMode_Down)//向下計數(shù)
else
就是向上計數(shù)......也就是能指明正反轉(zhuǎn),,,,,
所以我就打開他的正交解碼功能,,就是把A,B信號線既接正交解碼引腳上,又接在脈沖計數(shù)中斷引腳上
void EXTI0_IRQHandler(void)//反轉(zhuǎn)中斷
{
if(EXTI_GetITStatus(EXTI_Line0) != RESET) //???????EXTI Line??
{
EXTI_ClearITPendingBit(EXTI_Line0); //???????
if((TIM3->CR1 & TIM_CounterMode_Down) == TIM_CounterMode_Down)//???
{
cnt--;
}
}
}
/*************** ????1??,???? B1 *********************/
void EXTI1_IRQHandler(void)//正轉(zhuǎn)中斷
{
if(EXTI_GetITStatus(EXTI_Line1) != RESET) //???????EXTI Line??
{
EXTI_ClearITPendingBit(EXTI_Line1); //???????
if((TIM3->CR1 & TIM_CounterMode_Up) == TIM_CounterMode_Up)//???
{
if((TIM3->CR1 & TIM_CounterMode_Down) == TIM_CounterMode_Down);//???
else
cnt++;
}
}
}
這樣的話省了D觸發(fā)器了,,,,,,,
看了一眼K60的正交解碼的變量
雖然是32位了但是,,,,,,,是無符號的,,,,,,如果只用正交解碼功能,,反轉(zhuǎn)又要考慮算法了,,,,,,,,,,,,,,,,,,,,,