- 12V Akku mit 280 Ah bauen         
Seite 1 von 2 12 LetzteLetzte
Ergebnis 1 bis 10 von 17

Thema: Arduino Roboter - Servo reagiert nicht korrekt

  1. #1
    Benutzer Stammmitglied
    Registriert seit
    04.05.2015
    Beiträge
    41

    Arduino Roboter - Servo reagiert nicht korrekt

    Anzeige

    Powerstation Test
    Hallo,

    momentan bin ich dabei mit einem Arduino Uno, eimem Motorshield, einem Servo, einem Ultraschall-Sensor und einer RP5-Chassis einen Roboter zu konstruieren.

    Wenn er mein Sketch durchläuft und aur ein 10cm oder weniger entferntes Hindernis trifft, soll er stehen bleiben, den Sensor um 45° nach links und dann nach rechts, ebenfalls um 45° drehen. Dabei soll überprüft werden, in welcher Richtung kein Hindernis in 10cm auftaucht. In die Richtung drehen und weiterfahren. Also eigentlich "Standard". Der Sensor wird nach links gedreht, aber nicht nach rechts. Anbei mein Sketch, ich sehe den Fehler nicht.

    Code:
    #include <Servo.h>
    #define echoPin 7 // Echo Pin
    #define trigPin 6 // Trigger Pin
    
    Servo myservo;
    
    int richtungA        = 12;  //  Motor A
    int richtungB        = 13;  //  Motor B
    int geschwindigkeitA =  3;  // Geschwindigkeit Motor A
    int geschwindigkeitB = 11;  // Geschwindigkeit Motor B
    int bremseA          =  9;  // Bremse Motor A
    int bremseB          =  8;  // Bremse Motor B
    
    int geschwindigkeit = 255 ; // Geschwindigkeit der Motoren. 255 ist der Maximalwert!
    
    int maximumRange = 200; // Maximum range needed
    int minimumRange = 0; // Minimum range needed
    long a, b, duration, distance; // Duration used to calculate distance
    
    void kopflinks(){
      //Kopf links
      myservo.write(125);
      digitalWrite(trigPin, LOW); 
      delayMicroseconds(2); 
      digitalWrite(trigPin, HIGH);
      delayMicroseconds(10); 
     
     digitalWrite(trigPin, LOW);
     duration = pulseIn(echoPin, HIGH);
     b = duration/58.2;
    }
    
    
    void kopfrechts(){
      //Kopf rechts
      myservo.write(35);
      digitalWrite(trigPin, LOW); 
      delayMicroseconds(2); 
      digitalWrite(trigPin, HIGH);
      delayMicroseconds(10); 
     
     digitalWrite(trigPin, LOW);
     duration = pulseIn(echoPin, HIGH);
     a = duration/58.2;
      }
    
    
    void setup() 
    {
    
     myservo.attach(5);
      
      // Pins für Motorenrichtungen als Ausgang
      pinMode(richtungA, OUTPUT);
      pinMode(richtungB, OUTPUT);
      
      // Pins für Bremsen als Ausgang
      pinMode(bremseA, OUTPUT);
      pinMode(bremseB, OUTPUT);
    
      // Geschwindigkeiten setzen
      analogWrite(geschwindigkeitA, geschwindigkeit);
      analogWrite(geschwindigkeitB, geschwindigkeit);
      
      // Bremse an (Motoren aus)
      digitalWrite(bremseA, HIGH);
      digitalWrite(bremseB, HIGH);
    
      // 2 Sekunden warten nach einschalten
      delay(2000);
    
     //US-Sensor
     pinMode(trigPin, OUTPUT);
     pinMode(echoPin, INPUT);
    
     //LEDs
     pinMode(2, OUTPUT);
     pinMode(4, OUTPUT);
    
     //Servo 0-Position (etwas schief, daher 80°)
     myservo.write(80);
    
    }
    
    void loop() 
    {
    
     digitalWrite(trigPin, LOW); 
     delayMicroseconds(2); 
    
     digitalWrite(trigPin, HIGH);
     delayMicroseconds(10); 
     
     digitalWrite(trigPin, LOW);
     duration = pulseIn(echoPin, HIGH);
     
     //Distanzberechnung des Ultraschall-Signals.
     distance = duration/58.2;
    
      
      //Distanz größer 10cm, fährt vorwärts
      if (distance > 10){
      digitalWrite(richtungA, HIGH);
      digitalWrite(richtungB, HIGH);
      digitalWrite(bremseA, LOW);
      digitalWrite(bremseB, LOW);
      digitalWrite(2, HIGH);
      digitalWrite(4, LOW);
     }
    
     //Distanz kleiner gleich 10cm, Bremse ein, LED wechsel und Kopf drehen.
     else if (distance <=10){
    
      digitalWrite(bremseA, HIGH);
      digitalWrite(bremseB, HIGH);
      digitalWrite(2, LOW);
      digitalWrite(4, HIGH);
      delay(1000);
    
      kopflinks();
      delay(500);
      kopfrechts();
      myservo.write(80);
      
      if (a > 10 && b <=10){
      digitalWrite(richtungA, HIGH);
      digitalWrite(richtungB, LOW);
      digitalWrite(bremseA, LOW);
      digitalWrite(bremseB, LOW);
      delay(500);
      digitalWrite(bremseA, HIGH);
      digitalWrite(bremseB, HIGH);
      digitalWrite(richtungA, HIGH);
      digitalWrite(richtungB, HIGH);
      digitalWrite(bremseA, LOW);
      digitalWrite(bremseB, LOW);
      }
    
      else if (a<=10 && b > 10){
      digitalWrite(richtungA, LOW);
      digitalWrite(richtungB, HIGH);
      digitalWrite(bremseA, LOW);
      digitalWrite(bremseB, LOW);
      delay(500);
      digitalWrite(bremseA, HIGH);
      digitalWrite(bremseB, HIGH);
      digitalWrite(richtungA, HIGH);
      digitalWrite(richtungB, HIGH);
      digitalWrite(bremseA, LOW);
      digitalWrite(bremseB, LOW);
      }
     }
    }
    Ohne Servofunktion läuft der Roboter normal.

  2. #2
    Erfahrener Benutzer Robotik Einstein Avatar von i_make_it
    Registriert seit
    29.07.2008
    Ort
    Raum DA
    Alter
    56
    Beiträge
    2.814
    Versuchs mal mit
    Code:
      kopflinks();
      delay(500);
      kopfrechts();
      delay(500);
    Beim Überfliegen, scheint es mir so als ob Du dem RC-Servo gar keine Zeit lässt nach rechts zu fahren.
    Allerdings wird Dir das nichts bringen, Soweit ich das sehe, dürfte Kopflinks nahezu das selbe Messergebniss liefern wie wenn der Kopf gradeaus schaut.
    So ein RC-Servo hat ja eine Stellzeit z.B. 0,6s für 60°.
    Soweit ich das sehe startest Du die Servobewegung und trigerst danach sofort den Sensor.
    Die Messung müsste also erfolgen wärend sich der RC_Servo grade anfängt auf den Weg zur Zielposition zu machen.
    Da RC-Servos intern regeln, hat man keine Rückmeldung wann ein Servo seine Zielposition erreicht hat.
    Man schätzt entweder die Zeit die der RC_Servo braucht und arbeitet mit einem Delay und hofft das es keine Fehlersituation gibt in der der RC_Servo blokiert, oder man braucht ein Messystem daß einem das Erreichen der Zielposition anzeigt.
    Im Zweifelsfall auf jeder Seite einen Taster.
    Und wenn man rafiniert ist, spendiert man dem ganzen Betätigungsnocken, die bei Endlage 1 Taster 1 öffnen und bei Endlage 2 Taster 2 öffnen.
    in Mittelstellung sind beide Taster dann offen.
    Das heist wärend der Drehung sind beide Taster geschlossen.
    Strohm fließt also nur wärend der Bewegung (Strom sparen) und man kann kabelbrüche erkennen, da ja nach dem Drehbefehl beide Taster auf 1 gehen müssen und in den Endlagen jeweils 1 Taster auf 1 bleibt.
    kommen diese Bitmuster nicht zustande, dann ist irgendwo ein Fehler und der Roboter kann in einen Fehlermodus gehen. (z.B. blinken und piepen)

  3. #3
    Benutzer Stammmitglied
    Registriert seit
    04.05.2015
    Beiträge
    41
    Danke für den Hinweis mit dem Delay. Das hat ein Teil des Problems gelöst. Der Servo dreht sich nun von nach links, dann kommt der "ping", dann dreht er sich nach rechts "ping". Aber die Werte werden nicht in den Variable a und b gespeichert. Ich habe die Bedingungen angepasst auf if (a<b), dann drehe dich in die eine Richtung oder (a>b), dann in die andere Richtung. Wenn ich mir die Werte für a und b ausgeben lasse, steht immer 0 da, damit ist keine Bedingung erfüllt und der Kopf geht hin und her und hin und her.

  4. #4
    Erfahrener Benutzer Roboter-Spezialist
    Registriert seit
    13.01.2014
    Beiträge
    454
    Blog-Einträge
    3
    Hallo wendtdir,
    ich habe deinen Code mal ein wenig aufgeräumt (nicht vollständig). Vergleiche mal mit deinem Code.
    Ich hoffe es fällt auf, dass der aufgeräumte Code deutlich lesbarer ist. Wenn es um Fehlersuche geht, ist Lesbarkeit ein sehr wichtiger Faktor. Da fallen dann Dinge ins Auge, die vorher im "Gewusel" untergegangen sind.
    Code:
    #include <Servo.h>
    
    #define echoPin    7  
    #define trigPin    6  
    #define motorAPin 12
    #define motorBPin 13
    #define vAPin      3
    #define vBPin     11
    #define bremseAPin 9
    #define bremseBPin 8
    #define led1Pin    2
    #define led2Pin    4
    #define servoPin   5
    
    Servo myservo;
    
    void kopf_rechts(){
      myservo.write(35);
    }
    
    void kopf_links(){
      myservo.write(125);
    }
    
    void kopf_mitte(){
      //Servo 0-Position (etwas schief, daher 80°)
      myservo.write(80);
    }
    
    void geradeaus_fahren() {
      digitalWrite(motorAPin, HIGH);
      digitalWrite(motorBPin, HIGH);
      digitalWrite(bremseAPin, LOW);
      digitalWrite(bremseBPin, LOW);
      digitalWrite(led1Pin, HIGH);
      digitalWrite(led2Pin, LOW);
    }
    
    unsigned long ping_cm() {
      // Ultraschall-Messung
      // gibt Distanz in cm zurück
      digitalWrite(trigPin, LOW); 
      delayMicroseconds(2); 
      digitalWrite(trigPin, HIGH);
      delayMicroseconds(10); 
      digitalWrite(trigPin, LOW);
      // maximale Pulslänge bei 3m Distanz,  v_Schall = 340m/s
      const unsigned long maxPulse_us = (3 * 2 * 1000000) / 340;  
      unsigned long duration = pulseIn(echoPin, HIGH, maxPulse_us);
      if (duration == 0) return 300;  //kein Echo -> Rückgabe max.Messweite
      return duration / 58.2;
    }
    
    void setup(){
      myservo.attach(servoPin);
      kopf_mitte();
     
      // Pins für Motorenrichtungen als Ausgang
      pinMode(motorAPin, OUTPUT);
      pinMode(motorBPin, OUTPUT);
      
      // Pins für Bremsen als Ausgang
      pinMode(bremseAPin, OUTPUT);
      pinMode(bremseBPin, OUTPUT);
    
      // Geschwindigkeiten voll setzen
      analogWrite(vAPin, 255);
      analogWrite(vBPin, 255);
      
      // Bremse an (Motoren aus)
      digitalWrite(bremseAPin, HIGH);
      digitalWrite(bremseBPin, HIGH);
    
      // 2 Sekunden warten nach einschalten
      delay(2000);
    
      //US-Sensor
      pinMode(trigPin, OUTPUT);
      pinMode(echoPin, INPUT);
    
      //LEDs
      pinMode(led1Pin, OUTPUT);
      pinMode(led2Pin, OUTPUT);
    }
    
    void loop(){
      //Distanz mit Ultraschall-Sensor ermitteln
      long distance = ping_cm();
      
      //Distanz größer 10cm, fährt vorwärts
      if (distance > 10) {
        geradeaus_fahren();
      }
      else { //Distanz kleiner gleich 10cm
        // Bremse ein, LED wechsel und Kopf drehen.
        digitalWrite(bremseAPin, HIGH);
        digitalWrite(bremseBPin, HIGH);
        digitalWrite(led1Pin, LOW);
        digitalWrite(led2Pin, HIGH);
    
        kopf_links();   
        delay(500);
        long links_cm = ping_cm();
    
        kopf_rechts(); 
        delay(500);
        long rechts_cm = ping_cm();
          
        if (links_cm > 10 && rechts_cm <= 10){
          digitalWrite(motorAPin, HIGH);
          digitalWrite(motorAPin, LOW);
          digitalWrite(bremseAPin, LOW);
          digitalWrite(bremseBPin, LOW);
          delay(500);
          geradeaus_fahren();
        }
        else if (links_cm <= 10 && rechts_cm > 10){
          digitalWrite(motorAPin, LOW);
          digitalWrite(motorAPin, HIGH);
          digitalWrite(bremseAPin, LOW);
          digitalWrite(bremseBPin, LOW);
          delay(500);
          geradeaus_fahren();
        }
    
        kopf_mitte();
        delay(500);
      }
    }
    Geändert von Sisor (17.11.2016 um 22:32 Uhr) Grund: Änderung der Ultraschall-Messroutine

  5. #5
    Erfahrener Benutzer Robotik Einstein Avatar von i_make_it
    Registriert seit
    29.07.2008
    Ort
    Raum DA
    Alter
    56
    Beiträge
    2.814
    Was mir persöhnlich an dem Code nicht gefällt, ist die verwendung der vielen Delays
    delay()
    delayMicroseconds()
    PulseIn()
    sind ja alles Befehle die das Programm solange anhalten bis sie fertig sind.
    Das wird also ein extrem langsam reagierneder Roboter.
    Da der Prozessor die meiste Zeit damit verbringt zu warten.
    Duch geschickte Programmierung und Verwendung von Interrupts und Timern sowie Vergleichen, kann mann das Programm weiterlaufen lassen und somit auch andere Aufgaben ausführen.
    Soll der Roboter also später mehr Funktionern bekommen wäre es sinnvoll sich frühzeitig mit den anderen Möglichkeiten zu befassen, bevor man sehr viel komplett neu entwickeln muß.

    Ist halt meine persöhnliche Meinung.

    Momentan lasse ich mit einem Arduino Nano, zwei hc-sr04 auf zwei RC-Microservos, drei IR Reflexlichtschranken (Für Linienfolger), 4 Schalter (Bumper für Kolisionserkennung) und zwei gehackte RC-Servos (als Getriebemotoren für die Räder), laufen.
    Das sind 4 Ausgänge für die RC-Servos, 2 Ausgänge für die Trigger der HC-SR04, die beiden Interrupteingänge für die HC-SR04, 3 Digitaleingänge für die IR-Lichtschranken, und ein Analogeingang für die Bumper (ein Widerstandsnetzwerk generiert verschiedene Spannungen für die Schalterkombinationen).
    Als nächstes kommt Odometrie hinzu womit nochmal 4 Digitaleingänge belegt werden.
    Da der dann nicht ständig Denkpausen einlegen soll, gibt es bisher im Code kein einziges Delay.
    Und es soll auch keins hinzukommen.

  6. #6
    Erfahrener Benutzer Roboter-Spezialist
    Registriert seit
    13.01.2014
    Beiträge
    454
    Blog-Einträge
    3
    Die Servo-Bibliothek verwendet Timer und Interrupts.

  7. #7
    Erfahrener Benutzer Robotik Einstein Avatar von i_make_it
    Registriert seit
    29.07.2008
    Ort
    Raum DA
    Alter
    56
    Beiträge
    2.814
    Ich meinte ja auch explizit:
    delay()
    delayMicroseconds()
    PulseIn()
    und nicht die Servo Lib.

    PulseIn wartet im Fehlerfall z.B. per Default eine Sekunde bis es weitergeht.
    Mit den beiden "delay(500)" wären das im Extremfall bei einem Messzyklus (links-rechts) 3 Sekunden.
    Und beim Gradeausfahren eine Sekunde von einer Fehlmessung bis zur nächsten Messung die dann eventuell ein gültiges Ergebniss liefert.
    Der hc-sr04 z.B. ist bei einer Fehlmessung nach maximal 0,2 Sekunden wieder bereit zur nächten Messung

  8. #8
    Benutzer Stammmitglied
    Registriert seit
    04.05.2015
    Beiträge
    41
    Hallo,

    erstmal danke ich euch für die Hilfe. Es ist mein erster Roboter den ich momentan am Basteln bin, daher ist mein Wissen natürlich auch nicht besonders groß.

    Ich glaube Sisor hatte mir auch mal Links zu den Interrupt-Programmierungen gepostet, als ich nicht klar kam mit meiner Schaltung Alarmanlage/Temeperaturmesser.

    Ich möchte erstmal mit den zugegebenermaßen beschränkten Fertigkeiten ein kleines Ziel erreichen. Ich bin auch am überlegen, wie ich es einrichten kann, wenn er denn endlich so läuft, wie ich es möchte, dass er gleichzeitig zwei Abläufe ausführt, nämlich wendet und den Kopf wieder in "0"-Position bringt. Ist das möglich durch Interrupts?

    Hier ein Bild meines kleinen:

    Klicke auf die Grafik für eine größere Ansicht

Name:	Rob.jpg
Hits:	12
Größe:	53,4 KB
ID:	32206
    Geändert von wendtdir (19.11.2016 um 13:20 Uhr)

  9. #9
    Erfahrener Benutzer Roboter-Spezialist
    Registriert seit
    13.01.2014
    Beiträge
    454
    Blog-Einträge
    3
    Ich möchte erstmal mit den zugegebenermaßen beschränkten Fertigkeiten ein kleines Ziel erreichen. Ich bin auch am überlegen, wie ich es einrichten kann, wenn er denn endlich so läuft, wie ich es möchte, dass er gleichzeitig zwei Abläufe ausführt, nämlich wendet und den Kopf wieder in "0"-Position bringt. Ist das möglich durch Interrupts?
    Hallo wenddir,

    lass dich nicht verrückt machen. Z.Z. hat dein Programm vier Hauptaufgaben:
    - Fahren
    - Kopfposition ändern
    - Ultraschallmessung
    - Entscheidungen aufgrund der Messung treffen

    Das Fahren passiert in dem Programm automatisch. Dein Roboter kann Messungen machen oder die Kopfposition ändern, während er fährt.
    Da dein Roboter nur einen Sensor besitzt, ist für die Entscheidung, was zu tun ist, auch nur das eine Messergebnis entscheidend. Der Roboter muss nicht gleichzeitig mehrere Sensoren auswerten. Daher muss dein Programm auch keine weiteren Dinge parallel abarbeiten. Interupt- oder schedulerbasierte Programmierung ist also z.Z. nicht nötig.

    Der Post von I_make_it bezieht sich darauf, dass das Programm umgeschrieben werden müsste, wenn du das System um weitere Sensoren erweiterst wie z.B. Kontaktsensoren oder ähnliches.

    Das von dir oben genannte Verhalten wäre dann so realisierbar.
    Code:
        
    void kopf_mitte() {
      //Servo 0-Position (etwas schief, daher 80°)
      myservo.write(80);
    }
    void wenden() {  
       digitalWrite(motorAPin, HIGH);
       digitalWrite(motorAPin, LOW);
       digitalWrite(bremseAPin, LOW);
       digitalWrite(bremseBPin, LOW);
       delay(500); // Hier Wendezeit eintragen
    }
    void loop() {
      // ...
      kopf_mitte(); // Servo beginnt Bewegung
      wenden(); // Servo bewegt sich während des Wendens rasch zur Mittelposition
      // ...
    Geändert von Sisor (19.11.2016 um 15:16 Uhr)

  10. #10
    Erfahrener Benutzer Robotik Visionär Avatar von oberallgeier
    Registriert seit
    01.09.2007
    Ort
    Oberallgäu
    Beiträge
    8.685
    .. möchte, dass er gleichzeitig zwei Abläufe ausführt, nämlich wendet und den Kopf wieder in "0"-Position bringt. Ist das möglich durch Interrupts? ..
    Die triviale Antwort ist : ja.

    Das hatte ich zuerst mit meinem Dottie (siehe "Innenleben" - klick) gemacht. Servo agiert - und gleichzeitig schwenkt der Fahrebot, siehe Video durch Klick auf Bild. Nur - ich habs in C programmiert, Arduino-language kann ich nit. Aber es sollte sicher gehen - und Sisor schrieb ja auch schon wie´s geht.
    Ciao sagt der JoeamBerg

Seite 1 von 2 12 LetzteLetzte

Ähnliche Themen

  1. SPI SDO Pegel nicht korrekt
    Von Che Guevara im Forum Elektronik
    Antworten: 2
    Letzter Beitrag: 16.01.2014, 16:54
  2. Servo reagiert nicht
    Von Jeti im Forum Elektronik
    Antworten: 24
    Letzter Beitrag: 04.04.2013, 16:19
  3. [ERLEDIGT] Servo reagiert nicht
    Von miicchhii im Forum Motoren
    Antworten: 15
    Letzter Beitrag: 20.07.2011, 20:11
  4. Antworten: 7
    Letzter Beitrag: 28.04.2011, 19:44
  5. Pic führtprogramm nicht korrekt aus
    Von cavorca im Forum PIC Controller
    Antworten: 3
    Letzter Beitrag: 16.04.2007, 20:01

Berechtigungen

  • Neue Themen erstellen: Nein
  • Themen beantworten: Nein
  • Anhänge hochladen: Nein
  • Beiträge bearbeiten: Nein
  •  

Labornetzteil AliExpress