I
Ian Collins
jacob said:Flash Gordon a écrit :
No. As you (may) know, you have to write extern "C" in C++
Er, so? Wrap the header in extern "C" {..} and away you go.
jacob said:Flash Gordon a écrit :
No. As you (may) know, you have to write extern "C" in C++
jacob navia said:Flash Gordon a écrit :
No. As you (may) know, you have to write
extern "C"
in C++
You know what's an awesome feature in a programming language? An "eval"
operation, which interprets strings as code. Incredibly flexible and
powerful.
Do you think we should add eval() to C? I don't.
I do not think this is a fair representation of his argument.
I believe his argument is that he does not feel that adding A and B would
improve C. For instance, I don't think that adding operator overloading would
improve C. I think it would make C worse. One of the things I like about
C is the confidence that I know what it's doing. Operator overloading is
a great fit for an OO language, where I expect objects to have internal state
and define their own rules. It's a poor fit for a language where I
expect operations to map very closely onto machine instructions.
-s
Under lcc-win you can do:
The only thing that operator overloading adds to the language is the possibility
for the USER to add itsb own numeric types. Besides, if you do not like that
feature, do not use it. The language stays the same otherwise.
unsigned long a,b,c;
// ...
c = a+b;
This translates into a function call in some machines.
(Specially 16 bit machines).
jacob said:Flash Gordon a écrit :
No. As you (may) know, you have to write
extern "C"
in C++
That's a viewpoint. Another viewpoint (mine) is that experience has told
me NOT to use any widget set but use the API. This way, I have been able
to run programs written under windows 16 bits
today under windows 64 bits.
Richard Heathfield said:bartc wrote:
Which is an alternative to writing (with some help from cdecl):
int (*x[10])(const char *);
Yes.
The typedefs at least turn it from impossible to possible, although in my
scheme, it would just be:
[10]*function(* const char)int x;
EEK!
bartc said:The typedefs at least turn it from impossible to possible, although in my
scheme, it would just be:
[10]*function(* const char)int x;
It doesn't *look* that different, but I managed to write it in a few seconds
from your description above, and anyone can understand it by reading from
left to right.
jacob navia said:Flash Gordon a écrit :
No. As you (may) know, you have to write
extern "C"
in C++
What does it means "GUI-intensive" and why should java be better at
solving these kinds of problems? Excluding subjective aesthetic
consideration on the resulting GUIs, I think some operations
"could" (but I'm unaware of recent changes so I could be wrong) be a bit
slower than an analogous C or C++ counterpart. Of course if maximum
speed is not required that's ok. A slow menu is not that cool btw...
jacob navia said:Seebs a écrit :
Under lcc-win you can do:
typedef void (fnptr)(void);
fnptr fn;
fn = CompileString("int main(void)(printf("\hello\"); )");
fn();
and you get "hello"...
This is called a JIT (Just in Time Compiler). Lcc-win has a JIT
version.
The only thing that operator overloading adds to the language is the possibility
for the USER to add itsb own numeric types. Besides, if you do not like that
feature, do not use it. The language stays the same otherwise.
jacob said:Ian Collins a écrit :
Because the C standards committee has always decided NOT to add such
an improvement to the language. The C++ standard committee decided that
such an improvement WAS NECESSARY, and they develop one.
I have written most of the STL clone (AVL trees, RB trees, bitstrings,
flexible arrays, lists (double/single linked) and the whole library
is extremely fast and compact. It is just less than 100K when you use
all of it. If you use just the lists module it is less than 5K.
But apparently all this work is a waste of time. The committee decided
a feature freeze already, and I will not be ready to present this to
the next meeting in April.
Richard said:Casey said:Peter Van Der Linden in "Expert C Programming - Deep C Secrets" 1994
and I imagine others have pointed out some of C's idiosyncrasies.
Has thought been given given to a cleaned up C? Possibly called C+.
According to Stroustrup [TC++PL, intro], the name "C+" has already been
taken, by a language unrelated to either C or C++.
If you replaced "No" by "Yes", I'd agree. Yes, you have to write
extern "C"; that's *how* the API is available to C++.
I'm sure you didn't mean to imply that typing extern "C" is too
difficult, but I can't figure out what point you were making.
Can you clarify?
"C's magic low level abilities allow us to do things no other language
can achieve"
jacob said:Nick Keighley a écrit :
For instance:
A full blown IDE under 1MB
A fast floating point package with 352 bits precision, in C+asm. The
core in assembler, the full C math library plus many other
special functions in C. Size: 204K.
I like C. The problem with modern softawre is BLOAT. Too much
features you do not need, too much inefficiency, the
contrary of "small is beautiful".
Well, I guess its a philosophical question anyway.
bartc wrote:And the typedef statement itself, which instead of having the type-spec
on one side, and the alias for it on the other, is all mixed up:typedef int tenvector[10];the typedef name is in the middle of the type-spec!
That's actually irrelevant - the typedef is remarkably easy to use.
To
define a synonym for an existing type, all you have to do is pretend
you're defining an object of that type, and stick typedef on the front.
Let's say you want a synonym for an array of ten pointers to function
taking const char * and returning int.
Start simple:
int func(const char *);
That isn't actually what we want, but it's the first step. Now let's
turn it into a type synonym definition:
s/int/typedef int/
gives us
typedef int func(const char *);
func is now a type synonym. We use this in the next stage:
func arr[10];
Again, not quite what we want, so we:
s/func/typedef func/
to give
typedef func arr[10];
arr is now a synonym for "array of ten pointers to function taking const
char * and returning int".
It's hard to see how this could be made much simpler.
bartc said:Which is an alternative to writing (with some help from cdecl):
int (*x[10])(const char *);Yes.The typedefs at least turn it from impossible to possible, although in my
scheme, it would just be:
[10]*function(* const char)int x;EEK!
That's supposed to be my reaction to the C equivalent...
My version actually is based on Algol-68 syntax, something like this I
believe, without 'const' attribute:
[10]ref proc(ref char)int x;
I changed 'ref' to '*' in order to be more C-friendly, obviously that didn't
work...
Ben Bacarisse said:bartc said:The typedefs at least turn it from impossible to possible, although in my
scheme, it would just be:
[10]*function(* const char)int x;
It doesn't *look* that different, but I managed to write it in a few
seconds
from your description above, and anyone can understand it by reading from
left to right.
I think the word "function" is redundant in your scheme. Just as []s
announce an array, ()s announce a function. If you are concerned
about readability then "[]" could become "array []" and "*" could be
"pointer to" but if you are happy with minimal syntax elsewhere in the
scheme I'd suggest it for functions too.
jacob navia said:Seebs a écrit :
Under lcc-win you can do:
typedef void (fnptr)(void);
fnptr fn;
fn = CompileString("int main(void)(printf("\hello\"); )");
fn();
and you get "hello"...
This is called a JIT (Just in Time Compiler). Lcc-win has a JIT
version.
Can you do:
typedef void (fnptr)(void);
void tt(const char *msg) {
char fnbuff[1000];
fnptr fn;
snprintf(fnbuff,1000,"int main(void)(printf(\"%s\"); )");
fn = CompileString(fnbuff);
fn();
}
?
Because, if so, aren't you shipping the compiler around with the
executable.
What if the code to be compiled relies on a library that
isn't on the target machine?
And if not, what does it give you apart
from losing a static function (or, I admit, a group of closely related
static functions all with the same boilerplate).
The only thing that operator overloading adds to the language is the possibility
for the USER to add itsb own numeric types. Besides, if you do not like that
feature, do not use it. The language stays the same otherwise.
The danger with operator overloading is that it lets the USER add their
own NON NUMERIC [I thought I'd try adopting your STYLE for a while, but
got out of BREATH fairly QUICKLY] types. Look how long it took C++ to
get << and >> for streams - about 10 seconds.
The thing is though, this thread started by asking about what was got
wrong when C was designed. You immediately jumped in telling us about
what you've added to it. I think the intent of the question was what is
wrong with the underlying design.
Well I'd start with the inconsistencies in the standard library.
Half the functions that send things to a file take the file pointer as
the first argument. The other half take it as the last. puts is /not/
fputs defaulting to stdout.
By now these quirks are all second nature to me, but they are deeply
ugly and very confusing to newcomers.
Then I'd move on to the use of break inside both loops and switches -
leading to the behaviour of one of break/continue changing when you
embed a loop inside a switch clause but not the other.
jacob said:Thomas Richter a crit :Nobody is proposing that. This is an old trick in this discussions. Any
improvement to C is impossible because the old code. That is plainly not
true. The old code and the associated functions can be kept, but C
should be able to use BUFFERS of KNOWN length, avoiding the most common
cause of buffer overflows. Let's be explicit:typedef struct {size_t length; char data[];} Buffer;memset can be done as "Memset", for instance like this
bool Memset(Buffer *dest, int char_val);[...] Strcpy is just as trivial, but with a character
buffer.
There is already a secure version of strcpy, see below.
Then C is not the right tool for the job. As said below, there are other
languages that can help. If you want to use C, there are also secured
functions with buffer sizes that help. But even they won't prevent you
from shooting yourself into the foot. C has no means to prevent bad
pointers being dereferences, to give another example. If you want a
child-proof C, use Java - it does have all the features you seem to
miss. (And no, I've absolutely nothing against Java, no irony here.)
Of course I make mistakes, who wouldn't? And your point is? Again, if
"avoiding errors" is your concern, use another tool and make other
errors. Just that languages like java or C++ are neither the right tool
for *some* jobs.
A lot of this repeats so... said:boring...
Then where is the damn problem?
Then keep it simple. If writing a list or a tree is not simple for you,
you are in the wrong field for applying C for that task.
Or look for a
library that helps you. C is a small language; why whould that stuff be
required in the language core - it would often not fit into the
application because C doesn't have all the mechanisms of higher
languages to hide the implementation behind the design.
It's obvious that you want to use C just for the means of it. Get over
it, C is not the "saves all my problems" language. It is good in some
cases, and bad in others. String manipulation and generic container
support are not the strong parts of C, so why force it? If you want to
make C another language, I again ask why. There *are* already other
languages in first place you can use.
Want to reply to this thread or ask your own question?
You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.