How to access java-classes in the default-package?
Asked Answered
C

5

109

I'm working now together with others in a grails project. I have to write some Java-classes. But I need access to an searchable object created with groovy. It seems, that this object has to be placed in the default-package.

My question is: Is there a way to access this object in the default-package from a Java-class in a named package?

Couturier answered 12/11, 2008 at 12:36 Comment(0)
G
136

You can’t use classes in the default package from a named package.
(Technically you can, as shown in Sharique Abdullah's answer through reflection API, but classes from the unnamed namespace are not in scope in an import declaration)

Prior to J2SE 1.4 you could import classes from the default package using a syntax like this:

import Unfinished;

That's no longer allowed. So to access a default package class from within a packaged class requires moving the default package class into a package of its own.

If you have access to the source generated by groovy, some post-processing is needed to move the file into a dedicated package and add this "package" directive at its beginning.


Update 2014: bug 6975015, for JDK7 and JDK8, describe an even stricter prohibition against import from unnamed package.

The TypeName must be the canonical name of a class type, interface type, enum type, or annotation type.
The type must be either a member of a named package, or a member of a type whose outermost lexically enclosing type is a member of a named package, or a compile-time error occurs.


Andreas points out in the comments:

"why is [the default package] there in the first place? design error?"

No, it's deliberate.
JLS 7.4.2. Unnamed Packages says: "Unnamed packages are provided by the Java SE platform principally for convenience when developing small or temporary applications or when just beginning development".

Gerard answered 12/11, 2008 at 12:44 Comment(18)
As I've explained to other people before, the default package is a second-class citizen in the Java world. Just Don't Do That. :-)Pizor
Seems I have to learn Groovy (on a very time-constrained project) or code in the default-package. :-(Couturier
@ChrisJester-Young if so why is it there in the first place? design error?Cirone
Why it was deprecated?Wager
@SuzanCioc do you mean: "why after J2SE1.4, can you no longer access the default package?"Gerard
@Gerard yes this is what I meanWager
@SuzanCioc Because classes from the unnamed namespace are not in scope in an import declaration: the risk of naming collision was simply too important in that unnamed namespace.Gerard
@Gerard what is the principal difference between "root" package and any other top-level packages, like com or org? May be let us prohibit to import them too?Wager
@SuzanCioc com or org are based on a naming convention, as opposed to the "root" package, which is part of the language specification. You have to trust conventions, but the language can evolve its specification.Gerard
I agree that both conventions ans specifications can evolve. So if we judge to evolve root package specification because of dangerous name clash, then we can evolve conventions on the same reason: NEVER put your class into one-level named package or you will be in danger!Wager
@SuzanCioc I agree. The difference, though, is that conventions evolve by their users, while specifications evolves by the language designers: two different communities, both in scale and in decision process.Gerard
Downvote because although I agree with the sentament, Sharique's answer is correct and this one is wrong.Luzon
@BillK Good point. i have included a reference to Sharique's answer. I have also documented a recent (2014) evolution of JLS8 (Java Language Specification) reinforcing the fact a type must be a member of a type package.Gerard
Thanks, -1 removed... I wonder how the JLS change will affect Groovy scripts because they are generally in the default package (It wouldn't make much sense to have them anywhere else). I suppose groovy could assign a fake package automatically--that would be no more hacky than anything else in groovy :)Luzon
@VonC, Aside from reflection, what about the new invocation mechanisms?Cirone
@VonC, Yep, I heard they gave dynamic coding a complete rehaul, making scripting languages like JS, Scala, and Kotlin First class.Cirone
@Cirone "why is [the default package] there in the first place? design error?" No, it's deliberate. JLS 7.4.2. Unnamed Packages says: "Unnamed packages are provided by the Java SE platform principally for convenience when developing small or temporary applications or when just beginning development".Itchy
@Itchy Thank you for this feedback. I have included your comment in the answer for more visibility.Gerard
T
63

In fact, you can.

Using reflections API you can access any class so far. At least I was able to :)

Class fooClass = Class.forName("FooBar");
Method fooMethod = fooClass.getMethod("fooMethod", String.class);

String fooReturned = (String)fooMethod.invoke(fooClass.newInstance(), "I did it");
Timbre answered 18/2, 2009 at 14:10 Comment(3)
For Scala, the following code works: val bar = "hi"; val fooClass = Class.forName("FooClass"); val fooMethod = fooClass.getMethod("foo", classOf[Array[String]]); val fooReturned = fooMethod.invoke(fooClass.newInstance(), Array(bar));Radiotransparent
Great answer, +1. Since the question was groovy, however, you can use duck typing (like Duct taping I guess)... Class.forName("FooBar").newInstance().fooMethod("I did it")Luzon
Well, if I have two classes named FooBar from two differents jar files, which one will be called?Ayotte
M
7

Use jarjar to repackage the jar file with the following rule:

rule * <target package name>.@1

All classes in the default package of the source jar file will move to the target package, thus are able to access.

Mcneese answered 13/3, 2014 at 2:37 Comment(2)
a demo of how to use jarjar would be greatAricaarick
@BiGGZ: See the README.md file on the jarjar github repo to see how to use jarjar with ant, gradle or the command line.Mackintosh
K
3

You can use packages in the Groovy code, and things will work just fine.

It may mean a minor reorganization of code under grails-app and a little bit of a pain at first, but on a large grails project, it just make sense to organize things in packages. We use the Java standard package naming convention com.foo.<app>.<package>.

Having everything in the default package becomes a hindrance to integration, as you're finding.

Controllers seem to be the one Grails artifact (or artefact) that resists being put in a Java package. Probably I just haven't figured out the Convention for that yet. ;-)

Kumquat answered 12/11, 2008 at 14:12 Comment(0)
P
1

just to complete the idea:

From inside default-package you can access objects resided in named packages.

Pronouncement answered 26/4, 2020 at 8:10 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.