[c] 구조와 노조의 차이점

a struct와 a 의 차이를 나타내는 좋은 예가 union있습니까? 기본적으로 struct멤버의 모든 메모리를 union사용하고 가장 큰 멤버 메모리 공간 을 사용 한다는 것을 알고 있습니다. 다른 OS 수준의 차이가 있습니까?



답변

공용체를 사용하면 요소가 모두 같은 위치에 저장되므로 요소 중 하나만 사용해야합니다. 여러 유형 중 하나 일 수있는 것을 저장하려는 경우 유용합니다. 반면에 구조체는 각 요소에 대해 별도의 메모리 위치를 가지며 모든 요소를 ​​한 번에 사용할 수 있습니다.

그들의 사용에 대한 구체적인 예를 들기 위해 나는 얼마 전에 Scheme 인터프리터에서 일하고 있었고 Scheme 데이터 유형을 본질적으로 C 데이터 유형에 오버레이했습니다. 여기에는 구조체에 값 유형을 나타내는 열거 형과 해당 값을 저장하는 공용체를 저장하는 작업이 포함되었습니다.

union foo {
  int a;   // can't use both a and b at once
  char b;
} foo;

struct bar {
  int a;   // can use both a and b simultaneously
  char b;
} bar;

union foo x;
x.a = 3; // OK
x.b = 'c'; // NO! this affects the value of x.a!

struct bar y;
y.a = 3; // OK
y.b = 'c'; // OK

편집 : xb를 ‘c’로 설정하는 것이 궁금하다면 xa의 값을로 바꾸면 기술적으로 정의되지 않습니다. 대부분의 현대 컴퓨터에서 char은 1 바이트이고 int는 4 바이트이므로 xb에 값 ‘c’를 주면 xa의 첫 번째 바이트에도 동일한 값이 부여됩니다.

union foo x;
x.a = 3;
x.b = 'c';
printf("%i, %i\n", x.a, x.b);

인쇄물

99, 99

두 값이 같은 이유는 무엇입니까? int 3의 마지막 3 바이트는 모두 0이므로 99로도 읽습니다. xa에 더 큰 숫자를 넣으면 항상 그렇지는 않습니다.

union foo x;
x.a = 387439;
x.b = 'c';
printf("%i, %i\n", x.a, x.b);

인쇄물

387427, 99

실제 메모리 값을 자세히 보려면 ​​16 진수로 값을 설정하고 인쇄 해 보겠습니다.

union foo x;
x.a = 0xDEADBEEF;
x.b = 0x22;
printf("%x, %x\n", x.a, x.b);

인쇄물

deadbe22, 22

0x22가 0xEF를 덮어 쓴 위치를 명확하게 볼 수 있습니다.

그러나

C에서는 int의 바이트 순서가 정의 되어 있지 않습니다. 이 프로그램은 내 Mac에서 0xEF를 0x22로 덮어 썼지 만 int를 구성하는 바이트 순서가 반대로되어 0xDE를 덮어 쓰는 다른 플랫폼이 있습니다. 따라서 프로그램을 작성할 때 이식 가능하지 않기 때문에 통합에서 특정 데이터를 덮어 쓰는 동작에 의존해서는 안됩니다.

바이트 순서에 대한 자세한 내용은 endianness를 확인하십시오 .


답변

짧은 대답은 다음과 같습니다. 구조체는 레코드 구조입니다. 구조체의 각 요소는 새로운 공간을 할당합니다. 그래서 같은 구조체

struct foobarbazquux_t {
    int foo;
    long bar;
    double baz;
    long double quux;
}

(sizeof(int)+sizeof(long)+sizeof(double)+sizeof(long double))각 인스턴스에 대해 최소 바이트를 메모리에 할당합니다. (아키텍처 정렬 제약 조건으로 인해 컴파일러가 구조체를 채울 수 있기 때문에 “적어도”)

반면에

union foobarbazquux_u {
    int foo;
    long bar;
    double baz;
    long double quux;
}

하나의 메모리 청크를 할당하고 네 개의 별명을 제공합니다. 따라서 sizeof(union foobarbazquux_u) ≥ max((sizeof(int),sizeof(long),sizeof(double),sizeof(long double))다시 정렬에 대한 추가 가능성이 있습니다.


답변

‘struct’와 ‘union’의 차이점을 보여주는 좋은 예가 있습니까?

가상의 통신 프로토콜

struct packetheader {
   int sourceaddress;
   int destaddress;
   int messagetype;
   union request {
       char fourcc[4];
       int requestnumber;
   };
};

이 가상의 프로토콜에서 “메시지 유형”에 따라 헤더의 다음 위치는 요청 번호이거나 4 자 코드이지만 둘다는 아닙니다. 요컨대, 공용체는 동일한 저장 영역 위치가 둘 이상의 데이터 유형을 나타내도록 허용합니다. 여기서 한 번에 하나의 데이터 유형 만 저장하려는 것이 보장됩니다.

유니온은 시스템 프로그래밍 언어로서 C의 유산에 기반을 둔 하위 수준의 세부 사항으로, “겹치는”저장 위치가 때때로 이런 식으로 사용됩니다. 한 번에 여러 유형 중 하나만 저장되는 데이터 구조가있는 경우 공용체를 사용하여 메모리를 절약 할 수 있습니다.

일반적으로 OS는 구조체와 공용체를 신경 쓰거나 알지 못합니다. 모두 단순히 메모리 블록입니다. 구조체는 여러 데이터 객체를 저장하는 메모리 블록으로, 객체가 겹치지 않습니다. 공용체는 여러 데이터 개체를 저장하지만 이들 중 가장 큰 데이터 만 저장할 수있는 메모리 블록이므로 한 번에 하나의 데이터 개체 만 저장할 수 있습니다.


답변

귀하의 질문에 당신이 이미 상태로 사이의 주요 차이점 unionstructunion동안 회원, 노동 조합의 sizeof 연산자는 하나라는 것을 서로 정도의 메모리를 오버레이 struct회원 (사이에서 선택 패딩) 각각 하나씩 배치됩니다. 또한 공용체는 모든 멤버를 포함 할 수있을만큼 충분히 크며 모든 멤버에 맞는 정렬을 갖습니다. 따라서 int2 바이트 주소에만 저장할 수 있고 너비는 2 바이트이고 long은 4 바이트 주소에만 저장할 수 있으며 4 바이트 길이 라고 가정 해 봅시다 . 다음 노조

union test {
    int a;
    long b;
}; 

가질 수있다 sizeof(4), 그리고 4. 노동 조합 모두의 정렬 요구 사항 및 구조체는 시작 부분에서 끝 부분에 패딩을 가지고 있지만, 수 없습니다. 구조체에 쓰면 쓴 멤버의 값만 변경됩니다. 공용체 멤버에게 쓰면 다른 모든 멤버의 값이 유효하지 않게됩니다. 이전에 작성한 적이 없으면 액세스 할 수 없습니다. 그렇지 않으면 동작이 정의되지 않습니다. GCC는 가장 최근에 쓰지 않았더라도 실제로 노조원으로부터 읽을 수있는 확장 기능을 제공합니다. 운영 체제의 경우 사용자 프로그램이 공용체 또는 구조에 기록하는지 여부는 중요하지 않습니다. 이것은 실제로 컴파일러의 문제입니다.

공용체와 구조체의 또 다른 중요한 속성은 포인터가 멤버의 유형을 가리킬 수 있다는 입니다. 따라서 다음이 유효합니다.

struct test {
    int a;
    double b;
} * some_test_pointer;

some_test_pointer는 int*또는을 가리킬 수 있습니다 double*. 이 유형의 주소를 캐스팅하는 경우 testint*, 그것의 첫 번째 멤버, 가리 킵니다 a사실. 노동 조합도 마찬가지입니다. 따라서 공용체는 항상 올바른 정렬을 갖기 때문에 공용체를 사용하여 일부 유형을 유효하게 만들 수 있습니다.

union a {
    int a;
    double b;
};

그 조합은 실제로 int와 double을 가리킬 수 있습니다.

union a * v = (union a*)some_int_pointer;
*some_int_pointer = 5;
v->a = 10;
return *some_int_pointer;    

C99 표준에 명시된 바와 같이 실제로 유효합니다.

객체는 다음 유형 중 하나를 갖는 lvalue 표현식으로 만 저장된 값에 액세스해야합니다.

  • 객체의 유효 유형과 호환되는 유형
  • 구성원 중 위에서 언급 한 유형 중 하나를 포함하는 집계 또는 공용체 유형

컴파일러는 v->a = 10;값에 영향을 줄 수 있으므로 를 최적화하지 않습니다 *some_int_pointer(그리고 10대신 함수가 반환 됩니다 5).


답변

A union는 몇 가지 시나리오에서 유용합니다.
union커널 용 장치 드라이버 작성과 같은 매우 낮은 수준의 조작을위한 도구 일 수 있습니다.

그 예로는 절개되어 float사용 번호 union(A)의 struct비트 필드와 함께 float. 에 번호를 저장하면 float나중에 float해당 부분의 특정 부분에 액세스 할 수 있습니다 struct. 이 예는 union데이터를보기 위해 다른 각도를 사용 하는 방법을 보여줍니다 .

#include <stdio.h>

union foo {
    struct float_guts {
        unsigned int fraction : 23;
        unsigned int exponent : 8;
        unsigned int sign     : 1;
    } fg;
    float f;
};

void print_float(float f) {
    union foo ff;
    ff.f = f;
    printf("%f: %d 0x%X 0x%X\n", f, ff.fg.sign, ff.fg.exponent, ff.fg.fraction);

}

int main(){
    print_float(0.15625);
    return 0;
}

위키 백과에 대한 단 정밀도 설명을 살펴보십시오 . 나는 거기에서 예제와 매직 넘버 0.15625를 사용했다.


union또한 여러 대안이있는 대수 데이터 유형을 구현하는 데 사용될 수 있습니다. O’Sullivan, Stewart 및 Goerzen의 “Real World Haskell”책에서 그 예를 찾았습니다. 차별 조합 섹션 에서 확인하십시오 .

건배!


답변

union “및 ” struct “는 C 언어의 구성 입니다. 하나 또는 다른 키워드를 사용하는 경우 다른 코드를 생성 하는 컴파일러 이기 때문에 이들 사이의 “OS 수준”차이를 말하는 것은 부적절 합니다.


답변

비 기술적으로 말하는 것은

가정 : 의자 = 메모리 블록, 사람 = 변수

구조 : 인원이 3 명인 경우에는 해당하는 크기의 의자에 앉을 수 있습니다.

노동 조합 : 3 명이 앉는 경우 의자 하나만 앉게되므로 앉을 때 모두 같은 의자를 사용해야합니다.

기술적으로 말하면 다음을 의미합니다.

아래에 언급 된 프로그램은 구조와 연합을 심층적으로 다룹니다.

struct MAIN_STRUCT
{
UINT64 bufferaddr;
union {
    UINT32 data;
    struct INNER_STRUCT{
        UINT16 length;
        UINT8 cso;
        UINT8 cmd;
           } flags;
     } data1;
};

총 MAIN_STRUCT 크기 = bufferaddr의 경우 sizeof (UINT64) + union의 경우 sizeof (UNIT32) + 패딩의 경우 32 비트 (프로세서 아키텍처에 따라 다름) = 128 비트. 구조를 위해 모든 멤버는 메모리 블록을 연속적으로 얻습니다.

Union은 최대 크기 멤버의 메모리 블록을 하나 얻습니다 (여기서는 32 비트). 유니언 내부에 하나 이상의 구조가 있습니다 (INNER_STRUCT) 그 멤버는 총 크기 32 비트 (16 + 8 + 8)의 메모리 블록을 얻습니다. 통합적으로 INNER_STRUCT (32 비트) 멤버 또는 data (32 비트)에 액세스 할 수 있습니다.