====== A small list of C brainteasers ======
Here's a small list of C brainteasers that a few friends and I came up with and/or found. (All problems, except where specified, in C99.)
* Can you write some C of the form ''sizeof ( ... printf("hey!\n") ... )'' which results in ''hey!'' being printed?
* No, you may not put a close paren in the first ''...''.
* ''sizeof'' comes in two forms in C -- ''sizeof ( TYPENAME )'', and ''sizeof EXPRESSION''. Can you do it with both forms?
* Can you do it without emitting any warnings?
* Can you write some C that has the tokens ''[ * ]'' in a row?
* How about the tokens ''restrict 5''?
* Can you write an ''a.c'' that results in: jwise@jwise-lt-linux:~$ grep '#' a.c
jwise@jwise-lt-linux:~$ grep '{' a.c
jwise@jwise-lt-linux:~$ cc -std=c99 -pedantic -Wall -o a a.c && ./a
hey!
* ... portably?
* Can you write an ''a2.c'' that results in: jwise@jwise-lt-linux:~$ grep '#' a2.c
jwise@jwise-lt-linux:~$ grep '{' a2.c
jwise@jwise-lt-linux:~$ grep '\[' a2.c
jwise@jwise-lt-linux:~$ grep '"' a2.c
jwise@jwise-lt-linux:~$ grep "'" a2.c
jwise@jwise-lt-linux:~$ gcc -std=c89 -pedantic -o a2 a2.c && ./a2
hey!
* ... portably?
* Can you write a program ''a3.c'' that results in: jwise@jwise-lt-linux:~$ grep '"' a3.c
jwise@jwise-lt-linux:~$ grep "'" a3.c
jwise@jwise-lt-linux:~$ grep ';' a3.c
jwise@jwise-lt-linux:~$ gcc -std=c99 -pedantic -Wall -o a3 a3.c && ./a3
hey!
* Portably?
* Can you write a program ''a4.c'' that results in: jwise@jwise-lt-linux:~$ grep '"' a4.c
jwise@jwise-lt-linux:~$ grep "'" a4.c
jwise@jwise-lt-linux:~$ grep ';' a4.c
jwise@jwise-lt-linux:~$ grep -E '{[^}]|{$' a4.c
jwise@jwise-lt-linux:~$ gcc -std=c99 -pedantic -Wall -o a4 a4.c && ./a4
hey!
* Without using the tricks from ''a'' or ''a2''?
* The only implemenation I know of requires the compiler to take a specific view of the specification; you'll need ''gcc'''s view of the specification, not ''clang'''s.
* //(taken from elsewhere on the Web)// Without running it, what is the output of the following program? \\ ''int main(void) {int n = sizeof(0)["abcdefghij"]; printf("%d\n", n); return 0; }''
* It is not machine-specific.
* A common idiom in C is ''assert(!"That's not gone well")''. Unfortunately, a common idiom in C amongst careless programmers is ''assert("That's not gone well")'', or the potentially even more amusing ''assert("!That's not gone well")''. Can you write a macro that chokes with a compile time failure on the latter two cases?
* More concretely: it is easy to write a static assert macro that fails to compile if a given expression is either a) false, or b) not compile-time constant. Your task is to write a static assert macro that fails to compile if a given expression is a) true **and** b) a compile-time constant!
* Here are some test cases, if you care:
ASSERT_CHECKER(time());
ASSERT_CHECKER(0);
ASSERT_CHECKER("abc"[2]);
ASSERT_CHECKER(!"biteme");
ASSERT_CHECKER("biteme");
ASSERT_CHECKER(2 + 2);
The last two should fail, by the C99 specification. There is an easier way to do it with a GCC extension, but you do not need any GCC extensions to do this.
* //(C++11-only)// Find a use for the token sequence ''......''.
I've enabled comments below, but please don't post spoilers.
~~DISQUS~~