Why does java.util.Properties implement Map<Object,Object> and not Map

后端 未结 5 553
死守一世寂寞
死守一世寂寞 2020-12-03 09:43

The java.util.Properties class is meant to represent a map where the keys and values are both Strings. This is because Properties objects are used

相关标签:
5条回答
  • 2020-12-03 09:53

    The reason: Liskov substitution principle and backwards compatibility. Properties extends Hashtable and thus must accept all messages that Hashtable would accept - and that means accepting put(Object, Object). And it has to extend plain Hashtable instead of Hashtable<String, String> because Generics were implemented in the downwards-compatibe way via type erasure, so once the compiler has done its thing, there are no generics.

    0 讨论(0)
  • A one-liner (two-liner for no warnings) for creating Map from Properties:

    @SuppressWarnings({ "unchecked", "rawtypes" })
    Map<String, String> sysProps = new HashMap(System.getProperties());
    
    0 讨论(0)
  • 2020-12-03 10:12

    It was originally intended that Properties would indeed extends Hashtable<String,String>. Unfortunately the implementation of bridge methods caused a problem. Properties defined in such a way causes javac to generate synthetic methods. Properties should define, say, a get method that returns a String but needs to override a method that returns Object. So a synthetic bridge method is added.

    Suppose you had a class written in the bad old 1.4 days. You've overridden some methods in Properties. But what you haven't done is overridden the new methods. This leads to unintended behaviour. To avoid these bridge methods, Properties extends Hashtable<Object,Object>. Similarly Iterable does not return a (read-only) SimpleIterable, because that would have added methods to Collection implementations.

    0 讨论(0)
  • 2020-12-03 10:14

    Backwards compatibility.

    0 讨论(0)
  • 2020-12-03 10:19

    Because they did it in a hurry in the early days of Java, and didn't realise what the implications would be four versions later.

    Generics were supposed to be part of the design of Java from the beginning, but the feature was dropped as being too complicated and, at the time, unnecessary. As a result, lots of code in the standard libraries is written with the assumption of non-generic collections. It took the prototype language "Pizza" from Martin Odersky to show how they could be done fairly well while maintaining near perfect backwards compatibility, with both Java code and bytecode. The prototype led to Java 5, in which the collections classes were retrofitted with generics in a way that allowed old code to keep working.

    Unfortunately, if they were to retroactively make Properties inherit from Map<String, String>, then the following previously valid code would stop working:

    Map<Object, Object> x = new Properties()
    x.put("flag", true)
    

    Why anybody would do that is beyond me, but Sun's commitment to backwards compatibility in Java has gone beyond heroic into the pointless.

    What's now appreciated by most educated observers is that Properties should never have inherited from Map at all. It should instead wrap around Map, exposing only those features of Map that make sense.

    Since reinventing Java, Martin Odersky has gone on to create the new Scala language, which is cleaner, inherits fewer mistakes, and breaks new ground in a number of areas. If you're finding Java's niggles annoying, take a look at it.

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