C에서 void 포인터에 대한 포인터 산술
특정 유형 (예 int
: char
,, float
..)에 대한 포인터 가 증가하면 해당 데이터 유형의 크기만큼 값이 증가합니다. 경우 void
크기의 데이터에 대한 포인터하는 포인트가 x
증가 점에, 어떻게 나올까요 x
앞서 바이트? 컴파일러 x
는 포인터의 가치에 더하는 것을 어떻게 알고 있습니까?
최종 결론 : a에 대한 산술 void*
은 C와 C ++에서 모두 불법 입니다.
GCC는이를 확장으로 허용합니다. 산술 void
및 함수 포인터를 참조하십시오 (이 섹션은 설명서의 "C 확장"장의 일부입니다). Clang과 ICC void*
는 GCC와의 호환성을 위해 산술을 허용 할 것 입니다. 다른 컴파일러 (MSVC와 같은)는에 대한 산술을 허용하지 void*
않으며 GCC는 -pedantic-errors
플래그가 지정되거나 플래그가 지정된 경우이를 허용하지 않습니다 -Werror-pointer-arith
(이 플래그는 코드베이스도 MSVC로 컴파일해야하는 경우에 유용합니다).
C 표준은 말한다
따옴표는 n1256 초안에서 가져옵니다.
추가 작업 상태에 대한 표준 설명 :
6.5.6-2 : 또한 두 피연산자 모두 산술 유형을 갖거나 한 피연산자는 객체 유형을 가리키는 포인터이고 다른 피연산자는 정수 유형이어야합니다.
따라서 여기서 질문 void*
은 "객체 유형"에 대한 포인터인지 또는 "객체 유형"인지 void
에 대한 것입니다. "객체 유형"의 정의는 다음과 같습니다.
6.2.5.1 : 유형은 객체 유형 ( 객체 를 완전히 설명하는 유형), 함수 유형 (함수를 설명하는 유형) 및 불완전한 유형 (객체를 설명하지만 크기를 결정하는 데 필요한 정보는없는 유형)으로 분할됩니다 .
그리고 표준은 다음 void
과 같이 정의 됩니다.
6.2.5-19 :
void
유형은 빈 값 집합으로 구성됩니다. 완료 할 수없는 불완전한 유형입니다.
void
불완전한 유형 이므로 객체 유형이 아닙니다. 따라서 추가 연산에 유효한 피연산자가 아닙니다.
따라서 포인터에서 포인터 산술을 수행 할 수 없습니다 void
.
노트
원래 void*
C 표준의 다음 섹션으로 인해 산술이 허용되는 것으로 생각 되었습니다.
6.2.5-27 : void에 대한 포인터는 문자 유형에 대한 포인터 와 동일한 표현 및 정렬 요구 사항을 가져야 합니다 .
하나,
동일한 표현 및 정렬 요구 사항은 함수에 대한 인수, 함수의 값 반환 및 공용체 멤버와의 호환성을 의미합니다.
이 수단 그래서 printf("%s", x)
여부 같은 의미를 갖고 x
유형이 char*
나 void*
있지만, 당신은 산술 할 수있는 것을 의미하지 않는다 void*
.
편집자 주 : 이 답변은 최종 결론을 반영하기 위해 편집되었습니다.
포인터에는 포인터 산술이 허용되지 않습니다 void*
.
포인터를 x 바이트 앞으로 앞으로 증가시키는 char 포인터로 캐스트하십시오.
void *
정확히 이런 이유로 타입에 대한 포인터 산술을 할 수 없습니다 !
C 표준은 허용하지 않습니다 무효 포인터 연산을. 그러나, GNU C는 의 크기를 고려하여 허용 무효 IS를 1
.
C11 표준 §6.2.5
단락-19
void
유형 값의 빈 세트를 포함하고; 그것은이다 불완전한 개체 유형 을 완료 할 수 없습니다.
다음 프로그램은 GCC 컴파일러에서 정상적으로 작동합니다.
#include<stdio.h>
int main()
{
int arr[2] = {1, 2};
void *ptr = &arr;
ptr = ptr + sizeof(int);
printf("%d\n", *(int *)ptr);
return 0;
}
다른 컴파일러가 오류를 생성 할 수 있습니다.
포인터 산술을 수행하기 전에 다른 유형의 포인터로 캐스트해야합니다.
무효 포인터는 메모리 청크를 가리킬 수 있습니다. 따라서 컴파일러는 void 포인터에서 포인터 산술을 시도 할 때 증가 / 감소 할 바이트 수를 알지 못합니다. 따라서 void 포인터는 포인터 산술에 관여하기 전에 먼저 알려진 형식으로 타입 캐스트해야합니다.
void *p = malloc(sizeof(char)*10);
p++; //compiler does how many where to pint the pointer after this increment operation
char * c = (char *)p;
c++; // compiler will increment the c by 1, since size of char is 1 byte.
컴파일러는 유형 캐스트로 알고 있습니다. 주어진 void *x
:
x+1
1 바이트를 추가x
하고 포인터는 바이트로 이동x+1
(int*)x+1
sizeof(int)
바이트를 추가 하고 포인터는 바이트로 이동x + sizeof(int)
(float*)x+1
주소sizeof(float)
바이트 등
첫 번째 항목은 이식성이 없으며 C / C ++의 Galateo에 위배되지만 C 언어가 정확합니다. 즉, 대부분의 컴파일러에서 적절한 플래그가 필요할 수있는 무언가로 컴파일됩니다 (예 : -Wpointer-arith).
참고 URL : https://stackoverflow.com/questions/3523145/pointer-arithmetic-for-void-pointer-in-c
'Programing' 카테고리의 다른 글
VIM에서 실제로 하나의 긴 줄을 여러 줄로 나누려면 어떻게해야합니까? (0) | 2020.06.02 |
---|---|
Spark 데이터 프레임에 전체 열 내용을 표시하는 방법은 무엇입니까? (0) | 2020.06.02 |
VS.NET 2013에서 __vwd / js / artery를 비활성화하려면 어떻게해야합니까? (0) | 2020.06.02 |
시스템에 남아있는 실행 파일이 없을 때 Windows 서비스를 제거하는 방법은 무엇입니까? (0) | 2020.06.02 |
UIWebView 내에서 HTML 및 로컬 이미지 사용 (0) | 2020.06.02 |