Løpende gjennomsnitt for mikrokontrollerprosjektene dine: 6 trinn
Løpende gjennomsnitt for mikrokontrollerprosjektene dine: 6 trinn
Anonim
Løpende gjennomsnitt for mikrokontrollerprosjektene dine
Løpende gjennomsnitt for mikrokontrollerprosjektene dine

I denne instruksen vil jeg forklare hva et løpende gjennomsnitt er og hvorfor du bør bry deg om det, samt vise deg hvordan det skal implementeres for maksimal beregningseffektivitet (ikke bekymre deg for kompleksitet, det er veldig enkelt å forstå, og jeg vil gi et brukervennlig bibliotek for dine arduino -prosjekter også:)

Løpende gjennomsnitt, også ofte referert til som glidende gjennomsnitt, glidende gjennomsnitt eller løpende gjennomsnitt, er et begrep som brukes for å beskrive gjennomsnittlig verdi av siste N -verdier i dataserier. Den kan beregnes som et normalt gjennomsnitt, eller du kan bruke et triks for å få en minimal innvirkning på kodens ytelse.

Trinn 1: Use Case: Utjevning av ADC -målinger

Brukskasse: Utjevning av ADC -målinger
Brukskasse: Utjevning av ADC -målinger

Arduino har en anstendig 10 bit ADC med veldig lite støy. Når man måler verdi på en sensor som potensiometer, fotoresistor eller andre høyfjølende komponenter, er det vanskelig å stole på at målingen er korrekt.

En løsning er å ta flere målinger hver gang du vil lese sensoren din og gjennomsnittlig dem. I noen tilfeller er dette en levedyktig løsning, men ikke alltid. Hvis du ønsket å lese ADC 1000 ganger i sekundet, må du 10 000 hvis du tar gjennomsnittlig 10 målinger. Et enormt sløsing med beregningstid.

Min foreslåtte løsning er å ta målinger 1000 ganger i sekundet, oppdatere løpende gjennomsnitt hver gang og bruke den som nåværende verdi. Denne metoden introduserer litt ventetid, men reduserer beregningskompleksiteten til applikasjonen din, noe som gir deg mye mer tid til ytterligere behandling.

På bildet ovenfor brukte jeg løpende gjennomsnitt av de siste 32 målingene. Du vil se at denne metoden ikke er 100% feilsikker, men den forbedrer nøyaktigheten betydelig (det er ikke verre enn å gjennomsnitt 32 prøver hver gang). Hvis du ville beregne et gjennomsnitt på 32 målinger hver gang, ville det ta over 0,25 ms på Arduino UNO for målinger alene!

Trinn 2: Use Case: Measuring DC Component of Microphone Signal

Brukskasse: Måling av DC -komponent i mikrofonsignal
Brukskasse: Måling av DC -komponent i mikrofonsignal
Brukskasse: Måling av DC -komponent i mikrofonsignal
Brukskasse: Måling av DC -komponent i mikrofonsignal
Brukskasse: Måling av DC -komponent i mikrofonsignal
Brukskasse: Måling av DC -komponent i mikrofonsignal

Arduino kan måle spenninger mellom 0 og Vcc (normalt 5 V). Lydsignalet er helt AC, og hvis du vil måle det på en mikrokontroller, må du forspille det rundt 1/2 Vcc. I et Arduino UNO -prosjekt vil det bety omtrent 2,5 V (DC) + lydsignal (AC). Når du bruker 10 bit ADC og 5 V strømforsyning, bør 2,5 V bias være lik måling på 512. Så for å få en AC -verdi på signalet, bør 512 trekkes fra ADC -måling, og det er det, ikke sant?

I en ideell verden ville det være sant. Dessverre er det virkelige livet mer komplisert og vår signalforstyrrelse har en tendens til å drive. Svært vanlig er 50 Hz støy (60 Hz hvis du bor i USA) fra det elektriske nettverket. Vanligvis er det ikke altfor problematisk, men det er godt å vite at det eksisterer. Mer problematisk er lineær drift fra oppvarming av komponenter. Du setter nøye inn DC offset -korreksjon ved start, og den forsvinner sakte mens applikasjonen din kjører.

Jeg vil illustrere dette problemet med en (musikk) beat -detektor. Du konfigurerer fjerning av skjevhet og slag er klare (bilde 2). Etter en stund beveger DC -bias seg og slag er knapt merkbar for mikrokontrolleren (bilde 3). Slaggjenkjenningsalgoritmen vil bli utforsket grundig i en fremtidig instruksjon, ettersom den overskrider omfanget av denne artikkelen.

Heldigvis er det en måte å konstant beregne lydens DC -forskyvning. Det vil ikke komme som en overraskelse at løpende gjennomsnitt, emnet for dette instruerbare, gir en løsning.

Vi vet at gjennomsnittsverdien for ethvert vekselstrømssignal er 0. Ved å bruke denne kunnskapen kan vi trekke fra at gjennomsnittsverdien for vekselstrøm+likestrømssignal er dens likestrøm. For å fjerne det kan vi ta et løpende gjennomsnitt av de siste verdiene og trekke det fra gjeldende ADC -lesning. Vær oppmerksom på at du må bruke et langt nok løpende gjennomsnitt. For lyd bør en tiendedel av et sekund (antall prøver avhenger av samplingshastigheten) være tilstrekkelig, men vet at lengre gjennomsnitt fungerer bedre. På det første bildet kan du se eksempel på ekte DC -forspenningsberegning med løpende gjennomsnitt med 64 elementer ved 1 kHz samplingsfrekvens (mindre enn jeg anbefalte, men det fungerer fortsatt).

Trinn 3: Beregning

Beregning
Beregning

Du kan forestille deg løpende gjennomsnitt som gjennomsnittsvekt for mennesker på legeventerommet. Legen er ferdig med å undersøke en pasient og samtidig går en ny inn i venterommet.

For å finne ut gjennomsnittlig vekt for alle ventende pasienter i venterommet, kan sykepleier deretter spørre hver pasient om vekten deres, legge disse tallene opp og dele med antall pasienter. Hver gang legen godtar ny pasient, vil sykepleieren gjenta hele prosessen.

Du tenker kanskje: "Dette høres ikke så effektivt ut … Det må være en bedre måte å gjøre dette på." Og du ville ha rett.

For å optimalisere denne prosessen kan sykepleieren føre en oversikt over totalvekten til den nåværende pasientgruppen. Når legen har ringt den nye pasienten inn, vil sykepleieren spørre ham om vekten hans og trekke den fra gruppen totalt og la ham gå. Sykepleier ville da spørre pasienten som nettopp gikk inn i venterommet om vekten hans og legge den til summen. Gjennomsnittlig vekt på pasientene etter hvert skift vil være summen av vekter dividert med antall pasienter (ja, det samme som før, men nå spurte sykepleier bare to personer om vekten deres i stedet for dem alle). Jeg skjønner at dette avsnittet kan ha vært litt forvirrende, så se illustrasjonen ovenfor for ytterligere klarhet (eller still spørsmål i kommentarer).

Men selv om du ikke syntes det siste avsnittet var forvirrende, kan det hende du har spørsmål som hva som skal være i akkumulatoren i begynnelsen, hvordan legger jeg det jeg nettopp leste i en faktisk C -kode? Det vil bli behandlet i neste trinn, hvor du også får kildekoden min.

Trinn 4: Koden

Koden
Koden

For å beregne løpende gjennomsnitt må du først lagre de siste N -verdiene. du kan ha en matrise med N -elementer og flytte hele innholdet ett sted hver gang du legger til et element (ikke gjør dette), eller du kan overskrive et gammelt element og justere pekeren til neste element som skal kastes (vær så snill å gjøre dette:)

Akkumulator bør starte initialisert til 0, det samme gjelder for alle elementer i forsinkelseslinje. I andre tilfeller vil løpende gjennomsnitt alltid være feil. Du vil se at delayLine_init tar seg av å initialisere forsinkelseslinjen, du bør ta vare på akkumulatoren selv.

å legge til et element for å forsinke linjen er like enkelt som å redusere indeksen til det nyeste elementet med 1, og sørg for at det ikke peker på siden av forsinkelseslinjematrisen. etter å ha redusert indeksen når den er 0, vil den gå rundt til 255 (fordi det er et 8 -bits usignert heltall). Modulo (%) -operatør med størrelsen på forsinkelseslinjeserien vil sikre at indeksen peker til et gyldig element.

Å beregne et løpende gjennomsnitt bør være lett å forstå hvis du fulgte analogien min i forrige trinn. Trekk det eldste elementet fra akkumulatoren, legg den nyeste verdien til akkumulatoren, skyv den nyeste verdien til forsinkelseslinjen, returner akkumulatoren delt på antall elementer.

Lett, ikke sant?

Du er velkommen til å eksperimentere med å bruke den vedlagte koden for å bedre forstå hvordan alt dette fungerer. Slik det er nå, leser arduino analog verdi på analog pin A0 og skriver ut "[ADC -verdi], [løpende gjennomsnitt]" på seriell port med 115200 baudhastighet. Hvis du åpner arduinos serielle plotter med riktig overføringshastighet, ser du to linjer: ADC -verdi (blå) og utjevnet verdi (rød).

Trinn 5: Tillegg

Tillegg
Tillegg

Det er noen få ting du ikke nødvendigvis trenger å vite for å bruke løpende gjennomsnitt i prosjektet ditt. Det vil ikke skade å vite.

forsinkelse: Jeg vil begynne med å snakke om illustrasjon av dette trinnet. Du vil legge merke til at løpende gjennomsnitt av flere elementer gir større forsinkelse. Hvis responstiden din for å endre verdi er kritisk, vil du kanskje bruke et kortere løpende gjennomsnitt eller øke samplingsfrekvensen (mål oftere).

Går videre.

initialisering: Da jeg snakket om initialisering av akkumulator og forsinkelseselementer, sa jeg at du skulle initialisere dem alle til 0. Alternativt kan du initialisere forsinkelseslinjen til alt du liker, men akkumulatoren skal starte som en sum av de nyeste N -elementene i forsinkelseslinjen (hvor N er antall elementer i løpende gjennomsnitt). Hvis akkumulator starter som en hvilken som helst annen verdi, vil beregnet gjennomsnitt være feil - enten for lavt eller for høyt, alltid med samme beløp (forutsatt de samme innledende forholdene). Jeg foreslår at du prøver å lære hvorfor dette er slik ved å bruke noen "penn og papirsimulering".

akkumulatorstørrelse: Du bør også merke deg at akkumulatoren skal være stor nok til å lagre summen av alle elementene i forsinkelseslinjen hvis de alle er positive eller negative maks. Det betyr praktisk talt at akkumulator skal være én datatype større enn forsinkelseslinjeelementer og signert hvis forsinkelseslinjeelementer er signert.

triks: Lange forsinkelseslinjer tar mye minne. Det kan fort bli et problem. Hvis du er svært begrenset med hukommelse og ikke bryr deg så mye om nøyaktighet, kan du tilnærme løpende gjennomsnitt ved å utelate forsinkelse helt og gjøre dette i stedet: trekk 1/N * akkumulator fra akkumulatoren og legg til ny verdi (på eksempel på 8 langt løpende gjennomsnitt: akkumulator = akkumulator * 7/8 + newValue). Denne metoden gir feil resultat, men det er en grei metode for å beregne løpende gjennomsnitt når du har lite minne.

lingvistikk: "løpende gjennomsnitt/gjennomsnitt" brukes vanligvis når det refereres til gjennomsnittlig sanntid mens "glidende gjennomsnitt/gjennomsnitt" vanligvis betyr at algoritmen kjører på statiske datasett, for eksempel Excel -regneark.

Trinn 6: Konklusjon

Jeg håper dette var lett å forstå, og at det vil hjelpe deg i dine fremtidige prosjekter. Send gjerne spørsmål i kommentarene nedenfor hvis det er noe uklart.

Anbefalt: