답변:
strerror()
오류 번호에 대해 사람이 읽을 수있는 문자열을 얻는 데 사용할 수 있습니다 . 이것은 동일한 문자열로 인쇄 perror()
되지만 표준 오류 출력 이외의 오류 메시지를 형식화하는 경우 유용합니다.
예를 들면 다음과 같습니다.
#include <errno.h>
#include <string.h>
/* ... */
if(read(fd, buf, 1)==-1) {
printf("Oh dear, something went wrong with read()! %s\n", strerror(errno));
}
Linux는 또한 명시 적으로 스레드 안전 변형을 지원합니다 strerror_r()
.
strerror()
스레드 안전하지만 스레드 안전하지 않습니다 strerror_r()
. MT-Safe 또는 Thread-Safe 기능은 다른 스레드가있을 때 호출해도 안전합니다. MT-Safe에서 MT는 멀티 스레드를 나타냅니다. -p26, GNU C 라이브러리 char * strerror(int errnum )
[기능] 예비 : | MT- 안전하지 않은 경주 : 공격 | AS 안전하지 않은 힙 i18n | AC 안전하지 않은 mem | 1.2.2.1 절 [POSIX 안전 개념], 2 페이지를 참조하십시오. -p58, GNU C 라이브러리
errno
스레드로부터 안전합니다. 보세요 :#define errno *__errno_location()
errno
GNU libc strerror
는 내부 전역 버퍼에 씁니다 . 항상 그렇지는 않지만 경우에 따라 다릅니다. 다른 스레드가 읽는 동안 해당 버퍼를 덮어 씁니다.
errno
하지 strerror()
.
perror
오류 코드를 실행 하는 대신 errno
다음의 한 줄짜리를 사용하여 시스템에서 전체 값 목록을 검색 할 수 있습니다 .
cpp -dM /usr/include/errno.h | grep 'define E' | sort -n -k 3
Linux에는 각 오류 코드의 의미를 바로 알 수있는 매우 깔끔한 도구도 있습니다. 우분투에서 : apt-get install errno
.
그런 다음 예를 들어 오류 유형 2에 대한 설명을 얻으려면 errno 2
터미널에 입력 하십시오.
함께 errno -l
하면 모든 오류와 그에 대한 설명과 함께 목록을 얻을. 이전 포스터에서 언급 한 다른 방법보다 훨씬 쉽습니다.
perror
/를 포함하는 strerror
것이 좋습니다. 그러나 errno -l
그들을 찾는 것이 좋습니다.
다음은 errno -l
가독성을 위해 다시 포맷 한 결과입니다 .
1 EPERM Operation not permitted
2 ENOENT No such file or directory
3 ESRCH No such process
4 EINTR Interrupted system call
5 EIO Input/output error
6 ENXIO No such device or address
7 E2BIG Argument list too long
8 ENOEXEC Exec format error
9 EBADF Bad file descriptor
10 ECHILD No child processes
11 EAGAIN Resource temporarily unavailable
11 EWOULDBLOCK Resource temporarily unavailable
12 ENOMEM Cannot allocate memory
13 EACCES Permission denied
14 EFAULT Bad address
15 ENOTBLK Block device required
16 EBUSY Device or resource busy
17 EEXIST File exists
18 EXDEV Invalid cross-device link
19 ENODEV No such device
20 ENOTDIR Not a directory
21 EISDIR Is a directory
22 EINVAL Invalid argument
23 ENFILE Too many open files in system
24 EMFILE Too many open files
25 ENOTTY Inappropriate ioctl for device
26 ETXTBSY Text file busy
27 EFBIG File too large
28 ENOSPC No space left on device
29 ESPIPE Illegal seek
30 EROFS Read-only file system
31 EMLINK Too many links
32 EPIPE Broken pipe
33 EDOM Numerical argument out of domain
34 ERANGE Numerical result out of range
35 EDEADLK Resource deadlock avoided
35 EDEADLOCK Resource deadlock avoided
36 ENAMETOOLONG File name too long
37 ENOLCK No locks available
38 ENOSYS Function not implemented
39 ENOTEMPTY Directory not empty
40 ELOOP Too many levels of symbolic links
42 ENOMSG No message of desired type
43 EIDRM Identifier removed
44 ECHRNG Channel number out of range
45 EL2NSYNC Level 2 not synchronized
46 EL3HLT Level 3 halted
47 EL3RST Level 3 reset
48 ELNRNG Link number out of range
49 EUNATCH Protocol driver not attached
50 ENOCSI No CSI structure available
51 EL2HLT Level 2 halted
52 EBADE Invalid exchange
53 EBADR Invalid request descriptor
54 EXFULL Exchange full
55 ENOANO No anode
56 EBADRQC Invalid request code
57 EBADSLT Invalid slot
59 EBFONT Bad font file format
60 ENOSTR Device not a stream
61 ENODATA No data available
62 ETIME Timer expired
63 ENOSR Out of streams resources
64 ENONET Machine is not on the network
65 ENOPKG Package not installed
66 EREMOTE Object is remote
67 ENOLINK Link has been severed
68 EADV Advertise error
69 ESRMNT Srmount error
70 ECOMM Communication error on send
71 EPROTO Protocol error
72 EMULTIHOP Multihop attempted
73 EDOTDOT RFS specific error
74 EBADMSG Bad message
75 EOVERFLOW Value too large for defined data type
76 ENOTUNIQ Name not unique on network
77 EBADFD File descriptor in bad state
78 EREMCHG Remote address changed
79 ELIBACC Can not access a needed shared library
80 ELIBBAD Accessing a corrupted shared library
81 ELIBSCN .lib section in a.out corrupted
82 ELIBMAX Attempting to link in too many shared libraries
83 ELIBEXEC Cannot exec a shared library directly
84 EILSEQ Invalid or incomplete multibyte or wide character
85 ERESTART Interrupted system call should be restarted
86 ESTRPIPE Streams pipe error
87 EUSERS Too many users
88 ENOTSOCK Socket operation on non-socket
89 EDESTADDRREQ Destination address required
90 EMSGSIZE Message too long
91 EPROTOTYPE Protocol wrong type for socket
92 ENOPROTOOPT Protocol not available
93 EPROTONOSUPPORT Protocol not supported
94 ESOCKTNOSUPPORT Socket type not supported
95 ENOTSUP Operation not supported
95 EOPNOTSUPP Operation not supported
96 EPFNOSUPPORT Protocol family not supported
97 EAFNOSUPPORT Address family not supported by protocol
98 EADDRINUSE Address already in use
99 EADDRNOTAVAIL Cannot assign requested address
100 ENETDOWN Network is down
101 ENETUNREACH Network is unreachable
102 ENETRESET Network dropped connection on reset
103 ECONNABORTED Software caused connection abort
104 ECONNRESET Connection reset by peer
105 ENOBUFS No buffer space available
106 EISCONN Transport endpoint is already connected
107 ENOTCONN Transport endpoint is not connected
108 ESHUTDOWN Cannot send after transport endpoint shutdown
109 ETOOMANYREFS Too many references: cannot splice
110 ETIMEDOUT Connection timed out
111 ECONNREFUSED Connection refused
112 EHOSTDOWN Host is down
113 EHOSTUNREACH No route to host
114 EALREADY Operation already in progress
115 EINPROGRESS Operation now in progress
116 ESTALE Stale file handle
117 EUCLEAN Structure needs cleaning
118 ENOTNAM Not a XENIX named type file
119 ENAVAIL No XENIX semaphores available
120 EISNAM Is a named type file
121 EREMOTEIO Remote I/O error
122 EDQUOT Disk quota exceeded
123 ENOMEDIUM No medium found
124 EMEDIUMTYPE Wrong medium type
125 ECANCELED Operation canceled
126 ENOKEY Required key not available
127 EKEYEXPIRED Key has expired
128 EKEYREVOKED Key has been revoked
129 EKEYREJECTED Key was rejected by service
130 EOWNERDEAD Owner died
131 ENOTRECOVERABLE State not recoverable
132 ERFKILL Operation not possible due to RF-kill
133 EHWPOISON Memory page has hardware error
Vim에서 테이블 형식을 사용하여 열을 정렬했습니다.
:%Tab /^[^ ]*\zs /r1l1l1
:%Tab /^ *[^ ]* *[^ ]*\zs /l1
errno
s 를 다루는 몇 가지 유용한 기능이 있습니다 . (명확하게하기 위해 이것들은 내장되어 있습니다 libc
-일부 사람들은 영어를 읽는 것보다 코드를 읽는 것이 더 명확하기 때문에 샘플 구현을 제공하고 있습니다.)
#include <string.h>
char *strerror(int errnum);
/* you can think of it as being implemented like this: */
static char strerror_buf[1024];
const char *sys_errlist[] = {
[EPERM] = "Operation not permitted",
[ENOENT] = "No such file or directory",
[ESRCH] = "No such process",
[EINTR] = "Interrupted system call",
[EIO] = "I/O error",
[ENXIO] = "No such device or address",
[E2BIG] = "Argument list too long",
/* etc. */
};
int sys_nerr = sizeof(sys_errlist) / sizeof(char *);
char *strerror(int errnum) {
if (0 <= errnum && errnum < sys_nerr && sys_errlist[errnum])
strcpy(strerror_buf, sys_errlist[errnum]);
else
sprintf(strerror_buf, "Unknown error %d", errnum);
return strerror_buf;
}
strerror
전달 된 오류 번호를 설명하는 문자열을 반환합니다. 주의, 이것은 스레드 또는 인터럽트 안전이 아닙니다. 문자열을 다시 쓰고 다음 호출에서 동일한 포인터를 리턴 할 수 있습니다. strerror_r
걱정할 필요가 있다면 사용하십시오 .
#include <stdio.h>
void perror(const char *s);
/* you can think of it as being implemented like this: */
void perror(const char *s) {
fprintf(stderr, "%s: %s\n", s, strerror(errno));
}
perror
지정한 메시지와 현재를 설명하는 문자열 errno
을 표준 오류로 인쇄합니다.
some people find reading code clearer than reading English
진실.
이것은 코드를 찾는 것보다 빠르며 errno.h
여기에 게시 된 대부분의 솔루션보다 짧으며 타사 도구를 설치할 필요가 없습니다.
perl -E 'say $!=shift' 2
수확량
No such file or directory
-E
는 다음 인용 코드를 Perl 스크립트로 취급합니다. say
표준 출력에 인수를 쓰고 있습니다. $!
의 값을 보유하는 특수 변수입니다 errno
. 문자열 컨텍스트에서 사용하면 해당 오류 문자열이 생성됩니다. 스크립트 2
는 shift
명령 을 사용하여이 변수에 값 을 할당합니다. 이 명령은 인수 배열의 헤드를 잘라 내고이 @ARGV
헤드를 그 자리에 배치합니다. 명령 줄은 다음과 같이 쓸 수도 있습니다 perl -E 'say $!=2'
.
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
int main(int i, char *c[]) {
if (i != 2)
fprintf(stderr, "Usage: perror errno\n");
else {
errno = atoi(c[1]);
perror("");
}
exit(0);
}
Solaris에서 작동합니다.
cc perror.c -o perror
<<이 줄을 사용하여 컴파일하십시오.
perror(NULL);
합니까? 와 perror("");
의 그것은 형식입니다 : ERROR NAME
. perror(NULL);
그것의 출력 과 함께ERROR NAME
여기 문서가 있습니다. 그것은 그것이 무엇을 의미하고 그들과 어떻게해야하는지 알려줄 것입니다. 숫자 값을 사용하지 말고 여기에 나열된 상수를 사용해야합니다. 숫자는 시스템마다 다를 수 있습니다.
errnoname
라이브러리에 관심이있을 것 입니다. 그것은 가능한 모든 errno 이름을 C 함수로 모으는 지루한 부분을 처리합니다.이 함수는 숫자에서 errno 매크로 이름을 얻는 데 사소하게 사용될 수 있습니다. 이를 기반으로 사용하면 perror
빠르고 쉽게 변형을 만들 수 있습니다.
다음 스크립트를 사용합니다.
#!/usr/bin/python
import errno
import os
import sys
toname = dict((str(getattr(errno, x)), x)
for x in dir(errno)
if x.startswith("E"))
tocode = dict((x, getattr(errno, x))
for x in dir(errno)
if x.startswith("E"))
for arg in sys.argv[1:]:
if arg in tocode:
print arg, tocode[arg], os.strerror(tocode[arg])
elif arg in toname:
print toname[arg], arg, os.strerror(int(arg))
else:
print "Unknown:", arg
내의 .bashrc 파일에 다음과 같은 기능을 가지고 - 그것은 헤더 파일에서의 errno 값 (될 수 있습니다 조회 /usr/include/errno.h
, /usr/include/linux/errno.h
등 등)
헤더 파일이 컴퓨터에 설치된 경우 작동합니다 .-)
일반적으로 헤더 파일에 오류가 있습니다 + 다음은 주석의 설명입니다. 다음 중 하나 :
./asm-generic/errno-base.h:#define EAGAIN 11 / * 다시 시도 * /
function errno()
{
local arg=$1
if [[ "x$arg" == "x-h" ]]; then
cat <<EOF
Usage: errno <num>
Prints text that describes errno error number
EOF
else
pushd /usr/include
find . -name "errno*.h" | xargs grep "[[:space:]]${arg}[[:space:]]"
popd
fi
}