Optimering er ændring af et system for at forbedre dets effektivitet. Et system kan være et enkelt computerprogram , en digital enhed, en samling af computere eller endda et helt netværk.
Selvom målet med optimering er at opnå et optimalt system, opnås et virkelig optimalt system ikke altid i optimeringsprocessen. Et optimeret system er normalt optimalt for kun én opgave eller gruppe af brugere: et eller andet sted kan det være vigtigere at reducere den tid, det tager for programmet at fuldføre arbejdet, selv på bekostning af at forbruge mere hukommelse; i applikationer, hvor hukommelse er vigtigere, kan langsommere algoritmer med mindre hukommelseskrav vælges.
Desuden er der ofte ingen universel løsning (fungerer godt i alle tilfælde), så ingeniører bruger afvejningsløsninger til kun at optimere nøgleparametre. Derudover overstiger indsatsen for at opnå et helt optimalt program, der ikke kan forbedres yderligere, næsten altid det udbytte, der kan opnås heraf, hvorfor optimeringsprocessen som udgangspunkt er afsluttet, før fuld optimalitet er nået. Heldigvis, i de fleste tilfælde, selv med dette, opnås mærkbare forbedringer.
Optimering skal ske med omhu. Tony Hoare sagde først, og Donald Knuth gentog ofte senere det berømte ordsprog: "For tidlig optimering er roden til alt ondt." Det er meget vigtigt at have en stemt algoritme og en fungerende prototype til at begynde med.
Nogle opgaver kan ofte udføres mere effektivt. For eksempel et C -program , der summerer alle heltal fra 1 til N:
int i , sum = 0 ; for ( i = 1 ; i <= N ; i ++ ) sum += i ;Forudsat at der ikke er noget overløb her , kan denne kode omskrives som følger ved at bruge den passende matematiske formel :
int sum = ( N * ( N + 1 )) / 2 ;Udtrykket "optimering" indebærer normalt, at systemet bevarer den samme funktionalitet. Imidlertid kan betydelige præstationsforbedringer ofte opnås ved at fjerne overflødig funktionalitet. For eksempel, forudsat at programmet ikke behøver at understøtte mere end 100 elementer på input , er det muligt at bruge statisk allokering i stedet for den langsommere dynamiske allokering .
Optimering fokuserer hovedsageligt på enkelt eller gentagen eksekveringstid, hukommelsesforbrug, diskplads, båndbredde eller en anden ressource. Dette kræver normalt afvejninger – én parameter optimeres på bekostning af andre. For eksempel forbedrer en forøgelse af softwarecache- størrelsen på noget kørselsydelse, men øger også hukommelsesforbruget . Andre almindelige afvejninger omfatter kodegennemsigtighed og udtryksfuldhed, næsten altid på bekostning af de-optimering. Komplekse specialiserede algoritmer kræver mere fejlfindingsindsats og øger risikoen for fejl .
I operationsforskning er optimering problemet med at bestemme inputværdierne for en funktion, for hvilken den har en maksimum- eller minimumværdi. Nogle gange er der begrænsninger på disse værdier, sådan en opgave er kendt som begrænset optimering .
I programmering betyder optimering normalt at ændre koden og dens kompileringsindstillinger for en given arkitektur for at producere mere effektiv software.
Typiske problemer har så mange muligheder, at programmører normalt kun kan tillade, at en "god nok" løsning bruges.
For at optimere skal du finde en flaskehals ( engelsk bottleneck - bottleneck): en kritisk del af koden, som er hovedforbrugeren af den nødvendige ressource. Forbedring af omkring 20 % af koden indebærer nogle gange ændring af 80 % af resultaterne ifølge Pareto-princippet . Lækage af ressourcer (hukommelse, håndtag osv.) kan også føre til et fald i hastigheden af programudførelse. For at søge efter sådanne lækager bruges specielle fejlfindingsværktøjer, og profiler bruges til at opdage flaskehalse .
Det arkitektoniske design af et system har en særlig stærk indflydelse på dets ydeevne. Valg af algoritme påvirker effektiviteten mere end noget andet designelement. Mere komplekse algoritmer og datastrukturer kan håndtere et stort antal elementer godt, mens simple algoritmer er gode til små mængder data - omkostningerne ved at initialisere en mere kompleks algoritme kan opveje fordelen ved at bruge den.
Jo mere hukommelse et program bruger, jo hurtigere kører det typisk. For eksempel læser et filterprogram typisk hver linje, filtrerer og udsender den linje direkte. Derfor bruger den kun hukommelse til at gemme én linje, men dens ydeevne er normalt meget dårlig. Ydeevnen kan forbedres betydeligt ved at læse hele filen og derefter skrive det filtrerede resultat, men denne metode bruger mere hukommelse. Resultatcache er også effektiv, men kræver mere hukommelse at bruge.
Optimering med hensyn til processortid er især vigtig for beregningsprogrammer, hvor matematiske beregninger har en stor andel. Her er nogle optimeringsteknikker, som en programmør kan bruge, når han skriver programkildekode.
I mange programmer skal en del af dataobjekterne initialiseres , det vil sige, at de skal tildeles startværdier. En sådan opgave udføres enten i begyndelsen af programmet, eller for eksempel i slutningen af løkken. Korrekt objektinitialisering sparer værdifuld CPU-tid. Så når det for eksempel kommer til initialisering af arrays, vil det sandsynligvis være mindre effektivt at bruge en loop end at erklære det array som en direkte tildeling.
I det tilfælde, hvor en væsentlig del af programmets tid er afsat til aritmetiske beregninger, er betydelige reserver til at øge programmets hastighed gemt i den korrekte programmering af aritmetiske (og logiske) udtryk. Det er vigtigt, at forskellige aritmetiske operationer adskiller sig væsentligt i hastighed. På de fleste arkitekturer er de hurtigste operationer addition og subtraktion. Multiplikation er langsommere, efterfulgt af division. For eksempel udføres beregningen af værdien af udtrykket , hvor er en konstant, for flydende komma-argumenter hurtigere i formen , hvor er en konstant beregnet på programkompileringsstadiet (faktisk er den langsomme divisionsoperation erstattet af den hurtige multiplikationsoperation). For et heltalsargument er det hurtigere at beregne udtrykket i formen (multiplikationsoperationen erstattes af additionsoperationen) eller at bruge venstreforskydningsoperationen (som ikke giver en forstærkning på alle processorer). Sådanne optimeringer kaldes driftsstyrkereduktion . Multiplicering af heltalsargumenter med en konstant på x86- familieprocessorer kan udføres effektivt ved hjælp af assembler- instruktioner og i stedet for at bruge instruktioner : LEASHLADDMUL/IMUL
; Kildeoperand i register EAX ADD EAX , EAX ; gange med 2 LEA EAX , [ EAX + 2 * EAX ] ; gange med 3 SHL EAX , 2 ; gange med 4 LEA EAX , [ 4 * EAX ] ; en anden måde at implementere multiplikation med 4 LEA EAX , [ EAX + 4 * EAX ] ; gange med 5 LEA EAX , [ EAX + 2 * EAX ] ; gange med 6 ADD EAX , EAX ; etc.Sådanne optimeringer er mikroarkitektoniske og udføres normalt gennemsigtigt for programmøren af optimeringskompileren .
Relativt meget tid bruges på at kalde subrutiner (passere parametre på stakken , gemme registre og returadresser, kalde kopikonstruktører). Hvis subrutinen indeholder et lille antal handlinger, kan den implementeres inline ( engelsk inline ) - alle dens udsagn kopieres til hvert nyt opkaldssted (der er en række begrænsninger for inline substitutioner: for eksempel må subrutinen ikke være rekursiv ). Dette eliminerer omkostningerne ved at kalde subrutinen, men øger størrelsen af den eksekverbare fil. I sig selv er stigningen i størrelsen af den eksekverbare fil ikke signifikant, men i nogle tilfælde kan den eksekverbare kode gå ud over instruktionscachen , hvilket vil føre til et betydeligt fald i programafviklingshastigheden. Derfor har moderne optimeringskompilere normalt optimeringsindstillinger for kodestørrelse og eksekveringshastighed.
Ydeevnen afhænger også af typen af operander. For eksempel, i Turbo Pascal-sproget , på grund af implementeringen af heltalsaritmetik, viser additionsoperationen sig at være den langsomste for operander af typen Byteog ShortInt: på trods af at variabler optager en byte, er aritmetiske operationer for dem to-byte og når der udføres operationer på disse typer, nulstilles registrenes høje byte, og operanden kopieres fra hukommelsen til registrets lave byte. Dette medfører ekstra tidsomkostninger.
Når man programmerer aritmetiske udtryk, bør man vælge en sådan form for deres notation, så antallet af "langsomme" operationer minimeres. Lad os overveje et sådant eksempel. Lad det være nødvendigt at beregne et polynomium af 4. grad:
Forudsat at beregningen af graden udføres ved at gange grundtallet et vist antal gange, er det let at finde ud af, at dette udtryk indeholder 10 multiplikationer ("langsomme" operationer) og 4 additioner ("hurtige" operationer). Det samme udtryk kan skrives som:
Denne form for notation kaldes Horners skema . Dette udtryk har 4 multiplikationer og 4 additioner. Det samlede antal operationer er reduceret med næsten det halve, og tiden for beregning af polynomiet vil også falde tilsvarende. Sådanne optimeringer er algoritmiske og udføres normalt ikke automatisk af compileren.
Udførelsestiden for cyklusser af forskellige typer er også forskellig. Udførelsestiden for en løkke med en tæller og en løkke med en postbetingelse, alt andet lige, udføres løkken med en forudsætning lidt længere (med ca. 20-30%).
Når du bruger indlejrede sløjfer, skal du huske på, at CPU-tiden brugt på at behandle en sådan konstruktion kan afhænge af rækkefølgen af de indlejrede sløjfer. For eksempel en indlejret løkke med en tæller i Turbo Pascal :
for j := 1 til 100000 gør for k := 1 til 1000 gør a := 1 ; | for j := 1 til 1000 gør for k := 1 til 100000 gør a := 1 ; |
Cyklussen i venstre kolonne tager omkring 10 % længere tid end i højre kolonne.
Ved første øjekast, både i det første og det andet tilfælde, udføres opgaveoperatøren 100.000.000 gange, og den tid, der bruges på det, bør være den samme i begge tilfælde. Men det er det ikke. Dette forklares af det faktum, at sløjfeinitiering (behandling af processoren af dens header for at bestemme tællerens indledende og endelige værdier samt tællerinkrementet) tager tid. I det første tilfælde initialiseres den ydre løkke 1 gang, og den indre løkke initialiseres 100.000 gange, det vil sige, at der i alt udføres 100.001 initialiseringer. I det andet tilfælde er der kun 1001 sådanne initialiseringer.
Indlejrede sløjfer med precondition og postcondition opfører sig på samme måde. Vi kan konkludere, at når du programmerer indlejrede løkker, hvis det er muligt, skal du gøre løkken med det mindste antal gentagelser til den yderste, og løkken med det største antal gentagelser til den inderste.
Hvis sløjferne indeholder hukommelsesadgange (normalt ved behandling af arrays), for den mest effektive brug af cachen og hardware-forhentning af data fra hukommelsen ( engelsk Hardware Prefetch ), bør rækkefølgen af omgåelse af hukommelsesadresser være så sekventiel som muligt. Et klassisk eksempel på en sådan optimering er omorganiseringen af indlejrede løkker, når der udføres matrixmultiplikation .
Optimeringen af invariante kodefragmenter er tæt forbundet med problemet med optimal loop-programmering. Inde i løkken kan der være udtryk, hvis fragmenter ikke på nogen måde afhænger af løkkens kontrolvariabel. De kaldes invariante kodestykker. Moderne compilere opdager ofte tilstedeværelsen af sådanne fragmenter og optimerer dem automatisk. Dette er ikke altid muligt, og nogle gange afhænger et programs ydeevne helt af, hvordan loopet er programmeret. Som et eksempel kan du overveje følgende programfragment ( Turbo Pascal-sprog ):
for i := 1 til n begynder ... for k := 1 til p gør for m : = 1 til q begynder a [ k , m ] : = Sqrt ( x * k * m - i ) + Abs ( u * i - x * m + k ) ; b [ k , m ] := Sin ( x * k * i ) + Abs ( u * i * m + k ) ; ende ; ... am := 0 ; bm := 0 ; for k := 1 til p gør for m := 1 til q begynder am : = am + a [ k , m ] / c [ k ] ; bm := bm + b [ k , m ] / c [ k ] ; ende ; ende ;Her er de invariante kodefragmenter summand Sin(x * k * i)i den første løkke over variablen mog operationen af division af array-elementet c[k]i den anden løkke over m. Værdierne af sinus og array-elementet ændres ikke i løkken over variablen m, derfor kan du i det første tilfælde beregne værdien af sinus og tildele den til en hjælpevariabel, der vil blive brugt i udtrykket inde i løkken. I det andet tilfælde kan du udføre divisionen efter slutningen af løkken over m. Dermed kan antallet af tidskrævende regneoperationer reduceres væsentligt.