-
Notifications
You must be signed in to change notification settings - Fork 258
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[BUG] Cannot deduce a braced-init-list (perhaps "parens-init-list" in Cpp2-speak) #1020
Comments
I think the general solution would be to emit the
so Finally, thanks to CTAD, |
Interestingly, clang rejects the #include <initializer_list>
#include <vector>
int main()
{
std::vector const vec = {11, 22, 33, 44, 55}; // OK
std::initializer_list const ints = {11, 22, 33, 44, 55}; // Error
} Repro on Godbolt with 3 major compilers. Some discussion on whether its valid. |
Actually the Cpp2 equivalent is even simpler:
Would that answer your immediate question/need? Unless you really want Also, deducing |
Yeah, that was the intention. I tried to minimise the repro code to describe the problem but should've linked to the full context. I was translating this code example from cppreference: #include <barrier>
#include <iostream>
#include <string>
#include <syncstream>
#include <thread>
#include <vector>
int main()
{
const auto workers = {"Anil", "Busara", "Carl"}; // <--- Deduced initializer_list
auto on_completion = []() noexcept
{
// locking not needed here
static auto phase =
"... done\n"
"Cleaning up...\n";
std::cout << phase;
phase = "... done\n";
};
std::barrier sync_point(std::ssize(workers), on_completion); // <--- Used here
auto work = [&](std::string name)
{
std::string product = " " + name + " worked\n";
std::osyncstream(std::cout) << product;
sync_point.arrive_and_wait();
product = " " + name + " cleaned\n";
std::osyncstream(std::cout) << product;
sync_point.arrive_and_wait();
};
std::cout << "Starting...\n";
std::vector<std::jthread> threads;
threads.reserve(std::size(workers)); // <--- Used here
for (auto const& worker : workers) // <--- Used here
threads.emplace_back(work, worker);
} This is a work in progress but I'm slowly copying and converting the examples from cppreference (under their permissive license) here: cpp2reference.com |
OK, thanks.
🤯 Mind blown 🤯 |
In the given example, why not use std::array? I mean, I know you're converting the examples but cpp2 does not need std::initializer_list, it has uniform initialization which just works. Any use of std::initializer_list as a container/range can be replaced by std::array or std::vector instead and cpp2 does not expose it in initialization. It'll be better if std::initializer_list remains merely an implementation detail and isn't taught with cpp2. |
Good point, I’ll update it to use std::array. I guess the main motivation for this issue is to find a terse Cpp2 equivalent for:
Because I think someone coming from C++ would try and reach for:
|
There have been discussions and suggestions about built-in arrays in cpp2. Maybe something like:
This could transpile to something like a cpp2::array and replace the uses cases of both std::array and std::initializer_list, kill two birds with one stone! |
Or, doing some zen-of-cpp2 thinking here, what about using defaults? If the declaration is of the form
we know there is no explicit type on the right-hand side... I'm pretty sure there's no way to smuggle one in, because Cpp2 deliberately doesn't have the comma operator which would be the main way I would think of to force it in Cpp1. So, since we know the RHS is a list, we could do the "deduction" ourselves as a language default, and emit But this is a late-night thought and I could be missing something. What do you think? |
Yes, I like that. So if:
defaults to std::array, could this:
default to std::tuple? |
I don't think changing the type based on the list contents is easy, for the compiler or the human. Feels a bit magical. For example, if it deduces to tuple we can't for-each over it. If you want tuple, just ask? But I suppose that works here too: Given that, perhaps the following is fine, which works today? (And has no magic... this is WYSIWYG code.)
And we could still add the default to |
That is a nice solution but maybe we we should try for something that solves other issues too like #542 and #568 (athough they both can be solved by #927 ). |
That sounds like a reasonable assumption to make , I kind of like it, however...
... I have seen that I totally agree with @AbhinavK00 , we should try to solve other issues along this one if possible. |
What is the cost (time and storage) difference between these two?
There is also a difference if you pass them to template functions, as
This results in two versions of |
Here's another case... should the user be able to expect they can modify the result? to be able to subscript?
Whereas:
|
I think this is convincing me that |
I guess it doesn't hurt sticking with |
Would it make sense for How difficult would it be for cppfront to detect these patterns and suggest changing the type to
|
After trying it out and running regressions, there are a few conflicts, such as I think I do prefer this which already works:
It's WYSIWYG/magic-free, and I think reasonably easy to type? Thanks again! I'm going to close this without change for now, but that still leaves open the door to deduction in the future. |
Describe the bug
Cannot deduce a C++ braced-init-list (written with parens in Cpp2). Instead I have to explicitly write out the type.
To Reproduce
I'm trying to translate this C++ code:
into Cpp2:
But the relevant line lowers to:
And the C++ compiler reports this error:
Repro on Godbolt
Changing the declaration to the following works as desired:
The text was updated successfully, but these errors were encountered: