Ramo e árvore amarrada em tikz

Ramo e árvore amarrada em tikz

Estou tentando criar uma árvore mostrando o processo de ramificação e vinculação para um problema de programação inteira no tikz. Consegui criar uma árvore com o primeiro nível exatamente da maneira que desejo, mas o látex não consegue compilar a árvore com o segundo nível e não tenho certeza. Meu código está muito mal otimizado, pois essa foi a única maneira de descobrir como fazer isso, e o código confuso significa que é difícil identificar onde estou errado.

Se alguém pudesse me mostrar uma maneira de limpar o código para que a maioria dos detalhes do nó esteja na pré-definição ou me mostrar onde estou errando na segunda árvore, isso seria ótimo.

Código para o primeiro nível:

\documentclass{article}
\usepackage[utf8]{inputenc}
\usepackage{tikz}
\usetikzlibrary{calc, shapes}

\begin{document}
\begin{figure}
\centering
\begin{tikzpicture}[
scale = 1.5, transform shape, thick,
tree node/.style = {align=center, inner sep=0pt, text centered, font = 
\scriptsize},
S/.style = {draw, circle, minimum size = 8mm, top color=white, bottom 
color=blue!20},
grow = down,  % alignment of characters
level 1/.style = {sibling distance=3cm},
level 2/.style = {sibling distance=4cm}, 
level 3/.style = {sibling distance=2cm}, 
level distance = 1.25cm]

\node [S, label={[font = \scriptsize]10:950}, label={[font = 
\scriptsize]170:1055.56}] {$S$}
    child{node [S, label={[font = \scriptsize]10:950}, label={[font = 
    \scriptsize]170:1000}] {$S_1$}edge from parent node[above left, font = 
    \scriptsize] {$x \leq 5$}}
    child{node [S, label={[font = \scriptsize]10:950}, label={[font = 
    \scriptsize]170:1033}]
    {$S_2$} edge from parent node[above right, font = \scriptsize] {$x \geq 
    6$}};
\end{tikzpicture}
\end{figure}
\end{document}

Código para o primeiro e segundo níveis, que não conseguem compilar:

\documentclass{article}
\usepackage[utf8]{inputenc}
\usepackage{tikz}
\usetikzlibrary{calc, shapes}

\begin{document}

\begin{figure}
\begin{tikzpicture}[
    scale = 1.5, transform shape, thick,
    tree node/.style = {align=center, inner sep=0pt, text centered, font = 
 \scriptsize},
    S/.style = {draw, circle, minimum size = 8mm, top color=white, bottom 
color=blue!20},
    grow = down,  % alignment of characters
    level 1/.style = {sibling distance=3cm},
    level 2/.style = {sibling distance=4cm}, 
    level 3/.style = {sibling distance=2cm}, 
    level distance = 1.25cm]
\node [S, label={[font = \scriptsize]10:950}, label={[font = 
\scriptsize]170:1055.56}] {$S$}
    child{node [S, label={[font = \scriptsize]10:950}, label={[font = 
    \scriptsize]170:1000}] {$S_1$} edge from parent node[above left, font = 
    \scriptsize] {$x \leq 5$}}
    child{node [S, label={[font = \scriptsize]10:950}, label={[font = 
    \scriptsize]170:1033}]
    {$S_2$} edge from parent node[above right, font = \scriptsize] {$x 
    \geq 6$}
        child{node [S, label={[font = \scriptsize]10:950}, label={[font 
             = \scriptsize]170:1033}]
            {$S_2,1$} edge from parent node[above right, font = \scriptsize] 
            {$x \leq 1$}
        child{node [S, label={[font = \scriptsize]10:950}, label={[font 
             = \scriptsize]170:1033}]
            {$S_2,2$} edge from parent node[above right, font = \scriptsize] 
            {$x \geq 2$}};
\end{tikzpicture}
\end{figure}
\end{document}

Desde já, obrigado!

Editar: um galho e uma árvore vinculada como a que estou tentando criar

Um galho e uma árvore amarrada como a que estou tentando criar

Responder1

Eu usaria um dos pacotes especializados para desenho de árvores. Mais especificamente, eu usaria forestou, na sua falta, tikz-qtree. Se você fizer isso, poderá especificar árvores de forma muito concisa e simples e garantir facilmente que elas sejam formatadas de forma consistente.

No caso de Forest, o pacote também fará grande parte do trabalho de layout para você, embora os rótulos não sejam tratados desta forma, portanto esta vantagem específica pode não ser muito benéfica para árvores deste tipo específico.

Aqui está um exemplo que define um branch and boundestilo para Forest. Quando aplicado a uma árvore, acontece o seguinte:

  • tree node, Se thicksão aplicados a todos os nós da árvore e o conteúdo é definido no modo matemático;

  • as bordas também são thick;

  • um pouco de espaço é adicionado à árvore para espaçar os galhos e os níveis;

  • depois que a árvore é analisada, o conteúdo de cada nó é dividido em cada dois pontos: <left label>:<node content>:<right label>:<edge label>;

    • A parte antes dos primeiros dois pontos se torna o rótulo esquerdo (em 170). A próxima parte é definida como o conteúdo do nó principal (com tree node, no modo matemático, etc.). A terceira parte é definida como o rótulo direito (em 10). A parte final se torna o rótulo do ramo: à esquerda do ponto médio para um ramo à esquerda e à direita para um ramo à direita.
  • o conteúdo dos rótulos das bordas pode ser definido no modo de texto ( text branch labels) ou no modo matemático ( maths branch labels);

    • prefixos e sufixos podem ser adicionados usando um dos

      • set branch labels={<left branch prefix>}{<left branch suffix>}{<right branch prefix>}{<right branch suffix>}

      • set maths branch labels={<left branch prefix>}{<left branch suffix>}{<right branch prefix>}{<right branch suffix>}

      • set text branch labels={<left branch prefix>}{<left branch suffix>}{<right branch prefix>}{<right branch suffix>}

  • uma linha horizontal é desenhada sob cada folha da árvore.

Então podemos especificar o que considero ser a sua segunda árvore, agora editada à luz dos comentários para variar os rótulos das bordas de acordo com o nível, com

\begin{forest}
  branch and bound,
  where level=1{
    set branch labels={x\leq}{}{x\geq}{},
  }{
    if level=2{
      set branch labels={}{\geq y}{}{\leq y},
    }{},
  }
  [1055.56:S:950
    [1000:S_1:950:5
    ]
    [1033:S_2:950:6
      [1033:{S_2,1}:950:1]
      [950:{S_2,2}:1033:2]
    ]
  ]
\end{forest}

que é obviamente um pouco mais conciso e produz

árvore <code>ramificação e limite</code>

Não sei quais são os critérios para esse tipo de árvore, então pode ser necessário alguns ajustes para funcionar bem. Por exemplo, talvez os rótulos das bordas nem sempre tenham o formato x\leqe x\req, ou talvez às vezes haja apenas um ou mais de dois filhos. No entanto, isto deve pelo menos ilustrar o poder potencial desta abordagem.

Código completo:

\documentclass[border=10pt]{standalone}
\usepackage{forest}
\tikzset{
  tree node/.style = {align=center, inner sep=0pt, font = \scriptsize},
  S/.style = {draw, circle, minimum size = 8mm, top color=white, bottom color=blue!20},
  tree node label/.style={font=\scriptsize},
}
\forestset{
  declare toks={left branch prefix}{},
  declare toks={right branch prefix}{},
  declare toks={left branch suffix}{},
  declare toks={right branch suffix}{},
  tree node left label/.style={
    label=170:#1,
  },
  tree node right label/.style={
    label=10:#1,
  },
  maths branch labels/.style={
    branch label/.style={
      if n=1{
        edge label={node [left, midway] {$\forestoption{left branch prefix}##1\forestoption{left branch suffix}$}},
      }{
        edge label={node [right, midway] {$\forestoption{right branch prefix}##1\forestoption{right branch suffix}$}},
      }
    },
  },
  text branch labels/.style={
    branch label/.style={
      if n=1{
        edge label={node [left, midway] {\foresteoption{left branch prefix}##1\forestoption{left branch suffix}}},
      }{
        edge label={node [right, midway] {\forestoption{right branch prefix}##1\forestoption{right branch suffix}}},
      }
    },
  },
  text branch labels,
  set branch labels/.style n args=4{%
    left branch prefix={#1},
    left branch suffix={#2},
    right branch prefix={#3},
    right branch suffix={#4},
  },
  set maths branch labels/.style n args=4{
    maths branch labels,
    set branch labels={#1}{#2}{#3}{#4},
  },
  set text branch labels/.style n args=4{
    text branch labels,
    set branch labels={#1}{#2}{#3}{#4},
  },
  branch and bound/.style={
    /tikz/every label/.append style=tree node label,
    maths branch labels,
    for tree={
      tree node,
      S,
      math content,
      s sep'+=20mm,
      l sep'+=5mm,
      thick,
      edge+={thick},
    },
    before typesetting nodes={
      for tree={
        split option={content}{:}{tree node left label,content,tree node right label,branch label},
      },
    },
    where n children=0{
      tikz+={
        \draw [thick]  ([yshift=-10pt, xshift=-2.5pt].south west) -- ([yshift=-10pt, xshift=2.5pt].south east);
      }
    }{},
  },
}
\begin{document}
\begin{forest}
  branch and bound,
  where level=1{
    set branch labels={x\leq}{}{x\geq}{},
  }{
    if level=2{
      set branch labels={}{\geq y}{}{\leq y},
    }{},
  }
  [1055.56:S:950
    [1000:S_1:950:5
    ]
    [1033:S_2:950:6
      [1033:{S_2,1}:950:1]
      [950:{S_2,2}:1033:2]
    ]
  ]
\end{forest}
\end{document}

Responder2

  • erros para o segundo código em seu mwe tem fonte ausente }apósedge from parent node
  • solução comtikz
  • seu código é limpo e reduzido definindo novos estilos para nós e rótulos de bordas
  • combinando as duas árvores da sua pergunta em uma (semelhante acfrem sua resposta) mnós nos tornamos:

\documentclass{article}
\usepackage[utf8]{inputenc}
\usepackage{tikz}
\usetikzlibrary{calc, shapes}

\begin{document}
    \begin{figure}
    \tikzset{thick,
         tree node/.style = {align=center, inner sep=0pt, font = \scriptsize},
every label/.append style = {font=\scriptsize},
                 S/.style = {draw, circle, minimum size = 11mm, inner sep=0pt,
                             top color=white, bottom color=blue!20},
               ENL/.style = {% edge node left
                             font=\footnotesize, left=1pt},
               ENR/.style = {% edge node right
                             font=\footnotesize, right=1pt},
                     grow = down,
         sibling distance = 2.8cm,
           level distance = 3cm
           }
    \newcommand\LB{% Lower bound
                    \tikz\draw[very thick] (-0.5,0) -- + (1,0);}

\centering
\begin{tikzpicture}
\node [S, label=10:950, label=170:1055.56] {$S$}
    child{node [S, label=10:950, label=170:1000, label=below:\LB] {$S_1$}
        edge from parent node[ENL] {$x \leq 5$}}
    child{node [S, label=10:950, label=170:1033] {$S_2$}
        child{node [S, label=10:950, label=170:1033, label=below:\LB] {$S_2,1$}
            edge from parent node[ENL] {$x \geq 6$}}
        child{node [S, label=10:950, label=170:1033, label=below:\LB] {$S_2,2$}
            edge from parent node[ENR] {$x \leq 1$}}
        edge from parent node[ENR] {$x \geq 6$}
            };
\end{tikzpicture}
\end{figure}
\end{document}

insira a descrição da imagem aqui

informação relacionada