„Šio“ konteksto „TypeScript“ klaidų taisymas senuosiuose joniniuose / kampiniuose projektuose naudojant RxJS

TypeScript

Suderinamumo iššūkių sprendimas senose Angular programose

Jei neseniai nuvalėte dulkes nuo senesnio ir susidūrėte su netikėtomis „TypeScript“ klaidomis, jūs ne vieni! 🛠️ Klaidos kaip "“ gali būti ypač klaidinantis seniai veikiančiose programose, kur naudojimo nutraukimas ir API pakeitimai apsunkina kūrimo procesą.

Šiame straipsnyje mes pasinersime į vieną iš dažniausiai pasitaikančių problemų, susijusių su , ypač naudojant neasinchronines funkcijas kontekstuose, kuriuose tikimasi asinchroninių. Dėl tokių neatitikimų dažnai atsiranda „TypeScript“ klaidų, kurios gali blokuoti kūrimą ir sustabdyti kūrimo eigą.

Išnagrinėsime, kaip įveikti šias „TypeScript“ kliūtis, suprasti pagrindinę priežastį ir pasidalinti metodais, kaip koreguoti RxJS kodą, kad išvengtumėte šių klaidų. Be to, paryškinsime naudingus įrankius kurie gali pagreitinti jūsų darbo eigą ir padaryti derinimą lengvai.

Nesvarbu, ar norite išspręsti problemas, ar gauti įžvalgų apie senojo kodo atnaujinimą, šiame vadove rasite įžvalgų ir praktinių veiksmų, kurių reikia norint greitai ir efektyviai išspręsti šias „TypeScript“ klaidas. ⚙️

komandą Aprašymas ir naudojimas
createEffect NgRx dalis, createEffect yra naudojama norint apibrėžti šalutinį poveikį, kurį sukelia išsiųsti veiksmai. Tai leidžia mums valdyti asinchroninę logiką Angular reaktyviajame programavimo modelyje, kuris yra labai svarbus valdant būseną sudėtingose ​​​​programose.
ofType Šis operatorius filtruoja veiksmus NgRx efektuose pagal veiksmo tipą. Tai užtikrina, kad būtų atliekami tik veiksmai, atitinkantys nurodytą tipą (šiuo atveju UPDATE_ORG_SUCCESS), todėl tam tikra logika gali būti taikoma tik norimiems veiksmams.
combineLatest „combinLatest“ yra RxJS operatorius, leidžiantis sujungti kelis stebimus objektus, skleidžiant naujausias reikšmes kaip naują kombinuotą masyvą, kai skleidžia bet kuris iš šaltinio stebimų elementų. Tai naudinga, kai reikia sinchronizuotų duomenų iš kelių šaltinių, pvz., iššūkių sąrašo ir metrikos čia.
switchMap Naudojamas vidiniam stebimam objektui išlyginti ir susieti su išoriniu stebimu, „switchMap“ atšaukia ankstesnių stebimų objektų prenumeratą, kai gaunama nauja reikšmė, todėl puikiai tinka tvarkyti besikeičiančius asinchroninius duomenis, pvz., organizacijos atnaujinimo įvykius šiame pavyzdyje.
filter RxJS operatorius, leidžiantis filtruoti reikšmes pagal nurodytą sąlygą. Čia filtras užtikrina, kad būtų apdorojamos tik nenulinės reikšmės, taip užkertant kelią vykdymo klaidoms dėl netikėtų nulinių reikšmių stebėjime.
map Pakeičia skleidžiamas reikšmes iš stebimo į naujas vertes, čia susiejant filtruotą iššūkių sąrašą ir metriką į veiksmą DataRetrieved. Šis metodas palaiko kodo funkcionalumą ir pašalina tarpinių kintamųjų deklaracijų poreikį.
provideMockActions Naudojama NgRx testavimui, provideMockActions sukuria netikrą veiksmų srautą, kuris imituoja veiksmų siuntimą vieneto bandymų metu. Tai padeda patikrinti elgsenos poveikį, nereikia siųsti realių veiksmų.
hot and cold Jasmine-Marbles, karštas ir šaltas sukuria stebimus bandomuosius srautus. Karštieji srautai rodo realaus laiko reikšmes, o šalti srautai – uždelstas arba buferines vertes, todėl galima tiksliai, laiku pagrįsti stebimų sekų testavimą.
toPromise Konvertuoja stebimą į pažadą, naudinga suderinamumui, kai pirmenybė teikiama asinchronizavimui / laukti. Šiame pavyzdyje tai leidžia Observables naudoti su asinchronine sintaksė šiuolaikiniam, skaitomam kodui, ypač senuose projektuose, pritaikytuose prie naujesnių asinchroninių struktūrų.

