-
Notifications
You must be signed in to change notification settings - Fork 1k
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
namespace internal access modifier #2497
Comments
The CLR doesn't provide special accessibility like that. It only works from nested type back to parent type(s). Also, the use of public nested types in this fashion is explicitly against the language design patterns: https://msdn.microsoft.com/en-us/library/ms229027.aspx It's more appropriate to define a public top-level interface and have a private nested implementation. |
Could work though my use case would be for a struct so interface would box. Alternative would be for separate type and use internal public class Thing
{
private ThingsThing _thing;
public ThingsThing MyThing => _thing;
}
public struct ThingsThing
{
public object Stuff { get; internal set; }
public void DoThings();
internal void DoThatThing();
} But then it has assembly level visibility rather than class level 😿 |
Maybe a assembly namespace level visibility would work, so |
The CLR doesn't offer that form of accessibility either. At best the compiler could enforce it internally but the member would still be just |
An example use-case is The constructor is |
@benaadams In that case the type itself could be defined as |
@alrz but then you can pass it back in a public function like |
@benaadams I see. This could use the EDIT: However, you can't prevent calling default ctor in a value type anyways. |
Enforcing that an |
I would like to support this feature. And would like it to use only The usefulness of this is just like And I think it should not need CLR support. Just compile time error |
Most of the time, types in different namespaces are (and IMO should be) loosely coupled and a "namespace internal" access modifier would definitely help to achieve that. |
I think many of the scenarios can be achieved with the new file-local types in C# 11, but I too really wish this were a feature! We now have assembly scoped types and file scoped types, now we just need namespace scoped types! :) File scoped types dont work if you want to have the class be public but just the constructor be file/namespace scoped. |
For me the use case of internal namespace modifier allows for gradual separation of code into separate projects or NuGets. Let's say I have some 100 services that all have the same structure: few public interfaces and models, IOC registration and the rest is completely internal to the service. In the end these services should really become NuGets but that requires:
This requires too much time and work. So the other option is to simply keep the services as different projects in the solution - which is what I'm doing now. 100 projects is not too many but the performance hit in Visual Studio is definitely there when opening or closing the solution. But with internal namespaces? I could have everything in a single project from the start, no need to suffer the performance hit with 100 projects. I can just have 100 folders in a single project. The "hidden internals" of my services would just stay hidden in its internal namespace. //public part of the service
namespace SomeServiceNamespace
{
public interface ISomeService
{
Task DoStuffAsync();
}
}
//hidden part of the service
internal namespace SomeServiceNamespace.Internals
{
public class ServiceImplementation : ISomeService
{
public async Task DoStuffAsync()
{
//...
}
}
} When there's enough resources (meaning development time and/or money) the code from internal namespaces can be moved not to a different project but to a completely different repository and become a NuGet in our local NuGet store with its own CI/CD. |
Genuine question: what purpose does that "hidden internals" have within these services? The reason I ask is that encapsulation to me means "stuff hidden away from the public eye". It's implementation details that I'm free to modify whenever I want without causing changes to the public API surface of my assemblies. Whether those implementation details are private or internal makes very little difference. If the thing exists in a class, I'll mark it private as that's what's available. If private didn't exist, I'd mark it internal and I'd really not care. It's all internal to the the assembly and so hidden away, it doesn't make any difference. The only way this becomes a concern is if one steps onto the InternalsVisibleTo slippery slope. At that point internal stuff is now public too and so there's a need for a way of saying "really internal to" (which is the important part of this thread to me). So what is the user case behind you wanting to hide some internal parts of an assembly from other parts? It all seems needlessly complicated to me. |
Say you want to have service interface ISomeService, it's some external dependency, whatever. You know it's gonna have at least two or three implementations. This is pretty normal in .NET MAUI when you are building something for multiple platforms. But also on backend when you mock your services for regression testing, etc. Now you want to follow this convention for namespaces:
Then
So what, right? This is what Microsoft normally does. My options are:
The "ugly" thing is that Of course you can use private classes but for that to work everything from a given implementation must be inside a private class of some other wrapper class which is also very ugly.
The problem persists.
This is ideal solution but as I said in the post above, this requires too much work to set up. With internal namespaces I could have nicer codebase with compiler checking that some implementations don't affect each other in any way. Since it's just a matter of ugly/nice code it's probably not justified enough to be implemented...but I thought that about required properties as well :) |
I think this is a duplicate of this: #6794 |
Dupe of #8848. Closing out. |
edited
internal
is assembly wide but sometimes I want the scope to be narrower; something likenamespace internal
so it isn't visible to the entire project, but just items in that namespace.When the assembly is created this could translate just to
internal
--- old ---
Thought this was part of
private protected
but doesn't seem to beI'd like a way of mark something of a nested type public to the enclosing type but private for everything else (or protected if visible to derived types)
The text was updated successfully, but these errors were encountered: