{"id":1398,"date":"2022-10-10T15:03:43","date_gmt":"2022-10-10T18:03:43","guid":{"rendered":"http:\/\/www.galirows.com.br\/meublog\/programacao\/?p=1398"},"modified":"2022-10-10T15:06:12","modified_gmt":"2022-10-10T18:06:12","slug":"sobre-a-union-da-linguagem-c","status":"publish","type":"post","link":"http:\/\/www.galirows.com.br\/meublog\/programacao\/sobre-a-union-da-linguagem-c\/","title":{"rendered":"Sobre a union da linguagem C"},"content":{"rendered":"\n<p>A uni\u00e3o (<em>union<\/em>) na linguagem C \u00e9 um tipo especial de dado que permite armazenar diferentes tipos de dados em um mesmo espa\u00e7o da mem\u00f3ria. Embora a <em>union<\/em> possa ter v\u00e1rios campos em sua defini\u00e7\u00e3o, apenas um deles pode conter um valor por vez. Esse tipo de funcionamento permite a utiliza\u00e7\u00e3o de um mesmo espa\u00e7o de mem\u00f3ria para m\u00faltiplos prop\u00f3sitos. Trago alguns exemplos ao final deste post.<\/p>\n\n\n\n<p>Para saber o b\u00e1sico sobre as <em>unions<\/em> leia <a rel=\"noreferrer noopener\" href=\"https:\/\/www.tutorialspoint.com\/cprogramming\/c_unions.htm\" target=\"_blank\">https:\/\/www.tutorialspoint.com\/cprogramming\/c_unions.htm<\/a> (em ingl\u00eas, ou <a href=\"http:\/\/www.bosontreinamentos.com.br\/programacao-em-linguagem-c\/unions-em-linguagem-c\/\" target=\"_blank\" rel=\"noreferrer noopener\">http:\/\/www.bosontreinamentos.com.br\/programacao-em-linguagem-c\/unions-em-linguagem-c\/<\/a> (em portugu\u00eas).<\/p>\n\n\n\n<p>A sintaxe para defini\u00e7\u00e3o de uma <em>union <\/em>\u00e9 similar ao de uma <em>struct<\/em>, ent\u00e3o inicio comparando esses dois tipo de dados na linguagem C. No c\u00f3digo a seguir \u00e9 definida uma <em>union<\/em> e uma <em>struct <\/em>com os mesmo campos, sendo que o primeiro armazena um valor inteiro e o segundo armazena um valor <em>float<\/em>. O c\u00f3digo apenas mostrar\u00e1 o espa\u00e7o ocupado (em Bytes) na mem\u00f3ria pelos 2 tipos de dados b\u00e1sico (<em>int <\/em>e <em>float<\/em>) e pelo 2 tipo definidos no c\u00f3digo.<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code lang=\"c\" class=\"language-c line-numbers\">#include &lt;stdio.h>\n\nunion teste {\n  int a;\n  float b;\n};\n\nstruct teste2 {\n  int a;\n  float b;\n};\n\nint main(void) {\n  \/\/4 4 4 8\n  printf(\"%lu %lu %lu %lu\\n\", sizeof(int), sizeof(float), sizeof(union teste), sizeof(struct teste2));\n  \n  return 0;\n}<\/code><\/pre>\n\n\n\n<p>Como resultado da execu\u00e7\u00e3o do c\u00f3digo acima \u00e9 mostrado o resultado a seguir. Isso porque a fun\u00e7\u00e3o <em>sizeof(int)<\/em> e <em>sizeof(float) <\/em>retornam o valor 4 (considerando que o computador \u00e9 64 bits, o tipo <em>int <\/em>ocupa 4 Bytes, sen\u00e3o ocuparia 2 Bytes). Observe que a <em>struct teste2 <\/em>ocupa 8 Bytes, uma vez que ela armazena um dado <em>int <\/em>(que ocupa 4 Bytes) e um dado <em>float <\/em>(que tamb\u00e9m ocupa outros 4 Bytes), ou seja, 4 + 4 = 8. Por outro lado, o espa\u00e7o ocupado pela <em>union teste<\/em> ocupa apenas 4 Bytes, mesmo tendo dois campos.  <\/p>\n\n\n\n<pre class=\"wp-block-code\"><code class=\"\">4 4 4 8<\/code><\/pre>\n\n\n\n<p>A seguir altero o c\u00f3digo para definir duas vari\u00e1veis para cada um dos tipos definidos no c\u00f3digo, sendo a vari\u00e1vel <em>x<\/em> definida com base na <em>union<\/em> e a vari\u00e1vel <em>x2<\/em> baseada na <em>struct<\/em>. Em seguida s\u00e3o mostradas as posi\u00e7\u00f5es da mem\u00f3ria (utilizando ponteiros), onde os valores ser\u00e3o armazenados.<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code lang=\"c\" class=\"language-c line-numbers\">#include &lt;stdio.h&gt;\n\nunion teste {\n  int a;\n  float b;\n};\n\nstruct teste2 {\n  int a;\n  float b;\n};\n\nint main(void) {\n  union teste x;\n  struct teste2 x2;;\n    \n  printf(\"%p %p\\n\", &amp;x2.a, &amp;x2.b); \/\/0x7fff9bbadfe8 0x7fff9bbadfec\n  printf(\"%p %p\\n\", &amp;x.a, &amp;x.b); \/\/0x7fff9bbadff0 0x7fff9bbadff0\n  \n  return 0;\n}<\/code><\/pre>\n\n\n\n<p>Um poss\u00edvel resultado da execu\u00e7\u00e3o do c\u00f3digo acima \u00e9 mostrado a seguir. Digo poss\u00edvel porque as regi\u00f5es da mem\u00f3ria variam de acordo com a disponibilidade durante a execu\u00e7\u00e3o do programa. Perceba que o campo <em>&#8220;a&#8221;<\/em> da struct \u00e9 uma regi\u00e3o diferente do campo <em>&#8220;b<\/em>&#8221; da <em>struct<\/em> (conforme endere\u00e7os de mem\u00f3ria mostrados na primeira linha). J\u00e1 os campos &#8220;<em>a<\/em>&#8221; e &#8220;<em>b<\/em>&#8221; da <em>union<\/em> est\u00e3o na mesma \u00e1rea de mem\u00f3ria (conforme endere\u00e7os de mem\u00f3ria mostrados na segunda linha).<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code class=\"\">0x7fff9bbadfe8 0x7fff9bbadfec\n0x7fff9bbadff0 0x7fff9bbadff0<\/code><\/pre>\n\n\n\n<p>Uma vez que ambos os campos da union s\u00e3o armazenados no mesmo espa\u00e7o de mem\u00f3ria, somente um dos campos de ser utilizado efetivamente para armazenar dados. O c\u00f3digo a seguir mostra essa utiliza\u00e7\u00e3o.<\/p>\n\n\n\n<p>Na linha 12 \u00e9 atribu\u00eddo zero ao valor do campo &#8220;<em>a<\/em>&#8221; e em seguida \u00e9 mostrado o valor dos campos &#8220;<em>a<\/em>&#8221; e &#8220;<em>b<\/em>&#8220;. Como resultado ser\u00e1 mostrado os valores &#8220;<em>0 0.000000<\/em>&#8220;. Isso porque o valor zero inteiro e float possuem a mesma representa\u00e7\u00e3o bin\u00e1ria (lembre-se que todos os valores, n\u00e3o importa o seu tipo, s\u00e3o armazenados de forma bin\u00e1ria na mem\u00f3ria). Na linha 15 o valor atribu\u00eddo passa a ser o n\u00famero 2 e com isso j\u00e1 \u00e9 poss\u00edvel perceber uma diferen\u00e7a em rela\u00e7\u00e3o aos valores dos campos &#8220;a&#8221; e &#8220;b&#8221;.<\/p>\n\n\n\n<p>Possivelmente \u00e9 com a atribui\u00e7\u00e3o na linha 18 que trago a melhor interpreta\u00e7\u00e3o do que ocorre. O valor 2.0 \u00e9 atribu\u00eddo ao campo &#8220;<em>b<\/em>&#8221; e com isso \u00e9 mostrado o valor 1073741824 no print do campo &#8220;<em>a<\/em>&#8220;. Esse valor parece um n\u00famero qualquer, mas n\u00e3o \u00e9. O n\u00famero 2.0 \u00e9 armazenado como a sequ\u00eancia bin\u00e1ria <em>0100 0000 0000 0000 0000 0000 0000 0000<\/em> (existem v\u00e1rias maneiras para representa\u00e7\u00e3o um n\u00famero <em>float <\/em>de forma bin\u00e1ria, o padr\u00e3o aqui mostrado considera o da IEEE 754). Transformando essa sequ\u00eancia bin\u00e1ria em um valor inteiro tem-se 1&#215;2<sup>30<\/sup> (apenas o bit 30 contribui para o resultado). Por fim, 2<sup>30<\/sup> resulta em 1073741824, que \u00e9 o valor que aparece quando foi realizado o print do campo &#8220;<em>a<\/em>&#8220;. Logicamente que, embora o campo &#8220;<em>a<\/em>&#8221; tenha um valor, o valor somente \u00e9 realmente \u00fatil para o campo &#8220;<em>b<\/em>&#8220;, que foi o campo ao qual o valor foi atribu\u00eddo.<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code lang=\"c\" class=\"language-c line-numbers\">#include &lt;stdio.h&gt;\n#include &lt;math.h&gt;\n\nunion teste {\n  int a;\n  float b;\n};\n\nint main(void) {\n  union teste x;\n  \n  x.a = 0;\n  printf(\"%i %f\\n\", x.a, x.b); \/\/0 0.000000\n\n  x.a = 2;\n  printf(\"%i %f\\n\", x.a, x.b); \/\/2 0.000000\n\n  x.b = 2.0;\n  printf(\"%i %f\\n\", x.a, x.b); \/\/1073741824 2.000000\n\n  \/\/1073741824 =&gt; 0100 0000 0000 0000 0000 0000 0000 0000 = 2^30\n  printf(\"%i %i %i\", x.a, pow(2,30));\n  \n  return 0;\n}<\/code><\/pre>\n\n\n\n<p>Para exemplificar o uso da <em>union<\/em>, um uso poderia ser para armazenar valores da altura de uma pessoa, sendo que o sistema \u00e9 configurado pelo usu\u00e1rio para poder armazenar essa altura em cent\u00edmetros ou metros, ou seja, como um dado inteiro ou <em>float<\/em> (como na <em>union<\/em> especifica aqui nos c\u00f3digos). <\/p>\n\n\n\n<p>Outro uso, pode ser exemplificado com os campos da <em>union<\/em> sendo de tipos de dados definidos em <em>structs<\/em>, uma para armazenar dados de pessoa f\u00edsica  e outra para pessoa jur\u00eddica. Com isso, dependendo do tipo de pessoa cadastrada, uma <em>union<\/em> atuar\u00e1 com os dados dos campos.<\/p>\n","protected":false},"excerpt":{"rendered":"<p>A uni\u00e3o (union) na linguagem C \u00e9 um tipo especial de dado que permite armazenar diferentes tipos de dados em um mesmo espa\u00e7o da mem\u00f3ria. Embora a union possa ter v\u00e1rios campos em sua defini\u00e7\u00e3o, apenas um deles pode conter um valor por vez. Esse tipo de funcionamento permite a utiliza\u00e7\u00e3o de um mesmo espa\u00e7o [&hellip;]<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"_monsterinsights_skip_tracking":false,"_monsterinsights_sitenote_active":false,"_monsterinsights_sitenote_note":"","_monsterinsights_sitenote_category":0,"footnotes":""},"categories":[3,58],"tags":[39,97],"class_list":["post-1398","post","type-post","status-publish","format-standard","hentry","category-c","category-codigo-com-analise","tag-ponteiros","tag-union"],"aioseo_notices":[],"amp_enabled":true,"_links":{"self":[{"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/posts\/1398","targetHints":{"allow":["GET"]}}],"collection":[{"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/comments?post=1398"}],"version-history":[{"count":8,"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/posts\/1398\/revisions"}],"predecessor-version":[{"id":1406,"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/posts\/1398\/revisions\/1406"}],"wp:attachment":[{"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/media?parent=1398"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/categories?post=1398"},{"taxonomy":"post_tag","embeddable":true,"href":"http:\/\/www.galirows.com.br\/meublog\/programacao\/wp-json\/wp\/v2\/tags?post=1398"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}