Drehimpulsgeber Ky-040: Unterschied zwischen den Versionen
| Zeile 115: | Zeile 115: | ||
<syntaxhighlight lang="c" style="background-color: #EFF1C1; font-size:larger"> | <syntaxhighlight lang="c" style="background-color: #EFF1C1; font-size:larger"> | ||
const | //--------------------------------------------------------------------------------------------------------- | ||
const | //--------------------------------------------------------------------------------------------------------- | ||
const | //Drehencoder KY040 auslesen (20 Rastungen 40 Positionen / Umdrehung) | ||
const int | // | ||
int CLK_pos_alt_s16; // letzte Position von CLK | // Anschlüsse: | ||
int CLK_pos_neu_s16; | |||
bool | // Drehencoder | Arduino | ||
int PosZaehler_s16 =0; // Zähler der Position | // ---------------------------------- | ||
float WinkelZaehler_f32 =0.0;// | // GND | GND | ||
//----------------------------------- | // + | +5V | ||
void | // SW | D4 | ||
// DT | D2 | |||
// CLK | D3 | |||
// | |||
// ausgabe der Position und der Drehrichtung und Winkel | |||
// | |||
// | |||
// | |||
// | |||
// | |||
// | |||
// https://wiki.hshl.de/wiki/index.php/Ky-040-Drehimpulsgeber#Pinbelegung | |||
PosZaehler_s16 | // | ||
//31.10.2025 Marc Ebmeyer | |||
// | |||
//--------------------------------------------------------------------------------------------------------- | |||
//--------------------------------------------------------------------------------------------------------- | |||
const byte SW_u8 = 4; // Pin zum Messen vom Schalter | |||
const byte DT_u8 = 2; // Pin zum Messen interrupt pin | |||
const byte CLK_u8 = 3; // Pin zum Messen des Taktes interrupt pin | |||
const unsigned int Pos_u16 = 40; // Anzahl an Positionen des Drehencoders | |||
volatile int CLK_pos_alt_s16; // letzte Position von CLK | |||
volatile int CLK_pos_neu_s16; | |||
volatile bool drehtimUhr_bit; // dreht der Drehencoder im Uhrzeigersinn? true =ja | |||
volatile int PosZaehler_s16 =0; // Zähler der Position | |||
volatile float WinkelZaehler_f32 =0.0;// winkelZähler initalisieren Startposition= o | |||
volatile bool drehtsich_bit =false; | |||
//------------------------------------------------------------------------------------------------------- | |||
void alarm_DT() | |||
{ | |||
CLK_pos_neu_s16 = digitalRead(CLK_u8); // auslesen der Positon von CLK und zwischen speichern in Hilfsvaraiable | |||
if (CLK_pos_neu_s16 != CLK_pos_alt_s16)// Drehencoder dreht sich | |||
{ | |||
if (digitalRead(DT_u8) != CLK_pos_neu_s16) // Drehrichtungserkennung | |||
{ // CLK ändert sich zuerst - | |||
PosZaehler_s16 ++; // hochzählen da im Uhrzeigersinn | |||
drehtimUhr_bit = true; //mit dem Uhrzeigersinn | |||
} | |||
else // DT ändert sich zuerst wodurch wir uns gegen dem Uhrzeigersinn drehen | |||
{ | |||
drehtimUhr_bit = false; // gegen dem Uhrzeigersinn | |||
PosZaehler_s16 --; // runterzählen | |||
} | |||
} | } | ||
CLK_pos_neu_s16 = digitalRead( | |||
drehtsich_bit =true; //AusgabeBit setzen das der Drehencoder sich dreht | |||
} | |||
//------------------------------------------------------------------------------------------------------- | |||
void alarm_CLK() | |||
{ | |||
CLK_pos_neu_s16 = digitalRead(CLK_u8); // auslesen der Positon von CLK und zwischen speichern in Hilfsvaraiable | |||
if (CLK_pos_neu_s16 != CLK_pos_alt_s16)// Drehencoder dreht sich | if (CLK_pos_neu_s16 != CLK_pos_alt_s16)// Drehencoder dreht sich | ||
{ | { | ||
if (digitalRead( | if (digitalRead(DT_u8) != CLK_pos_neu_s16) // Drehrichtungserkennung | ||
{ // CLK ändert sich zuerst - | { // CLK ändert sich zuerst - | ||
PosZaehler_s16 ++; // hochzählen da im Uhrzeigersinn | PosZaehler_s16 ++; // hochzählen da im Uhrzeigersinn | ||
drehtimUhr_bit = true; //mit dem Uhrzeigersinn | |||
} | } | ||
else // DT ändert sich zuerst wodurch wir uns gegen dem Uhrzeigersinn drehen | else // DT ändert sich zuerst wodurch wir uns gegen dem Uhrzeigersinn drehen | ||
{ | { | ||
drehtimUhr_bit = false; // gegen dem Uhrzeigersinn | |||
PosZaehler_s16 --; // runterzählen | PosZaehler_s16 --; // runterzählen | ||
} | } | ||
float WinkelZaehler_f32= ((float ) PosZaehler_s16/ | |||
Serial.print ("es dreht sich :-): ");//Ausgabe der Ergebnisse | } | ||
drehtsich_bit =true; //AusgabeBit setzen das der Drehencoder sich dreht | |||
} | |||
//------------------------------------------------------------------------------------------------------- | |||
void setup() //definition von Seriellerschnittstelle und der Pins als Eingang | |||
{ | |||
Serial.begin(115200); //übertragungsrate und initalisierung der seriellen Schnittstelle | |||
pinMode(SW_u8, INPUT); // definition als Eingang | |||
pinMode(DT_u8, INPUT_PULLUP); // definition als Eingang | |||
pinMode(CLK_u8, INPUT_PULLUP); // definition als Eingang | |||
CLK_pos_alt_s16=digitalRead(CLK_u8); // einlesen damit Wert vorhanden | |||
attachInterrupt(digitalPinToInterrupt(DT_u8), alarm_DT, CHANGE);// Interrupt 0 auf Pin 2 | |||
attachInterrupt(digitalPinToInterrupt(CLK_u8), alarm_CLK, CHANGE);// // Interrupt 1 auf Pin 3 | |||
CLK_pos_neu_s16 = digitalRead(CLK_u8); // auslesen der Positon von CLK und zwischen speichern in Hilfsvaraiable | |||
} | |||
//------------------------------------------------------------------------------------------------------- | |||
void loop() | |||
{ | |||
if(digitalRead(SW_u8)==false) // Wenn taster gedrückt wird Winkel auf null gesetzt | |||
{ | |||
PosZaehler_s16=0; | |||
} | |||
if (drehtsich_bit) | |||
{ | |||
float WinkelZaehler_f32= ((float ) PosZaehler_s16/ Pos_u16 )* 360;// teilen der Position durch die Anzahl an Positionen am Drehencoder umd Umrechnen auf 360 Grad | |||
Serial.print ("es dreht sich :-): ");//Ausgabe der Ergebnisse | |||
if (drehtimUhr_bit) //Drehrichtungausgabe | |||
{ | { | ||
Serial.print ("mit dem Uhrzeiger"); | |||
} | } | ||
else | |||
{ | { | ||
Serial.print("gegen den Uhrzeiger"); | |||
} | } | ||
Serial.print("Position: "); // Positionsausgabe | |||
Serial.print(PosZaehler_s16); | |||
Serial.print(" Winkel: "); // Positionsausgabe | |||
Serial.println( | Serial.println(WinkelZaehler_f32); | ||
drehtsich_bit=false; | |||
} | |||
else | |||
{ | |||
//Serial.println ("es dreht sich nicht ;-(: ");//Ausgabe der Ergebnisse | |||
} | } | ||
CLK_pos_alt_s16 = CLK_pos_neu_s16; //neuer Messwert | |||
CLK_pos_alt_s16 = CLK_pos_neu_s16; //neuer Messwert wwird u altem Messwert für den nächsten Durchlauf | |||
} | } | ||
Version vom 31. Oktober 2025, 17:34 Uhr

Autoren: Marc Ebmeyer
Einleitung
Einstellräder sind ein wichtiges haptisches Benutzer Interface, sie ermöglichen schnelle intuitive Bedingung von Geräten. Dieses wurde früher mit Ein- und Mehrgangs-Potentiometern gelöst, welche aber einem Verschleiß unterliegen und nur eine (in Ausnahme fällen bis zu 10) Umdrehungen zulassen, zudem sind sie kostenintensiv in der Herstellung. Um diese Probleme zu umgehen, wurden sogenannte Drehimpulsgeber (auch bekannt als Inkrementalgeber, Quadraturencoder, Drehencoder) konstruiert. Welche Kostengünstiger in der Fertigung, unendliche Umdrehungen bieten und mit integrierten Tastern daher kommen. Dieses wurde erst möglich durch die Einführung kostengünstiger Mikrocontroller, welche die Aufgabe der anfallenden Software Auswertung übernehmen. Bei der Drehimpulsmessung gibt es verschiedene verfahren, am häufigsten sind optische, magnetische und mechanische Verfahren. Das Grundprinzip sind aber immer eine oder mehre runde Lochscheiben, wobei die Löcher der einzelnen Scheiben gegeneinander versetzt angeordnet sind. Dreht sich die Scheibe so, triff das Licht (beim optischen Verfahren), welches durch die erste Scheibe abwechselnd durchgelassen und geblockt wird auf einen Sensor, der den Lichtwechsel detektiert und in ein elektrisches Signal umwandelt. Dadurch kann man bei Verwendung nur einer Scheibe, schon mal die Drehzahl und den Drehwinkel detektieren unter dem wissen der Anzahl an Schlitzen in der Lochscheibe. Da man die Anzahl an Unterbrechungen messen kann und die Zeitspanne in der sie ausgeführt wurden. Durch hinzufügen einer weiteren Scheibe deren Löcher leicht versetzt zu ersten Scheibe sind, lässt sich nun auch die Drehrichtung der Achse erkennen. Fügt man eine weitere Scheibe ein, bekommt man die Möglichkeit absolute Positionen zu bekommen. Dieses kann durch einen einzelnen Schlitz sein, der bei jeder Umdrehung die absolute Position bestätigt, oder auch mehrere mit unterschiedlichen Breiten, sodass man mehrere Referenz Positionen bekommt. Der hier verwendete Drehimpulsgeber ist ein einfacher mechanischer, bei ihm finden kleine Phasen versetzte Micro Schleifkontakte ihre Anwendung (siehe Abb.3) , welche über eine leitfähige Lochscheibe schleifen und so zwei Phasen versetzte Signale erzeugen, als wenn man zwei Lochscheiben hätte (siehe Abb.2).
Technische Übersicht
| Eigenschaft | Daten |
|---|---|
| Spannungsversorgung |
VCC 3-5 V |
| Rastungen |
20 |
| Abmessungen |
32*20*30 mm |
| Gewicht |
8 g |
Pinbelegung
siehe Abbildung 6
| Pin | Belegung | Benennung | Signal | Anschluss am Arduino |
|---|---|---|---|---|
| 1 | Masse | GND | 0 V | |
| 2 | Betriebsspannung | Vcc | 3-5 V | |
| 3 | Taster Ausgang | SW | 0 oder VCC V | Wird intern auf Masse gezogen beim betätigen des Tasters siehe Abb.4. Anschluss an z.B. D4 |
| 4 | Signalausgang_B | DT | 0 oder VCC V | An einem Interrupt fähigem Pin des Arduino Uno anschließen z.B. D2 |
| 5 | Signalausgang_A | CLK | 0 oder VCC V | An einem Interrupt fähigem Pin des Arduino Uno anschließen z.B. D3 |
Prinziperklärung
Die unter der Platine befindlichen drei Widerstände (siehe Abb.6) sind Pullup Widerstände für die zwei Encoder und den Taster. Der KY-040 ist ein Inkrementalgeber. Je nach dem ob man rechts oder links herum dreht, wird erst der Pin A oder der Pin B high, da beide versetzt auf der Drehencoderscheibe sitzen (siehe Abb.3). Damit bekommt man einen Gray-Code am Ausgang vom Signalausgang A und B. Gray-code ähneld dem Binär-Code nur wird bei ihm bei jedem Sprung nach oben oder unten jeweils nur ein Wert verändert. Damit bekommen wir die Drehrichtung, den Drehwinkel und die Drehgeschwindigkeit.
Geht zum Beispiel erst CLK auf high und dann DT, dann drehen wir rechts rum. Geht erst der DT Pin auf high und dann der CLK Pin, dann drehen wir links rum.
Teilen wir die Anzahl an CLK durch die vorhandene Inkremente und durch die Zeitspanne der Messung, bekommen wir die Drehzahl.


Bezug zur Prinzipskizze?
| Zahlenwert | Binär-Code | Gray-Code |
|---|---|---|
| 0 | 00 | 00 |
| 1 | 01 | 01 |
| 2 | 10 | 11 |
| 3 | 11 | 10 |
Messung


Demo
Die beiden Demos basieren auf den oben genanten Prinzip. Drückt man den Taster (Achsentaster), dann wird der Nullpunkt neu gesetzt, diese Änderung wird dann aber erst beim weiterdrehen angezeigt.
Demo-Quelltext DemoSwitchCase.ino
//---------------------------------------------------------------------------------------------------------
//---------------------------------------------------------------------------------------------------------
//Drehencoder KY040 auslesen (20 Rastungen 40 Positionen / Umdrehung)
//
// Anschlüsse:
// Drehencoder | Arduino
// ----------------------------------
// GND | GND
// + | +5V
// SW | D4
// DT | D2
// CLK | D3
//
// ausgabe der Position und der Drehrichtung und Winkel
//
//
//
//
//
//
// https://wiki.hshl.de/wiki/index.php/Ky-040-Drehimpulsgeber#Pinbelegung
//
//31.10.2025 Marc Ebmeyer
//
//---------------------------------------------------------------------------------------------------------
//---------------------------------------------------------------------------------------------------------
const byte SW_u8 = 4; // Pin zum Messen vom Schalter
const byte DT_u8 = 2; // Pin zum Messen interrupt pin
const byte CLK_u8 = 3; // Pin zum Messen des Taktes interrupt pin
const unsigned int Pos_u16 = 40; // Anzahl an Positionen des Drehencoders
volatile int CLK_pos_alt_s16; // letzte Position von CLK
volatile int CLK_pos_neu_s16;
volatile bool drehtimUhr_bit; // dreht der Drehencoder im Uhrzeigersinn? true =ja
volatile int PosZaehler_s16 =0; // Zähler der Position
volatile float WinkelZaehler_f32 =0.0;// winkelZähler initalisieren Startposition= o
volatile bool drehtsich_bit =false;
//-------------------------------------------------------------------------------------------------------
void alarm_DT()
{
CLK_pos_neu_s16 = digitalRead(CLK_u8); // auslesen der Positon von CLK und zwischen speichern in Hilfsvaraiable
if (CLK_pos_neu_s16 != CLK_pos_alt_s16)// Drehencoder dreht sich
{
if (digitalRead(DT_u8) != CLK_pos_neu_s16) // Drehrichtungserkennung
{ // CLK ändert sich zuerst -
PosZaehler_s16 ++; // hochzählen da im Uhrzeigersinn
drehtimUhr_bit = true; //mit dem Uhrzeigersinn
}
else // DT ändert sich zuerst wodurch wir uns gegen dem Uhrzeigersinn drehen
{
drehtimUhr_bit = false; // gegen dem Uhrzeigersinn
PosZaehler_s16 --; // runterzählen
}
}
drehtsich_bit =true; //AusgabeBit setzen das der Drehencoder sich dreht
}
//-------------------------------------------------------------------------------------------------------
void alarm_CLK()
{
CLK_pos_neu_s16 = digitalRead(CLK_u8); // auslesen der Positon von CLK und zwischen speichern in Hilfsvaraiable
if (CLK_pos_neu_s16 != CLK_pos_alt_s16)// Drehencoder dreht sich
{
if (digitalRead(DT_u8) != CLK_pos_neu_s16) // Drehrichtungserkennung
{ // CLK ändert sich zuerst -
PosZaehler_s16 ++; // hochzählen da im Uhrzeigersinn
drehtimUhr_bit = true; //mit dem Uhrzeigersinn
}
else // DT ändert sich zuerst wodurch wir uns gegen dem Uhrzeigersinn drehen
{
drehtimUhr_bit = false; // gegen dem Uhrzeigersinn
PosZaehler_s16 --; // runterzählen
}
}
drehtsich_bit =true; //AusgabeBit setzen das der Drehencoder sich dreht
}
//-------------------------------------------------------------------------------------------------------
void setup() //definition von Seriellerschnittstelle und der Pins als Eingang
{
Serial.begin(115200); //übertragungsrate und initalisierung der seriellen Schnittstelle
pinMode(SW_u8, INPUT); // definition als Eingang
pinMode(DT_u8, INPUT_PULLUP); // definition als Eingang
pinMode(CLK_u8, INPUT_PULLUP); // definition als Eingang
CLK_pos_alt_s16=digitalRead(CLK_u8); // einlesen damit Wert vorhanden
attachInterrupt(digitalPinToInterrupt(DT_u8), alarm_DT, CHANGE);// Interrupt 0 auf Pin 2
attachInterrupt(digitalPinToInterrupt(CLK_u8), alarm_CLK, CHANGE);// // Interrupt 1 auf Pin 3
CLK_pos_neu_s16 = digitalRead(CLK_u8); // auslesen der Positon von CLK und zwischen speichern in Hilfsvaraiable
}
//-------------------------------------------------------------------------------------------------------
void loop()
{
if(digitalRead(SW_u8)==false) // Wenn taster gedrückt wird Winkel auf null gesetzt
{
PosZaehler_s16=0;
}
if (drehtsich_bit)
{
float WinkelZaehler_f32= ((float ) PosZaehler_s16/ Pos_u16 )* 360;// teilen der Position durch die Anzahl an Positionen am Drehencoder umd Umrechnen auf 360 Grad
Serial.print ("es dreht sich :-): ");//Ausgabe der Ergebnisse
if (drehtimUhr_bit) //Drehrichtungausgabe
{
Serial.print ("mit dem Uhrzeiger");
}
else
{
Serial.print("gegen den Uhrzeiger");
}
Serial.print("Position: "); // Positionsausgabe
Serial.print(PosZaehler_s16);
Serial.print(" Winkel: "); // Positionsausgabe
Serial.println(WinkelZaehler_f32);
drehtsich_bit=false;
}
else
{
//Serial.println ("es dreht sich nicht ;-(: ");//Ausgabe der Ergebnisse
}
CLK_pos_alt_s16 = CLK_pos_neu_s16; //neuer Messwert wwird u altem Messwert für den nächsten Durchlauf
}
|- |}
Hardwareaufbau


Datenblätter
Literatur
Weiterführende Artikel
→ zurück zum Hauptartikel: HSHL-Mechatronik-Baukasten