AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / coding / Perguntas / 79586063
Accepted
derstauner
derstauner
Asked: 2025-04-22 17:08:46 +0800 CST2025-04-22 17:08:46 +0800 CST 2025-04-22 17:08:46 +0800 CST

erro "a expressão foi alterada após ser verificada", apesar de ter a configuração correta

  • 772

Estou enfrentando o erro expression has changed after it was checked. Sei que esta verificação só é executada no modo de desenvolvimento (executa duas vezes) e que não terei a mensagem de erro no modo de produção. Mas quero saber por que isso acontece.

Eu tenho um serviço de carregador baseado em sinal como este:

import { Injectable, signal } from '@angular/core';

@Injectable({ providedIn: 'root' })
export class LoaderService {
  private _isLoading = signal<boolean>(false);

  get isLoading(): boolean {
    return this._isLoading();
  }

  show() {
    this._isLoading.update(() => true);
  }

  hide() {
    this._isLoading.update(() => false);
  }
}

Eu também tenho este interceptador:

import {
  HttpEvent,
  HttpHandler,
  HttpInterceptor,
  HttpRequest,
  HttpResponse,
} from '@angular/common/http';
import { Injectable } from '@angular/core';
import { Observable, catchError, map, throwError } from 'rxjs';
import { USE_LOADING_SPINNER } from '../constants/app.constants';
import { LoaderService } from '../../shared/services/loader.service';

@Injectable()
export class LoadingInterceptor implements HttpInterceptor {
  constructor(private loaderService: LoaderService) {}

  intercept(
    req: HttpRequest<any>,
    next: HttpHandler
  ): Observable<HttpEvent<any>> {
    if (req.context.get(USE_LOADING_SPINNER)) this.loaderService.show();

    return next.handle(req).pipe(
      map((event: HttpEvent<any>) => {
        if (event instanceof HttpResponse) this.loaderService.hide();
        return event;
      }),
      catchError((error) => {
        this.loaderService.hide();
        return throwError(() => error);
      })
    );
  }
}

O USE_LOADING_SPINNERé sempre definido como verdadeiro, o que significa que tenho que marcar apenas as solicitações em que não quero ter nenhum indicador de carregamento.

Como o LoaderServiceé baseado em sinal, registro um effectno construtor de um componente como este:

constructor() {
    effect(() => {
      this.loading = this.loaderService.isLoading;
    });
  }

Observe que a loadingvariável não é um sinal, é uma variável booleana pura, já que o isLoadingmétodo retorna um valor booleano, não um sinal.

E no modelo eu tenho isso em um componente de grade:

[loading]="loading"

Esta configuração está funcionando. No roteamento inicial, o indicador de carregamento está sendo exibido.

Agora adiciono um método de atualização ao componente como este:

refresh() {
    //this.loaderService.show();
    this.gridData$ = this.userService.getUsers();
    //.pipe(tap(() => this.loaderService.hide()));
  }

Mas se eu clicar em atualizar, aparece o erro no título do tópico e o indicador de carregamento não aparece. O interceptador é acionado, eu verifiquei isso.

Mas se eu descomentar as chamadas, loaderServiceele funciona como esperado.

Por que recebo esse erro?

angular
  • 2 2 respostas
  • 64 Views

2 respostas

  • Voted
  1. Andrei
    2025-04-22T18:10:14+08:002025-04-22T18:10:14+08:00

    Espero que você entenda o que esse erro significa no caso comum de "configuração errada".

    No seu caso, você provavelmente tem uma marcação

      [loading]="loading" // change detection checks values from top to bottom. 
      // loading value is checked here and the value is false
    
      // later in the template, or somewhere in children
      [data]="gridData$ | async" // during execution of THIS LINE http request fires, 
     // interceptor code is executed, loading is set to true, which obviously causes that error.
    

    para corrigir isso, você deve transformar seu código para executar a chamada http durante o tratamento de eventos em vez de durante a execução do modelo. por exemplo, via rxjs

    private refresh$ = Subject<void>();
    gridData$ = this.refresh$.pipe(
      startWith(undefined), 
      exhaustMap(() => this.userService.getUsers())
    );
    
    refresh() {
      this.refresh$.next(); // this way http call will happen during this line execution
    }
    
    • 1
  2. Best Answer
    Naren Murali
    2025-04-22T17:15:48+08:002025-04-22T17:15:48+08:00

    Acredito que o erro expression has changed after it was checkedpossa ser devido à atualização loadingapós a inicialização da visualização. Em vez disso, opte por uma atualização baseada em sinal, que pode não gerar esse erro.


    Acredito que seja devido ao uso indevido de getterand signal, em vez de executar o sinal no nível de serviço, retornar o sinal real.

    Quando você quiser fazer algo com o estado anterior, use update, caso contrário set é uma opção melhor.

    Seu sinal é privado e você só quer retornar uma versão somente leitura do sinal, então usamos asReadonly()para remover as operações graváveis set​​quando updateo componente usa esse sinal.

    import { Injectable, signal } from '@angular/core';
    
    @Injectable({ providedIn: 'root' })
    export class LoaderService {
      private _isLoading = signal<boolean>(false);
    
      get isLoading(): boolean {
        return this._isLoading.asReadonly(); // prevent modification of the signal
      }
    
      show() {
        this._isLoading.set(true);
      }
    
      hide() {
        this._isLoading.set(false);
      }
    }
    

    Agora, no nível do componente, em vez de , effecto que é exagero, basta atribuir a propriedade de serviço a uma propriedade de componente.

    // or
    // loading = this.loaderService.isLoading;
    
    constructor() {
      this.loading = this.loaderService.isLoading;
    }
    

    No nível do modelo, execute o sinal.

    [loading]="loading()"
    
    • -1

