Retrofitting void methods to return its argument to facilitate fluency: breaking change?

后端 未结 3 1677
抹茶落季
抹茶落季 2020-12-24 15:00

\"API design is like sex: make one mistake and support it for the rest of your life\" (Josh Bloch on twitter)

There

相关标签:
3条回答
  • 2020-12-24 15:35

    If you have to deal with source compatability only, then just go ahead. Changing from void to a return type will not break.

    But to address the thing you actually want to do: I consider the problem with fluent interfaces is that the lines tend to get rather long and - after formatting - somewhat unreadable. For builders it works fine, but I'd probably not use it for anything else.

    Is this for playing with it, or because you've found that this is really great?

    0 讨论(0)
  • 2020-12-24 15:37

    If you can't retrofit, you still can wrap your class into a new one which uses the same methods but with correct returns (MyClassFluent). Or you can add new methods but with different names, instead of Arrays.sort() we could have Arrays.getSorted().

    I think that the solution isn't to force things, just deal with them.

    EDIT: I know I didn't answer to the "retrofitting of void methods" question, but your answer is already really clear.

    0 讨论(0)
  • 2020-12-24 15:47

    Unfortunately, yes, changing a void method to return something is a breaking change. This change will not affect source code compatibility (i.e. the same Java source code would still compile just like it did before, with absolutely no discernible effect) but it breaks binary compatibility (i.e. bytecodes that was previously compiled against the old API will no longer run).

    Here are the relevant excerpts from the Java Language Specification 3rd Edition:

    13.2 What Binary Compatibility Is and Is Not

    Binary compatibility is not the same as source compatibility.


    13.4 Evolution of Classes

    This section describes the effects of changes to the declaration of a class and its members and constructors on pre-existing binaries.

    13.4.15 Method Result Type

    Changing the result type of a method, replacing a result type with void, or replacing void with a result type has the combined effect of:

    • deleting the old method, and
    • adding a new method with the new result type or newly void result.

    13.4.12 Method and Constructor Declarations

    Deleting a method or constructor from a class may break compatibility with any pre-existing binary that referenced this method or constructor; a NoSuchMethodError may be thrown when such a reference from a pre-existing binary is linked. Such an error will occur only if no method with a matching signature and return type is declared in a superclass.

    That is, while the return type of a method is ignored by the Java compiler at compile-time during the method resolution process, this information is significant at run-time at the JVM bytecode level.


    On bytecode method descriptors

    A method's signature does not include the return type, but its bytecode descriptor does.

    8.4.2 Method Signature

    Two methods have the same signature if they have the same name and argument types.


    15.12 Method Invocation Expressions

    15.12.2 Compile-Time Step 2: Determine Method Signature

    The descriptor (signature plus return type) of the most specific method is one used at run time to perform the method dispatch.

    15.12.2.12 Example: Compile-Time Resolution

    The most applicable method is chosen at compile time; its descriptor determines what method is actually executed at run time.

    If a new method is added to a class, then source code that was compiled with the old definition of the class might not use the new method, even if a recompilation would cause this method to be chosen.

    Ideally, source code should be recompiled whenever code that it depends on is changed. However, in an environment where different classes are maintained by different organizations, this is not always feasible.

    A little inspection of the bytecode will help clarify this. When javap -c is run on the name shuffling snippet, we see instructions like this:

    invokestatic java/util/Arrays.asList:([Ljava/lang/Object;)Ljava/util/List;
                 \______________/ \____/ \___________________/\______________/
                    type name     method      parameters        return type
    
    invokestatic java/util/Collections.shuffle:(Ljava/util/List;)V
                 \___________________/ \_____/ \________________/|
                       type name        method     parameters    return type
    

    Related questions

    • java: what is this: [Ljava.lang.Object;?

    On non-breaking retrofitting

    Let's now address why retrofitting a new interface or a vararg, as explained in Effective Java 2nd Edition, does not break binary compatibility.

    13.4.4 Superclasses and Superinterfaces

    Changing the direct superclass or the set of direct superinterfaces of a class type will not break compatibility with pre-existing binaries, provided that the total set of superclasses or superinterfaces, respectively, of the class type loses no members.

    Retrofitting a new interface does not cause the type to lose any member, hence this does not break binary compatibility. Similarly, due to the fact that varargs is implemented using arrays, this kind of retrofitting also does not break binary compatibility.

    8.4.1 Formal Parameters

    If the last formal parameter is a variable arity parameter of type T, it is considered to define a formal parameter of type T[].

    Related questions

    • Difference between double… and double[] in formal parameter type declaration

    Is there absolutely no way to do this?

    Actually, yes, there is a way to retrofit a return value on previously void methods. We can't have two methods with the same exact signatures at the Java source code level, but we CAN have that at the JVM level, provided that they have different descriptors (due to having different return types).

    Thus we can provide a binary for e.g. java.util.BitSet that has methods with both the void and non-void return types simultaneously. We only need to publish the non-void version as the new API. In fact, that's the only thing we can publish at the API, since having two methods with the exact same signature is illegal in Java.

    This solution is a terrible hack, requiring special (and spec-defying) processing to compile BitSet.java to BitSet.class, and thus it may not be worth the effort to do so.

    0 讨论(0)
提交回复
热议问题