惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

MyScale Blog
MyScale Blog
OSCHINA 社区最新新闻
OSCHINA 社区最新新闻
Google DeepMind News
Google DeepMind News
C
Cisco Blogs
量子位
WordPress大学
WordPress大学
C
CXSECURITY Database RSS Feed - CXSecurity.com
The Hacker News
The Hacker News
C
Comments on: Blog
Blog — PlanetScale
Blog — PlanetScale
PCI Perspectives
PCI Perspectives
Martin Fowler
Martin Fowler
云风的 BLOG
云风的 BLOG
博客园 - 司徒正美
D
DataBreaches.Net
T
The Exploit Database - CXSecurity.com
有赞技术团队
有赞技术团队
Hugging Face - Blog
Hugging Face - Blog
Simon Willison's Weblog
Simon Willison's Weblog
Stack Overflow Blog
Stack Overflow Blog
月光博客
月光博客
T
Troy Hunt's Blog
L
Lohrmann on Cybersecurity
L
LangChain Blog
Security Latest
Security Latest
A
Arctic Wolf
博客园 - Franky
cs.AI updates on arXiv.org
cs.AI updates on arXiv.org
C
Check Point Blog
V
Vulnerabilities – Threatpost
博客园 - 聂微东
SecWiki News
SecWiki News
H
Hackread – Cybersecurity News, Data Breaches, AI and More
I
Intezer
腾讯CDC
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
N
News and Events Feed by Topic
E
Exploit-DB.com RSS Feed
Recent Commits to openclaw:main
Recent Commits to openclaw:main
Engineering at Meta
Engineering at Meta
Microsoft Security Blog
Microsoft Security Blog
Google DeepMind News
Google DeepMind News
Spread Privacy
Spread Privacy
Recorded Future
Recorded Future
C
CERT Recently Published Vulnerability Notes
Last Week in AI
Last Week in AI
大猫的无限游戏
大猫的无限游戏
V
Visual Studio Blog
钛媒体:引领未来商业与生活新知
钛媒体:引领未来商业与生活新知
小众软件
小众软件

DEV Community

Google I/O 2026 Wasn’t About Features — It Was About AI Becoming the Developer Environment Building an AI Vedic Astrology App in 25 Days — What Actually Worked (and What Didn't) Hermes Agent Has Four Memories — And That's Why It Doesn't Forget You Pressure Isn't Killing You -Your Relationship With It Is 🐳 How to Run Any Project in Docker: A Complete Guide AccessLens — a blind person's lanyard, powered by Gemma 4 on-device Glyph v0.2: the release is the joinery How I Built a Blazingly Fast, Privacy-First Batch Image Converter in the Browser Using OPFS and Web Workers FCoP 3.0: Why AI Agents Need a Track, Not a Brake Fibonacci: Quiz app which anyone can make revenue by viewing ads to the quiz contestants. The Subconscious Powered by Edge AI GPU Utilization Is Becoming the New Cloud Waste Crisis Cómo solucionar `docker run` con exit code 1 en Raspberry Pi JWT is a scam and your app doesn't need it 7 Agent Skill Packs That Actually Make AI Coders Better More Control, More Cost: Why Commanding AI Isn't Delegation SecureScan Synthadoc: We Built an AI Judge for Our AI Wiki Compiler - Here's What We Learned Cómo solucionar el error de permiso al ejecutar `pip.exe` en entorno virtual (Python 3.10 en Windows) Postgres-grade Serializable at 20k+ ops/s — on a laptop. Don’t try this at home. Pure Core, Imperative Shell in Rust with Stillwater Lean 4 for Programmers: Building a Todo List with Proof Trustless Bug Bounty Releases with a PoW-Gated DLC Oracle Building Autonomous DevOps Agents with MCP and LangChain Multimodal Gemma 4 Visual Regression & Patch Agent Git Time Machine — How Version Control Can Save Your Project My Dad Got an Electricity Bill He Couldn't Understand. Google I/O 2026 Just Made That Problem Solvable. My Dad Got an Electricity Bill He Couldn't Understand. Google I/O 2026 Just Made That Problem Solvable. Read Replicas Lie About Consistency. 4 Sync Modes Behind the Lie. Reviving My Coding Project with GitHub Copilot I Tried Gemini 3.5 Flash After Google I/O 2026 - Here is What I Found :)) Zero-Cost AI in VS Code Blueprints Might Be More Important Than Frameworks AI CareCompanion - Offline Health Assistant Long-Context Models Killed RAG. Except for the 6 Cases Where They Made It Worse. I Built a Neural Network Engine in C# That Runs in Your Browser - No ONNX Runtime, No JavaScript Bridge, No Native Binaries An In-Depth Overview of the Apache Iceberg 1.11.0 Release Your Agent Just Called the Same Tool 47 Times. Here's the 20-Line Detector. How I Built a Multi-System Astrology Bot in Python (And What Meta Banned Me For) Gemma 4 Has Four Variants. Here's How to Pick the Right One Before You Write a Single Line of Code. Log Level Strategies: Balancing Observability and Cost Why WebMCP Is the Most Important Thing Google Announced at I/O 2026 (And Nobody's Talking About It) Making LLM Calls Reliable: Retry, Semaphore, Cache, and Batch Google's 2x Energy Efficiency Claim Is Real — But Here's What They're Not Measuring What's actually going on with CORS, under the hood Language-Agnostic Code Generation: The Driver Plugin Model Why We Rewrote Our Python CLI in Go (and What We Gained) I added up everything Google gives developers for free after I/O 2026. It's kind of absurd The Dawn of Smarter Apps: My Take on Google I/O 2026 AI Announcements Why AI Agents Like Hermes Need a Semantic Execution Layer for the Physical World Why We Built TestSmith: The Test Coverage Problem Nobody Talks About How to Convert Bank Statement PDFs to Excel: The Complete 2026 Guide Have You Ever Used a Website That Keeps Working After You Turn Off Your Internet? From idea to indexed: how I launched a SaaS in 60 days with Laravel + React Building a local-first AI tutor for my daughter (and 10–14 year-olds in Austrian schools) with Gemma 4 EC2 SSH Not Connecting? Here Are the 5 Things That Were Wrong (And How I Fixed Them) Best AI Tools for HVAC Contractors 2026 From Closed Internal Stack to Open-Source Ecosystem: I Finally Shipped Three Years of .NET Infrastructure Scrumpan is offlically LIVE!! Building a BMI Calculator CLI with TypeScript — Types, Functions, and Vitest From Building WordPress Websites to Node.js APIs: My Honest Full Stack Journey XiHan Snore Coach: Privacy-First On-Device MedTech Guardian powered by Gemma 4 Mobile Why AI Coding Agents Hallucinate and How to Fix It mcp-probe v1.4.0: Contract assertions for production MCP servers Google I/O 2026 Wasn't About One More Model. It Was About the Agent Stack. How I built 100+ crypto calculators in 6 languages on Astro The Dawn of Local Multi-Agent Architectures: Why Gemma 4 Changes Everything for Cloud Developers # I Told My AI to Simulate a Planet for 10,000 Years. It Built the Whole Thing Itself. 18/30 Days System Design Questions! From Hackathon Chaos to Clean CLI: Reviving My Daily Routine Analyser with GitHub Copilot Building a Home Lab with Proxmox and Terraform (for Kubernetes) PolicyAware vs Guardrails vs AI Gateways vs Model Routers: The Comparison Every AI Engineer Needs to Read Partner: An AI That Does Research While You Sleep Rugby Fundamentals as Software Concepts - Mapping the Pitch to your Code Base I Let Claude Code Run Unsupervised for 24 Hours. Here's What Happened. Why Zed Is Replacing VS Code in My AI-Augmented Workflow Build a scroll-driven WebGL hero in 30 lines Karpathy's LLM Wiki? No Code with Claude or Github Copilot! Why Platform Governance and Transparency Matter for Developers and Freelancers I built a Flutter CLI that generates Clean Architecture in seconds Using an LLM to automate a task that used to take hours by hand CyberArena – Interactive Cyber Security Simulation & Threat Analysis Platform Tile Extractor Mathematical Functions in CSS: clamp, min, max and How They Simplify Responsiveness Polyglot Persistence in Microservices: Let the Domain Choose the Database 190 Countries, Zero API Calls: Shipping Static Data in a Chrome Extension Your AI Writes Code Fast. Here’s How to Check It Before Shipping qwen2.5-coder is too slow for Claude Code on a Mac. Here's the fix. Building Automated Text-to-Video Pipelines with AI Can Gemini Become an Offline AI Tutor? Lessons from Building Educational AI OPRIX : From a simple messaging web app to a well structured and enhanced UI messaging web app Why React + TypeScript Nullability Slowly Becomes Exhausting Why AI Agents Need a Project Layer - Part 1 Stop Hand-Editing MCP Configs: A Zero-Dependency Go CLI What I Learned Working With Microsoft, SQUAD(GTCO), and Different Tech Communities 🧠 Hermes Agent Assistant — A Modular AI Agent System with Planner, Executor & Memory Spring Boot Auto-Configuration Source Code: Nail This Interview Question The Ultimate Guide to Free AI API Keys: 6 Platforms You Need to Know Why 91% of AI Agents Fail in Production (And What the 9% Do Differently) TryHackMe | Battery | WALKTHROUGH
Cómo solucionar \"Text content does not match server-rendered HTML\" en Next.js App Router
Erick Eduard · 2026-05-24 · via DEV Community

Erick Eduardo Ramos

Cómo solucionar "Text content does not match server-rendered HTML" en Next.js App Router

Este error ocurre cuando el HTML generado por el servidor (SSR/SSG) no coincide con el árbol de React generado durante la primera renderización del cliente. Durante la hidratación, React espera que el DOM inicial coincida exactamente con el que generó el servidor; cualquier diferencia provoca este error crítico.

Causa raíz

La causa más frecuente en aplicaciones modernas es el uso de APIs del navegador (window, localStorage, Date.now(), etc.) directamente en el renderizado, lo que provoca que el contenido sea diferente entre SSR (donde no están disponibles) y CSR (donde sí lo están). Otras causas comunes incluyen:

  • Uso de typeof window !== 'undefined' en el cuerpo del componente (no dentro de useEffect)
  • Librerías CSS-in-JS mal configuradas (especialmente styled-components sin @emotion/react o @emotion/server)
  • Extensiones del navegador (como Dark Reader o ad blockers) que modifican el DOM
  • Metaetiquetas de detección automática en iOS (format-detection)
  • Minificación automática por CDN (Cloudflare Auto Minify)

Pasos para solucionarlo

✅ Paso 1: Identifica la fuente del desajuste

Busca en tu código:

  • Uso de Date, Math.random(), localStorage, window, navigator, etc.
  • Lógica condicional basada en typeof window fuera de hooks de efecto
  • Componentes que renderizan contenido dinámico sin protección

🔍 Tip rápido: Usa console.log('server' if !window else 'client') en el componente sospechoso y revisa el HTML fuente vs. el DOM del navegador.


✅ Paso 2: Aplica la solución según el caso

Caso A: Contenido dinámico (ej. fecha/hora actual, ID aleatorio)

Usa suppressHydrationWarning en el elemento específico:

// ✅ CORRECTO: Solo suprime advertencia en el elemento problemático
<time suppressHydrationWarning>{new Date().toLocaleDateString()}</time>

Enter fullscreen mode Exit fullscreen mode

⚠️ Importante: No lo uses en contenedores grandes (como <div> que envuelve todo el contenido). Solo en elementos atómicos.


Caso B: Lógica condicional basada en entorno (ej. window o localStorage)

Mueve la lógica a useEffect + estado local:

import { useState, useEffect } from 'react';

export default function ClientOnlyComponent() {
  const [isClient, setIsClient] = useState(false);

  useEffect(() => {
    setIsClient(true);
  }, []);

  // ✅ SSR renderiza siempre el fallback; CSR renderiza lo real tras hidratación
  return (
    <div>
      {isClient ? (
        <div>
          <p>Contenido del navegador (ej. localStorage: {localStorage.getItem('theme')})</p>
        </div>
      ) : (
        <p>Cargando...</p> {/* Contenido idéntico en SSR y CSR inicial */}
      )}
    </div>
  );
}

Enter fullscreen mode Exit fullscreen mode


Caso C: Componente que usa APIs del navegador (ej. window.matchMedia)

Desactiva SSR para ese componente con next/dynamic:

// components/ClientComponent.tsx
export default function ClientComponent() {
  const [width, setWidth] = useState(0);

  useEffect(() => {
    const handleResize = () => setWidth(window.innerWidth);
    window.addEventListener('resize', handleResize);
    handleResize(); // Inicializar
    return () => window.removeEventListener('resize', handleResize);
  }, []);

  return <p>Ancho: {width}px</p>;
}

// page.tsx
import dynamic from 'next/dynamic';

const ClientComponent = dynamic(() => import('../components/ClientComponent'), {
  ssr: false, // ✅ Evita renderizado en servidor
});

export default function Page() {
  return (
    <main>
      <h1>Página principal</h1>
      <ClientComponent />
    </main>
  );
}

Enter fullscreen mode Exit fullscreen mode


Caso D: iOS convierte números/teléfonos en enlaces

Agrega metaetiqueta en <head> (en layout.tsx):

// app/layout.tsx
export default function RootLayout({ children }: { children: React.ReactNode }) {
  return (
    <html lang="es">
      <head>
        <meta
          name="format-detection"
          content="telephone=no, date=no, email=no, address=no"
        />
      </head>
      <body>{children}</body>
    </html>
  );
}

Enter fullscreen mode Exit fullscreen mode


Caso E: Librerías CSS-in-JS (ej. styled-components)

Configura correctamente para SSR:

npm install @emotion/react @emotion/server

Enter fullscreen mode Exit fullscreen mode

// app/layout.tsx
import { EmotionIntl } from '@emotion/react';
import { CacheProvider } from '@emotion/react';
import createCache from '@emotion/cache';

const cache = createCache({ key: 'css' });

export default function RootLayout({ children }: { children: React.ReactNode }) {
  return (
    <html>
      <head>
        <style
          data-emotion={`css ${cache.key}`}
          dangerouslySetInnerHTML={{ __html: '' }}
        />
      </head>
      <body>
        <CacheProvider value={cache}>{children}</CacheProvider>
      </body>
    </html>
  );
}

Enter fullscreen mode Exit fullscreen mode

🔥 Alternativa recomendada: Usa @emotion/react directamente o styled-components con su configuración oficial para Next.js.


✅ Paso 3: Verifica configuraciones de CDN

Si usas Cloudflare:

  • Desactiva Auto Minify (HTML)
  • Desactiva Rocket Loader
  • Asegúrate de que Brotli no esté corrompiendo el HTML

🛠️ Prueba rápida: Despliega en entorno local sin CDN. Si el error desaparece, el problema está en la infraestructura.


Pro-tip: Diagnóstico profesional

  1. Revisa el HTML fuente (Ctrl+U) y compáralo con el DOM del navegador (F12 > Elements).
  2. Usa console.log('Hydration check:', window ? 'client' : 'server') en el componente sospechoso.
  3. Activa React DevTools Profiler y busca componentes con hydrate en rojo.
  4. En producción, usa suppressHydrationWarning solo como último recurso — nunca como solución principal.

💡 Regla de oro: Si el contenido cambia entre SSR y CSR, debe estar protegido con useEffect o ssr: false. El usuario nunca debe ver un "flash" de contenido diferente durante hidratación.

Aplica estos pasos en orden y el error desaparecerá. Si persiste, revisa logs de tu CDN y extensiones del navegador (prueba en modo incógnito).


🚀 ¿Quieres más soluciones técnicas?

Si te sirvió esta ayuda, suscríbete para recibir los errores más comunes de la semana y cómo evitarlos.
👉 Suscríbete aquí