relate perguntas

  • pipe de data não funciona para elementos com conjunto de propriedades FormControlName

  • Posso colocar o serviço do componente reutilizável no módulo principal em Angular de acordo com o cenário mencionado

  • A caixa de diálogo Mat não fecha depois que a vinculação de dados mat-dialog-close é verdadeira

  • Não é possível renderizar o componente personalizado em app.component.html

  • Reutilização de component.html em mais de 2 arquivos .ts

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Reformatar números, inserindo separadores em posições fixas

    • 6 respostas
  • Marko Smith

    Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não?

    • 2 respostas
  • Marko Smith

    Problema com extensão desinstalada automaticamente do VScode (tema Material)

    • 2 respostas
  • Marko Smith

    Vue 3: Erro na criação "Identificador esperado, mas encontrado 'import'" [duplicado]

    • 1 respostas
  • Marko Smith

    Qual é o propósito de `enum class` com um tipo subjacente especificado, mas sem enumeradores?

    • 1 respostas
  • Marko Smith

    Como faço para corrigir um erro MODULE_NOT_FOUND para um módulo que não importei manualmente?

    • 6 respostas
  • Marko Smith

    `(expression, lvalue) = rvalue` é uma atribuição válida em C ou C++? Por que alguns compiladores aceitam/rejeitam isso?

    • 3 respostas
  • Marko Smith

    Um programa vazio que não faz nada em C++ precisa de um heap de 204 KB, mas não em C

    • 1 respostas
  • Marko Smith

    PowerBI atualmente quebrado com BigQuery: problema de driver Simba com atualização do Windows

    • 2 respostas
  • Marko Smith

    AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos

    • 1 respostas
  • Martin Hope
    Fantastic Mr Fox Somente o tipo copiável não é aceito na implementação std::vector do MSVC 2025-04-23 06:40:49 +0800 CST
  • Martin Hope
    Howard Hinnant Encontre o próximo dia da semana usando o cronógrafo 2025-04-21 08:30:25 +0800 CST
  • Martin Hope
    Fedor O inicializador de membro do construtor pode incluir a inicialização de outro membro? 2025-04-15 01:01:44 +0800 CST
  • Martin Hope
    Petr Filipský Por que os conceitos do C++20 causam erros de restrição cíclica, enquanto o SFINAE antigo não? 2025-03-23 21:39:40 +0800 CST
  • Martin Hope
    Catskul O C++20 mudou para permitir a conversão de `type(&)[N]` de matriz de limites conhecidos para `type(&)[]` de matriz de limites desconhecidos? 2025-03-04 06:57:53 +0800 CST
  • Martin Hope
    Stefan Pochmann Como/por que {2,3,10} e {x,3,10} com x=2 são ordenados de forma diferente? 2025-01-13 23:24:07 +0800 CST
  • Martin Hope
    Chad Feller O ponto e vírgula agora é opcional em condicionais bash com [[ .. ]] na versão 5.2? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench Por que um traço duplo (--) faz com que esta cláusula MariaDB seja avaliada como verdadeira? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng Por que `dict(id=1, **{'id': 2})` às vezes gera `KeyError: 'id'` em vez de um TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob: MobileAds.initialize() - "java.lang.Integer não pode ser convertido em java.lang.String" para alguns dispositivos 2024-03-20 03:12:31 +0800 CST

Hot tag

python javascript c++ c# java typescript sql reactjs html

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve