A complexidade ciclomática mede o número de caminhos de execução independentes em uma função ou módulo Python, ajudando a identificar código difícil de testar e manter. Desenvolvida por Thomas J. McCabe em 1976, essa métrica é calculada como o número de pontos de decisão (if, for, while, etc.) mais um, revelando riscos em fluxos ramificados excessivos.

Mas o que é Complexidade Ciclomática?

Complexidade ciclomática (CC) quantifica a densidade de caminhos lógicos em um grafo de controle de fluxo. Em Python, cada estrutura condicional ou de loop adiciona ramificações: um if simples eleva a CC para 2, enquanto and/or em condições compostas multiplica caminhos independentes. A fórmula básica é CC = E - N + 2P, onde E são arestas, N nós e P componentes conectados, mas ferramentas como radon ou flake8 computam isso automaticamente.

E por que diminuir a CC importa para Pythonistas?

Código Python com CC alta (>10) aumenta o risco de bugs ocultos e eleva o custo de testes unitários para cobertura total. Funções longas com if-elif-else encadeados violam o Zen of Python (“Flat is better than nested”), complicando debugging em IDEs como PyCharm. Em microservices ou APIs Flask/FastAPI, CC elevada impacta deploy em Docker, pois refatorações viram gargalos em CI/CD.

Calculando CC em Código Python

Considere este exemplo problemático:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
def processar_usuario(usuario, eh_admin=False, eh_pago=False):
    if not usuario:
        return None
    if eh_admin and eh_pago:
        return "acesso_total"
    elif eh_admin:
        return "acesso_admin"
    elif eh_pago:
        return "acesso_basico"
    else:
        if usuario.ativo:
            return "acesso_limitado"
        return "bloqueado"

Aqui, CC ≈ 6 devido a ramificações múltiplas. Use radon cc arquivo.py para medir:

processar_usuario: CC=6 (alto risco)

Interpretação e Limites Recomendados

Faixa de CCNível de RiscoAção Sugerida
1-5BaixoManter como está
6-10ModeradoRefatorar se possível
11-20AltoDividir função imediatamente
>20CríticoRefatoração urgente

Valores acima de 10 sinalizam antipadrões em Python, como god functions em Django views.

Estratégias de Redução em Python

  • Extraia funções puras: Divida em helpers como validar_usuario() e determinar_nivel_acesso().
  • Use polimorfismo: Substitua condicionais por classes com @dataclass ou Enum.
  • Guard clauses: Prefira if not condicao: return para early returns.
  • Strategy Pattern: Dicionários mapeiam condições a funções: handlers = {eh_admin: handler_admin}.
  • Ferramentas: Integre pylint ou mypy no pre-commit hook Git para alertas automáticos.

Exemplo refatorado (CC reduzida para 2):

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
def processar_usuario(usuario, eh_admin=False, eh_pago=False):
    if not usuario:
        return None
    return determinar_nivel_acesso(usuario.ativo, eh_admin, eh_pago)

def determinar_nivel_acesso(ativo, eh_admin, eh_pago):
    if eh_admin and eh_pago:
        return "acesso_total"
    handlers = {
        (eh_admin, eh_pago): "acesso_basico",
        eh_admin: "acesso_admin"
    }
    return handlers.get((eh_admin, eh_pago), "acesso_limitado" if ativo else "bloqueado")

Integração em Workflows Python

Em projetos com pytest, mire 100% branch coverage em funções CC<10. No VS Code, extensões como “Python Docstring Generator” ajudam na documentação pós-refatoração. Para equipes, thresholds no GitHub Actions bloqueiam merges com CC>15, alinhando com práticas DevOps em Kubernetes.

Monitore CC regularmente para código limpo e escalável em Python. Experimente radon no seu repo hoje e compartilhe comigo em @riverfount@bolha.us sua maior redução de CC!