RxJS ir TypeScript suderinamumo supratimas senose Angular programose

Aukščiau pateikti scenarijai yra skirti konkrečiai dažnai pasitaiko senuose Angular projektuose, kai naudojamas RxJS: „šis“ kontekstas, kurio tipas „...“, nepriskiriamas metodo „šio“ tipui. Ši klaida paprastai įvyksta, kai funkcijos, kurios yra sinchroninės arba turi neapibrėžtą kontekstą, perkeliamos į asinchroninius metodus, todėl „TypeScript“ pažymi neatitikimą. Norėdami tai išspręsti, naudojame NgRx funkcija, kuri valdo asinchroninę logiką, stebėdama programos būsenos pokyčius ir vykdydama šalutinį poveikį, reaguodama į konkrečius veiksmus. Pirmajame pavyzdyje esantis NgRx efektas klausosi veiksmą, signalizuojant, kad organizacijos duomenys buvo atnaujinti, ir tada imama atitinkamų iššūkių sąrašų ir metrikos duomenų iš Observables.

Pagrindinė šios klaidos sprendimo dalis yra tinkamas stebėjimų tvarkymas ir tik būtinų duomenų apdorojimas. Tam, naudojamas operatorius RxJS, leidžiantis paimti naujausias reikšmes iš kelių stebimų objektų. Naudojant combinLatest, efektas gali stebėti iššūkių sąrašo ir metrikos duomenų srautų pokyčius, suaktyvindamas poveikį tik tada, kai šios reikšmės atnaujinamos. Tai padeda sinchronizuoti duomenis ir sumažinti nepageidaujamą šalutinį poveikį. Mes taip pat naudojame operatorius neįtrauktų nulinių verčių šiuose srautuose, užtikrinant, kad kitam operatoriui būtų perduoti tik galiojantys duomenys, o tai būtina programoms, kuriose gali būti duomenų neatitikimų.

Išfiltravus atitinkamus duomenis, operatorius susieja šias reikšmes į naują stebimą, šiuo atveju suaktyvina naują veiksmą, . Šiame kontekste „SwitchMap“ yra labai svarbus, nes jis atšaukia bet kokias ankstesnes duomenų srautų prenumeratas, kai tik atsiranda nauja emisija, užtikrindama, kad „Oservable“ išlaikytų tik naujausias vertes, išvengiant atminties nutekėjimo ir nenumatytų veiksmų dinaminėse programose. Ši RxJS operatorių grandinė ne tik užtikrina, kad mūsų duomenų tvarkymas būtų efektyvus, bet ir išlaiko kodą modulinį, nes kiekvienas transformacijos žingsnis yra aiškiai apibrėžtas. Kodas palaiko skaitomumą ir patikimumą, o tai būtina norint išlaikyti senas kodų bazes.

Alternatyviame pavyzdyje asinchronizavimo / laukimo sintaksė taikoma stebimam konvejeriui konvertuojant duomenų srautus į pažadus su . Šis metodas padeda kūrėjams tvarkyti asinchroninius duomenų srautus naudojant asinchronines funkcijas, pagerina skaitymo galimybes ir suteikia daugiau lankstumo tvarkant klaidas. Be to, atliekant vienetų testavimą su Jasmine/Karma, imitaciniai veiksmai sukuriami naudojant NgRx veiksmams imituoti ir ir šalta stebimi duomenys naudojami imituojant realaus laiko ir buferinių duomenų srautus. Šios testavimo priemonės yra labai svarbios norint patikrinti efektų veikimą ir užtikrinti, kad mūsų kodas tiksliai ir nuspėjamai apdorotų asinchroninius įvykius įvairiose aplinkose. Dėl šių įrankių šis sprendimas yra tvirtas, efektyvus ir puikiai tinka sudėtingam asinchroniniam būsenos valdymui kampinėse programose.

„Šio“ konteksto klaidų sprendimas pasenusiose Angular naudojant RxJS

Naudoja „TypeScript“ su „RxJS in Angular“, kad tvarkytų stebimą grandinės sujungimą naudojant modulinius ir optimizuotus sprendimus

import { Injectable } from '@angular/core';
import { Actions, ofType, createEffect } from '@ngrx/effects';
import { Observable, combineLatest, of } from 'rxjs';
import { switchMap, map, filter } from 'rxjs/operators';
import * as orgActions from './actions/orgActions';
import * as dataActions from './actions/dataActions';
@Injectable()
export class OrgEffects {
  constructor(private actions$: Actions,
              private dataChallenge: DataChallengeService,
              private dataMetric: DataMetricService) {}
  orgChangedSuccess$ = createEffect(() =>
    this.actions$.pipe(
      ofType(orgActions.UPDATE_ORG_SUCCESS),
      switchMap((org) => combineLatest([
        this.dataChallenge.challengeList$.pipe(filter(val => val !== null)),
        this.dataMetric.metrics$.pipe(filter(val => val !== null))
      ])
      .pipe(
        map(([challengeList, metrics]) =>
          new dataActions.DataRetrieved({ challengeList, metrics })
        )
      )
    ))
  );
}

Alternatyvus metodas, naudojant async/wait sintaksę Angular su RxJS

Įdiegia asinchronizavimą / laukimą su TypeScript Observables in Angular, kad išspręstų „šio“ privalomo konteksto problemas

import { Injectable } from '@angular/core';
import { Actions, ofType, createEffect } from '@ngrx/effects';
import { Observable, combineLatest, from } from 'rxjs';
import { switchMap, map, filter } from 'rxjs/operators';
import * as orgActions from './actions/orgActions';
import * as dataActions from './actions/dataActions';
@Injectable()
export class OrgEffects {
  constructor(private actions$: Actions,
              private dataChallenge: DataChallengeService,
              private dataMetric: DataMetricService) {}
  orgChangedSuccess$ = createEffect(() =>
    this.actions$.pipe(
      ofType(orgActions.UPDATE_ORG_SUCCESS),
      switchMap(async (org) => {
        const challengeList = await from(this.dataChallenge.challengeList$).pipe(filter(val => val !== null)).toPromise();
        const metrics = await from(this.dataMetric.metrics$).pipe(filter(val => val !== null)).toPromise();
        return new dataActions.DataRetrieved({ challengeList, metrics });
      })
    )
  );
}

Abiejų metodų vienetiniai testai naudojant jazminą / karmą kampu

„Jasmine“ ir „Karma“ bandymo atvejai, skirti patvirtinti stebimus tvarkymo ir asinchronizavimo metodus „Angular“ naudojant „TypeScript“

import { TestBed } from '@angular/core/testing';
import { provideMockActions } from '@ngrx/effects/testing';
import { cold, hot } from 'jasmine-marbles';
import { Observable } from 'rxjs';
import { OrgEffects } from './org.effects';
import * as orgActions from './actions/orgActions';
import * as dataActions from './actions/dataActions';
describe('OrgEffects', () => {
  let actions$: Observable<any>;
  let effects: OrgEffects;
  beforeEach(() => {
    TestBed.configureTestingModule({
      providers: [
        OrgEffects,
        provideMockActions(() => actions$)
      ]
    });
    effects = TestBed.inject(OrgEffects);
  });
  it('should dispatch DataRetrieved action when UPDATE_ORG_SUCCESS is triggered', () => {
    const action = orgActions.UPDATE_ORG_SUCCESS();
    const outcome = new dataActions.DataRetrieved({ challengeList: [], metrics: [] });
    actions$ = hot('-a', { a: action });
    const expected = cold('-b', { b: outcome });
    expect(effects.orgChangedSuccess$).toBeObservable(expected);
  });
});

Išplėstinė „TypeScript“ konteksto klaidų tvarkymo naudojant RxJS metodus Angular

Kalbant apie senus Angular projektus, RxJS Observables konteksto valdymas gali būti sudėtingas, ypač sudėtingų efektų ir asinchroninio duomenų tvarkymo atveju. Ši problema išryškėja dirbant su „TypeScript“, nes griežtas spausdinimas gali sukelti klaidų, jei kontekstas nėra tinkamai išsaugotas funkcijų skambučiuose. Vienas iš būdų išspręsti šias klaidas yra naudoti Angular's operatoriui arba naudojant , kurios nekuria savo 'tai' kontekste. Rodyklės funkcijos RxJS kode padeda užtikrinti, kad „tai“ teisingai nurodytų klasės egzempliorių, o ne funkcijos apimtį, taip sumažinant dažnas klaidas ir daront kodą nuspėjamesnį.

Kitas būdas apima naudojimą perduodant funkcijas kaip argumentus RxJS konvejeryje. Nors dažnai siejamas su „JavaScript“, tai gali būti galingas įrankis tvarkant asinchroninius duomenis „TypeScript“, užtikrinant, kad būtų išsaugota teisinga „šios“ nuoroda. Be to, kai atvaizduojate duomenis iš kelių srautų, ir forkJoin gali būti naudojamas stebimiems dalykams sinchronizuoti, ypač kai vienas stebimasis remiasi kito siunčiamais duomenimis. , priešingai nei „combinLatest“, laukia, kol visi šaltinio stebimi objektai bus baigti, prieš išskirdami vertes, todėl tai lengviau nuspėjama tais atvejais, kai kiekvienas stebimasis skleidžia tik vieną kartą.

Kūrėjai taip pat turėtų apsvarstyti galimybę naudoti siekiant supaprastinti derinimą, pvz., TypeScript Hero arba Angular Language Service. Šie plėtiniai padeda naršyti kodą ir teikti su kontekstu susijusius pasiūlymus, kurie yra neįkainojami atkuriant senesnes programas su sudėtingais RxJS diegimais. Plėtiniai, tokie kaip ESLint ir TSLint, taip pat padeda užtikrinti kodavimo standartų vykdymą, žymėti klaidas realiuoju laiku ir nukreipti pataisymus, o tai naudinga sprendžiant „šio“ konteksto klaidas arba nesuderinamų tipų priskyrimus. Kartu dėl šių metodų ir įrankių kodo priežiūra senose Angular programose žymiai sklandesnė ir sumažinamos dažniausios „TypeScript“ problemos.

  1. Kas sukelia „TypeScript“ konteksto klaidas?
  2. Šios klaidos dažnai atsiranda, kai kontekstas klasės metodu nesutampa su to, ko tikisi „TypeScript“. Naudojant RxJS padeda to išvengti, nes užtikrina, kad „tai“ išsaugoma numatyta nuoroda.
  3. Kaip gali padėti valdyti asinchroninius duomenis?
  4. padeda atšaukti ankstesnius stebimojo išleidimus, kai atsiranda naujas, todėl jis idealiai tinka tvarkyti asinchroninius duomenis, kurie dažnai atnaujinami, pvz., HTTP užklausas.
  5. Kodėl taip išspręsti kai kurias „šio“ konteksto klaidas?
  6. visam laikui nustato funkcijos kontekstas, padedantis išvengti konteksto neatitikimų, ypač perduodant klasės metodus kaip atgalinius iškvietimus.
  7. Koks skirtumas tarp ir RxJS?
  8. skleidžia, kai bet koks šaltinis Stebimasis skleidžia, o laukia, kol bus baigti visi stebimi šaltiniai, prieš išskirdami, todėl tinka vienkartiniams išmetimams.
  9. Gali pagerinti „TypeScript“ klaidų derinimą?
  10. Taip, tokie plėtiniai kaip TypeScript Hero ir Angular Language Service teikia atsiliepimus ir pasiūlymus realiuoju laiku, padeda veiksmingiau išspręsti konteksto ir spausdinimo klaidas.

Norint išspręsti „TypeScript“ konteksto klaidas dirbant su „RxJS Observables“, reikia kruopštaus požiūrio. Naudojant tokius operatorius kaip ir tokie įrankiai plėtiniai gali padėti lengviau valdyti šias problemas, ypač senesniuose Angular projektuose.

Šių strategijų ir įrankių priežiūra užtikrina, kad jūsų programa laikui bėgant išliks funkcionali ir efektyvesnė. Taikant nuoseklų požiūrį, konteksto ir asinchroninių duomenų tvarkymas naudojant „TypeScript“ taps racionalesnis ir padės užtikrinti, kad jūsų projektai būtų apsaugoti ateityje.

  1. Suteikia išsamų supratimą apie „TypeScript“ konteksto klaidų tvarkymą naudojant „Angular“ ir „RxJS“. Prieikite prie jo čia: RxJS oficiali dokumentacija
  2. Tyrinėja geriausią NgRx efektų, „TypeScript“ ir stebimų objektų naudojimo sudėtingose ​​programose praktiką. Patikrinkite išteklius adresu: NgRx efektų dokumentacija
  3. Siūlomos papildomos gairės dėl VS kodo plėtinių, naudingų Angular projektams, ypač „TypeScript“ klaidų valdymui. Daugiau žiūrėkite: „Visual Studio“ kodo plėtinių prekyvietė