-
-
Notifications
You must be signed in to change notification settings - Fork 2.2k
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
Just published the first V example to show you some features of the language. Very interested in your input. #3
Comments
Hi @medvednikov ! My answer will be off-topic: I just wanted to thank you for what you are doing with Volt and V, both look very promising ! I see a huge amount of feature requests, and I just wanted to say: Take your time, good softwares takes time. It's easy to get overwhelmed building Free softwares: sometimes it's better to say "no" or "not for now" in order to build great things in the long run :) Have a good day ! |
Here's an assortment of thoughts: From a Ruby and C background, I quite appreciate constants having to be uppercase. It also helps grepability (you know that you have to You may already want to put special thought into the HTTP library since I assume it will see a lot of usage. In particular, you may want to support both synchronous and asynchronous patterns; consider some standard library way to event loop all kinds of networking. I enjoy the design of the How do you return failure from main? This isn't as much of a concern on Windows and for GUI applications, but your example doesn't seem to address a case like I'm not sold on having both Looking at the printing examples: Is it accurate that a variable identifier can be any character. What happens if you try to What even is an Are you making any guarantees about the C code output of a function that only does arithmetic on integer types and has no branching? This may be relevant for the implementation of cryptographic algorithms in particular. Concurrency remains an open issue to address in another example, too, probably. |
Hi @xorhash Thanks a lot for your detailed feedback!
Absolutely!
Just like in go:
Can you explain what you mean?
Very good question. This will be one of the first topics in the docs. V has
Could you please give an example? Thanks |
Or did you mean what happens if we call |
As for I might remove
|
Just curious, why not go the other way with strings, and only use double quotes? That seems more cross language consistent as it means you get variable interpolation etc., and in my experience apostrophes are much more common in strong literals than double quotes, so if you’re going to pick one the double quotes will require less escaping. |
Also, off topic, but since the above is my first post on your project I just wanted to say hello! V looks fantastic, take the good stuff of Go and make it even better (to me anyway). Would love to help if there’s any way I can. Please let me know :) |
Will you keep raw string literals (backticks) from Go? |
@burlesona JSON is very popular right now, and it has lots of Also to me But I had your exact thoughts, maybe I'll do that. Just to be consistent with other languages. @slimsag |
@medvednikov yeah it’s kind of tricky to balance, I think this is why most languages have a few string literal options :) Maybe a block string literal syntax like Ruby’s heredoc (or many other examples) is a viable alternative for things like JSON input? But it does result in more than one way to make a string, which I agree isn’t perfect. Or finally, less easy to type, but you could just go straight for backticks or some other universal string delimiter. |
@medvednikov Not true -- |
Hello @medvednikov and congratulations for V! It seems to solve many of Go's issues and could be a nice "better C". A few questions:
|
Hi @oliwer
It's going to be either
There's no runtime, memory management will be similar to Rust.
Backticks are used for characters. Which feels a bit weird. Are there keyboard layouts that don't require shift to write
Thanks! I did some Perl in the past, that might have influenced the syntax :)
It already does. |
Would be cool to see an example of generics somewhere! |
+1 to seeing some example of generics. |
OT: Will it be possible to convert Go code to V? |
@slimsag @boyter I'll add an example with Generics soon. @rrjanbiah yes, but coroutines and channels have to be implemented first. |
Hi @medvednikov , does volt also use generics in it's codebase? Does it affect the compile times much? |
@International no. V compiler used to rely on generics to implement arrays and maps, but then I replaced them and made the code smarter (similar to Go's slices and maps). There will be a way for users to create their own generic containers without generating extra code for each type and slowing down compilation time. |
I've added an example with generics: https://github.com/vlang-io/V/blob/master/examples/generic_repository.v |
I think it looks pretty good. I like a lot of the semantic choices you've made even if I don't particularly like some of Go's syntax that you're inheriting. I'm sure I'd get used to it though (except Immutability and first-class optionals (
Either of these is good IMO, I don't like Go's capitalization rule for exports.
It's probably best to use double quotes for strings and single quotes for characters but I understand the ease of typing isn't great for you. I use single quotes in JavaScript/TypeScript all the time but with everything else I code in using doubles, it can be a bit of a context switch. Is there any short lambda syntax? How about pattern matching? It's one of my favorite features in Rust and ML languages, especially F#. Is this a typo in the generic example? Should |
Hey @citycide Thanks for your feedback!
Yes, I initially called it
Can you write a small example?
Not yet. Most likely there will be one. Need to think about the details.
Same :) |
I'm very skeptical about the "400 KB compiler with zero dependencies" part. It just transpiles to C and then compiles with gcc/clang, doesn't it? that implies a dependency on gcc or clang. can the V compiler emit machine code by itself? |
@Francesco149 this is a discussion about the syntax of the language. V can compile to machine code directly, as well as to C. |
Saw a couple of user examples that you published. V Looks interesting.
I would be glad if you can point me to some documentation. |
Also ABI compatibility with C/C++. Is it possible to link in C/C++ functions directly in V? |
|
As someone who spends 98% of his time in C & C++ and has no experience with Go, any sort of "V for C Programmers" documentation would be very helpful. 😄 Looking forward to seeing V's release! |
@awesomekyle V is language that can be learned in an hour. A documentation/tutorial page will be out soon, and you'll see ;) |
Sure, I'd be for that as well.
Seems like something like that could work. I know it seems that you are against "declaring" errors before, but what if we take the idea that you're working with and say that the errors are always local to where the are defined. So the bare
If we had qualified imports, this could be shortened to:
This seems to accomplish the namespacing without requiring pre-definition. Just a note to highlight why the clash is important. This might seem far-fetched but I'm positive it would happen at least once if the language became very popular with out it. Keeping with our example above, let's say that
All good at the time it's written. Now let's say we update our dependencies. I completely agree with regards to tooling. Though I would personally like to see those things implemented via something like LSP to prevent a single editor from ruling the language - np reason to not let people choose their tool for themselves! |
Warning: extremely long read! I'd mentioned that I had toyed with an idea of designing language of my own. Allow me to list features of this hypothetical language, perhaps something could serve as an inspiration. The language was intended to be similar to C (semicolons, {} brackets, C syntax kept where possible) and simpler than C (hopefully). I tried to stay away from all those complicated things promoted by academia. More interesting features of the language: 1] typical source file would look like:
I was inspired by Turbo Pascal. This model would replace C headers, while still keeping their advantage (they clearly show what is exported). Order of declarations was important:
This petty strictness would allow easier orientation in the code. 2] The language would allow one to read codebase of any size linearly, like a book. You inherit someone's else project/library, and you know where to start and where to go next. You do not need to jump back and forth, again and again. This trick makes reading of foreign code pleasure, instead of the usual pain. I was inspired how one Lisp implementation (named Wart) was organized: Basically, source files listing would look like:
Each source file would need unique numeric prefix (and these prefixes would need to look consistent - compiler ensures it), and code from file "030" would depend only and only on code from files "001" ... "029". The compiler will ensure that this is true. This numeric order would give firm hand to the code reader/explorer. Even subdirectories would follow this rule
Inside subdirectores the numbering would restart. This would allow to easily insert someone's else code:
The 3] The linear ordering above would allow to get rid of misfeature called "import" or "#include". Every C file starts with mandatory litany of:
This is madness, useless, harmful, attention killer. In my language whatever was exported by previous files would be automatically visible in all following files, even those in subdirectories. No need for imports whatever. Since it is assumed that the code would be developed mostly in linear fashion, name clashes would be rare. More about name clashes later. 4] Linear ordering would apply to functions inside source files too. Function A declared before function B cannot call B. 5] Since sometimes there mutual recursion between the functions,
6] The language would allow super-easy replacement of standard library. No need for language tricks or super-advanced tooling support If the project has subdirectory with some magic name (say " One could copy any part of the "old" stdlib there and modify it at will, remove not needed parts, or whatever. 7] Project wide configuration constants would be placed into the very first file of the project. Name of the file would be fixed (e.g. "
This way programmer is hinted to centralize project-wide settings into one, easy to spot place. No more chasing for important constants hidden somewhere deep in the source. This configuration file would be allowed to contain only compile time constants, not an ordinary code. Compiler would make sure of this. 8] Heavy emphasize on testing. I already describer it elsewehere. Testing machinery could do a lot of work - automatically check for memory leaks, check that test duration doesn't exceed given limit, run tests in random order, run only recently modified tests to speed the development. Test would not require to invent any new name. This is a big help. Imagine project with thousand of tests. 9] Since having many, many tests takes lot of visual space in source file, I considered feature called "tests companion file". There's source file, and there's another file, with almost the same name, containing only tests, to offload the burden from the regular source file. This test companion file would be able to see internal parts of "parental" regular source file, as if it was included at its end. It could look like:
The "040.my-file.x" could then look like:
Companion file "040.my-file.x.tests" would look like:
This would be handy feature for those obsessed with testing. 10] I planned to have compiler support for traces, for "whitebox" testing. I described it elsewhere. 11] The language would have simple module system. Source file name == module name would be the rule. No exceptions, no renaming. Source file names (and names for subdirectories) would be restricted to alphanumeric characters plus few more. Dot character would be excluded. Names of modules would not need to be unique. If there's ambiguity in a name, the compiler would request to disambiguate it with module name. If there's still ambiguity, the compiler would request to add more parts to the the module path. Say for:
if one calls function
according to the need to resolve the name. The shortest valid form is preferred. This would allow project to be rearranged (directories moved) with minimum of changes. Module could be renamed by renaming the file. 12] Experimental feature I considered for module qualified names. Typically, qualified name looks like: The most important part, I thought about making it the other way: Here the important name-part comes first, and the part keeping compiler happy comes later. 13] To keep the language simple, I planned to offload as much as possible to the tooling (mainly to the envisioned IDE). One biggie was constness. I came to conclusion, that An IDE could discover whether given function parameter is modified inside a function or not. It would then show it to the user. No need to do it manually 14] Another thing that could simplify the language was removing fixed size arrays, like:
Thinking of it, fixed sized arrays are almost never needed in real world code, apart of silly examples. One really wants to have resizable array (vector). (With fast allocator, resizable array is as fast as fixed size array on the stack.) Removing fixed size arrays and replacing them with resizable arrays would make generic code simpler, and it would eliminate many index-out-of-range errors. The compiler could be smart enough to internally convert resizable array into fixed array, when this is safe. 15] The code could contain examples. It would look like:
Examples are different from tests, they are copy/paste friendly snippets, not tricks invoking edge cases. Examples could come in two forms:
Complete examples would be checked by the compiler for validity (but they would not be run like tests). Incomplete examples would be checked as much as feasible, e.g. for unbalanced parenthesis. The goal was to make it easy to spot examples (big lettered There may be some potential to generate documentation using examples. 16] The language would not need any building system. No make, automake, CMake, qmake or other monster. All source files would in a single directory (except for the default standard library, but this is invisible). Nothing like C's One would then say "compile this directory", and that's it. 17] Whole project would be compiled at once. No object files later linked together, no caches for precompiled files, no bug due to stalled files It was hoped that compilation would be fast enough to keep this feasible. Whole project compilation would also allow for many optimizations. 18] Compiler would output C code (in one big file), which would then be processed by C compiler. This avoids the same kind of troubles as above. 19] Generating C code allows for (relatively) easy interfacing with C. One would need to manually write function prototypes for foreign C functions. Missing language features (like E.g. Some C constructs (macros, perhaps function pointers) would not be supported. 20] The language would not have global variables. There could be global constants (possibly generated during compile time), but no modifiable globals. 21] To make life easier when there are no global variables, I thought about providing "global context". This would be a structure, with a well known name, and this structure would be automatically passed into all functions which use it. E.g. if
Tests would declare context of their own. Some parts of global context would be mandatory (e.g. the default-allocator). Libraries would also need some way to use non-mandatory but commonly expected parts of global context. This is described next. 22] "If it compiles, use it!" Code may try something and hope for the best:
This would allow 3rd party libraries to use expected functionality from the context, but if this functionality was not needed, it use is automatically avoided. I hoped that global context and automatic conditional compilation would make reusing someone's else library easier. 23] Sometimes I would want to use 3rd party library, but only very, very small part of it. I do not want unwanted code from the library to interfere with the rest of project, or to be used by mistake. Here's the idea:
Graphically it would look like:
So if foreign library exposed functions from
Nowhere else I am allowed to call into This feature was hoped to make it possible to reuse code without fear. 24] The language would not offer one global heap allocator like C does. Instead, there would be common API, and several implementations (traditional block based allocators; allocators which just increments a pointer (arena allocators) and free() is no-op; their combinations). Having own allocators allows for:
Defensive allocators are really good for error catching. Arena allocators are incredibly fast. 25] assert would not be just an ordinary function. It would be able to show lot of information: The compiler will try to understand meaning of asserted expression. In some cases it should be able to statically check for the code using the information gleaned from assert. E.g. non-nullness:
When the compiler is unable to statically enforce asserted restriction (it gets too complicated fast), the check would stay there for run time, as expected.
26] The language would provide generics, but in simple way, without the usual ceremony. Asserts would be used, instead of complex type system.
Using asserts allow to make very complex requirements for generics, without the need to design complex types, concepts or whatever. Just show the compiler what you want to do with it. 27] Tiny feature to improve code self-documentatation:
This featurette allows one to notice possible trouble during debugging: if the code goes the "wrong" path, it may signal a problem. It serves as one character long documentation hint. It may also help to generate a little bit more efficient code. GCC and clang support feature called 28] Errors. I had covered them elsewhere. The emphasize was on simplicity, not versatility. Complicated errors would need more effort. 29] I planned to support code running at compile time. It could be used e.g. to initialize global constants. Syntax may look e.g:
I do not think compile time code needs to be as powerful as e.g. in Jai language. 30] One of major goals of the language was to reduce number of names (symbols). The less names one needs to invent, the better. This also pushes one to use shorter names, instead of the enforced eloquency of "isDataFlagInSubscriptionTableFullyElaborated". One way to reduce number of names is to allow function overloading. Some people are against function overloading, I never understood why. They seem to think that "FunctionWhichDoesThisAndThatAndCertainlyHasUniqueName" is great for understanding. As long as the compiler can safely resolve which overload to use, everything is permitted. Overload would be resolved:
Advanced IDE would be able to show which overload is used, and what are the alternatives. Tests do not require unique names, or any names at all. Imagine having 10,000 tests ... 31] To keep names simple, I would allow some non aplha-numeric characters to be use in names: - a separator (more readable than underscore), ? at the end for functions returning boolean, # at the end for function returning some kind of count. Perhaps few more. Some rules on characters use may be enforced by the compiler. Expressions using these characters would need to employ spaces, to make a distinction:
32] On the other hand, I woudn't allow UTF8 in code at all. Too dangerous to make a mischievous overload. Unicode has some specification for "safe" characters (http://unicode.org/reports/tr31/), but this is way too complicated. UTF8 in strings may or may not be allowed. I wasn't sure. 33] Since the code is organized linearly, it should be possible to insert new This would allow one to explore foreign codebase, or library, step by step, without the need to set up a new project. By default main in the last file is the used one, but command line switch could select a different main. Theoretically, this feature may also allow to generate more than one application from single codebase (e.g. command line version and GUI version). 34] Named parameters would be allowed. (They were mentioned before, in overloading section.) They could be also used for this trick:
looks clumsy. I would love to have this:
Smart compiler would rearrange tabular form into sequence of calls. Advanced IDE would make creating the table easier. 35] I planned to make compiler smart enough to recognize traditional tricks used in typography to reduce repetitive typing. E.g.
could be written as:
The compiler would notice strange uncompilable thing here, would try to guess if this is a shorthand, if this makes it working, it would apply it quietly. To me this old trick feels more natural than to invent strange and clumsy compiler constructs ( 36] My language would have very simple form of object oriented support. Some people see OOP as bad, but there are dangerous parts (inheritace) and harmless parts (information hiding). There would be two types of objects:
Single level of inheritance prevents one to create monstrous hierarchies., while still providing polymorphism. Potentially, polymorphic objects could replace traditional functions pointers, making the language more simple and robust (functions pointers make deductions about the code hard). Since the whole program is compiled at once, the compiler knows all subclasses, and could support safe and complete switch on them, instead of forcing the developer to add virtual function:
The compiler would make sure there's no class omitted in the switch. Advantage of the switch is, that related code is place in one place, not spread into many files. 37] Structures padding (and alignement) would be explicit, unlike in C. No more guessing what is going on inside.
Bit padding could be implemented in the same way:
The compiler would check against user's mistake - wrong padding, excessive padding. If "wrong" placement is needed:
38] Other thing I wished from the language was reduction of indentation. Code sitting on the left side reads better. There are few tricks that could be used:
could be perhaps written as:
Classes could be perhaps written like:
Some other tricks could be invented. The intention is to improve readability by moving the code left as much as possible. 39] I considered disabling pointer arithmetic by default. This would eliminate a lot of bugs. If pointer arithmetic is needed anyway, there would be some special form for it, clumsier to use by intention. 40] I wanted to have explicit hints what to inline and what not:
41] To make the language simpler, all structs and objects could be passed into function only by pointer parameters. No slicing or copying. Structures could be returned, and it would get RVO optimized. 42] Hello Wold examples and single file utilities could be written in simpler way:
Usual restriction and requirements wouldn't apply on single file projects:
However, there were problems I didn't solved fully: 1] How to implement paralelism. I certainly didn't want anything overcomplicated like Pony. One thing I planned were mutexes that self-check for deadlocks, in debug mode. 2] Debugging. I didn't find easy way to have reasonable debugger for the generated C code. I considered, that for debug mode, I would use bytecode interpreter, instead of the C generator. I do have some idea how to make debugger for interpreter. 3] I postponed selecting the name for the language. I wanted something googleable. 4] Dynamic libraries are problem. They break many assumption about the code, the interfere with the "everything is compiled at once" principle. Either I would not support them, or would restrict their use. For example, it shouldn't be possible to define polymorphic subclass in dynamic library. 5] Function pointers. They have the same problems as above. I though that, perhaps, they could be removed from the language and replaced by (less free to do anything) polymorphic objects. If someone managed to read it all, have my congratulation. I tried to describe novelty features. The rest would be mostly the same as C, with minor improvements and simplifications here or there. Questions welcomed. I guesstimated, that it could take up to decade to implement it. |
Hey, your code generator does not even have a block for D and yet on the compilation speed benchmark page you list D with a comment "segfaults after 6 minutes". What is the credibility of your benchmark? |
I am referring to the benchmark mentioned in hacker news. https://news.ycombinator.com/item?id=19405036 |
Hi @dsnippet No need to be so agressive :) I simply forgot to push the code. Done now. Go also segfaults. Java refuses to compile it. It's all because there's just too much code in one function. I'll improve the test soon. |
@medvednikov Your language description hits the most important points. I have different stylistic opinions in some cases, but eh. If you can deliver what you've said under a liberal open source license, that's what matters. Most importantly, I recommend finalizing some kind of 1.0 that people can rely on as soon as possible, even if it doesn't have every feature you want. If you get bogged down in everyone's suggestions and all the possibilities, that will prevent people from being able to use V soon as a stable product. |
I would not rush a 1.0. That's a great way to lock yourself in to backwards
compatibility issues for a long time.
…On Sat, Mar 16, 2019, 9:50 AM Tom Palmer ***@***.***> wrote:
@medvednikov <https://github.com/medvednikov> Your language description
hits the most important points. I have different stylistic opinions in some
cases, but eh. If you can deliver what you've said under a liberal open
source license, that's what matters. Most importantly, I recommend
finalizing some kind of 1.0 that people can rely on as soon as possible,
even if it doesn't have every feature you want. If you get bogged down in
everyone's suggestions and all the possibilities, that will prevent people
from being able to use V soon as a stable product.
—
You are receiving this because you were mentioned.
Reply to this email directly, view it on GitHub
<#3 (comment)>, or mute
the thread
<https://github.com/notifications/unsubscribe-auth/AD5V_NQrMm5-_XsEBBsXAOhYlCnTFdoIks5vXQSigaJpZM4a-WGd>
.
|
And Jai will be great once it's done, too ... Someday, maybe. |
So far I think it looks great. The syntax is Go-ish, but seems much more interesting to write than Go :). Great job. On a side note, Volt is an awesome project. |
Very small feature: basic types may provide minimum and maximum constants in form:
The dot syntax is just the simplest one. |
Having Gnu MultiPrecision Library.out-of-box in V syntax when we will have to deal with a huge number is extreme useful. |
Writing it here, as I do not have twitter account. There a poll about forcing name style for constants. While this would be desirable most of the time, there are situations standing against mandatory form. SCADA systems, for example. You have thousands of data points, and they all have been assigned name by an outside authority. These names have meaningful internal structure. You are not free to change them or invent your own substitution. |
This issue has been closed due to the inactivity. |
Very interested in your input.
https://github.com/vlang-io/V/blob/master/examples/users.v
https://github.com/vlang-io/V/blob/master/examples/hello_world_gui.v
https://github.com/vlang-io/V/blob/master/examples/generic_repository.v
The text was updated successfully, but these errors were encountered: