I apologize but I did try googling for this, and nothing good really came up.
https://github.com/rust-lang/rust/issues/17190#issuecomment-55372530
Which in my reading says itâs due to certain capabilities not being in the language. I figured Iâd ask this community the same question.
Edit: Found this which is a bit better at explaining. https://users.rust-lang.org/t/newbie-why-macros-vs-functions/1012
Comments
hi_im_nate ⢠32 points ⢠2016-06-30
Itâs so that you can use format parameters:
println!(âx is: {}â, x);
Since rust does not support varargs, this has to be implemented with a macro.
steveklabnik1 ⢠33 points ⢠2016-07-01
On top of that, it checks said format parameters at compile time, as well as secretly taking
&x
rather than justx
.lucidguppy ⢠8 points ⢠2016-07-01
Is it because varargs are unsafe? Or is it just a TODO?
minno ⢠14 points ⢠2016-07-01
I donât think there are any plans to make functions able to take a different number of values or different types (except through generics), like with overloading, optional parameters, or varargs.
Câs implementation of varargs is horrifically unsafe and unwieldy, and I donât think you could do a whole lot better with Rust. You would have absolutely no compile-time checking, and the runtime checks would make it really tedious to use. Youâd end up with a long block of
if let Some(s) = try_get_next_vararg_with_type::<&str>() { // stuff } else if let Some(x) = try_get_next_vararg_with_type::
() { // stuff } looneysquash ⢠9 points ⢠2016-07-01
Are you aware of how C++11 can do it with vararg templates?
See here, search for
tprintf
.hi_im_nate ⢠6 points ⢠2016-07-01
Itâs because varargs require hidden runtime overhead, and thatâs not something that Rust really tries to minimize.
[deleted] ⢠5 points ⢠2016-07-01
Parsing a format string at runtime has overhead compared to doing so at compile time, but I wouldnât call that hidden. The overhead of simply using a vararg function in C is basically negligible.
mutabah ⢠9 points ⢠2016-07-01
The core of
println!
isformat_args!
which is a compiler-implemented âmacroâ (also known as a syntax extension) that parses the format string and converts it into the format needed to print at runtime.
[deleted] ⢠9 points ⢠2016-07-01
println!() does a couple of things that a regular function canât do:
It parses the format string at compile time, and generates type safe code
It has a variable number of arguments
It has named arguments (âkeyword argumentsâ)
It takes parameters by reference implicitly
looneysquash ⢠3 points ⢠2016-07-01
That begs the question, will they ever be able to?
Personally, I would vote yes on 1-3, and no on 4.
For 1., I got the impression that constexpr stuff was planned, and not really controversial.
While 2. and 3. and 4. are more things that people will argue about.
Named arguments donât add any runtime overhead and arenât unsafe, and donât make the code harder to grep.
C style varargs are unsafe. C++ has vararg templates that are safe and use recursive expansion at compile time.
Java does varargs by implicitly converting the last arguments to an array, which requires that they all be the same type. I donât think that actually adds runtime overhead though, and is safe.
Maybe someone could come up with another way entirely. Maybe the arguments could be treated as a tuple and the compiler could type check each call separately, like it was using generics.
protestor ⢠3 points ⢠2016-07-02
About named arguments, a convention is to make a struct with the arguments and pass it as parameter. One can even have optional arguments by using the
Default
trait.Perhaps Rust could adopt a syntax sugar for named arguments that just convert them to a struct at compile time. For example:
#[derive(Default)] struct Point { x: i32, y: i32 }
fn draw_point(p: Point);
// explicit call draw_point(Point { x: 10, y: 10 });
// syntax sugar for the above draw_point(x = 10, y = 10)
// explicit optional parameter draw_point(Point { x: 10, ⌠Default::default() })
// syntax sugar for the above
draw_point(x = 10)
Actually. This could be a RFC.
looneysquash ⢠3 points ⢠2016-07-02
I personally think
Default::default()
is ridiculously long even for itâs normal purpose. There needs to be a really short way to write that for all uses. Something thatâs still grep able, but is short and easy to type.It seems odd to use
=
instead of:
, since you use:
for the struct literal syntax.But Iâm just nitpicking, I like the general idea.
There really ought to be sugar for defining such a function too, not just calling it.
protestor ⢠2 points ⢠2016-07-02
The thing is that
a: b
is the syntax for type ascription (and used in an example:foo(x: &[_], y: &[_]);
), which could lead to some ambiguity.Yeah,
Default
is unfortunately too verbose. :/edit: but actually the syntax
f(a = b)
is already taken too:a = b
is a valid expression and has type()
.[deleted] ⢠1 points ⢠2016-07-04
When the type is concrete (specific type or type parameter), Iâd use
T::default()
instead.When the type should be inferred from context, just like in
Default::default()
, you can also use the syntax<_>::default()
literally. For obvious reasons thatâs not very popular.looneysquash ⢠2 points ⢠2016-07-04
IMO, the word default, even once, is too long. I wish you could just say
...
. Maybe you could use....
. Or maybedft
, to match things likefn
andlet
.Uncaffeinated ⢠1 points ⢠2016-07-03
Itâs a shame thereâs no way to get rid of the redundant Point. Otherwise, the syntax would be pretty good already.
[deleted] ⢠1 points ⢠2016-07-01
.1. Is more than just constexpr instead code generation to use the correct trait for formatting. I donât see a way to implement that in regular rust even with constexpr.
[deleted] ⢠1 points ⢠2016-07-01
In Crystal variable (splat) arguments are packed as a tuple in the method argument. For example:
def foo(*args) args end
result = foo 1, âhelloâ typeof(result) # â Tuple(Int32, String)
Making varargs be tuples is very convenient and type safe, as one can forward them between methods.
We do something similar with named arguments: one can captured them with
**args
, and their type is a named tuple: a tuple where every key and its type is known at compile time.But since Rust requires type arguments in every method, I donât know if thatâs doable, probably notâŚ