File: OverloadedSetter.java

package info (click to toggle)
openjdk-11 11.0.4%2B11-1
  • links: PTS, VCS
  • area: main
  • in suites: sid
  • size: 757,028 kB
  • sloc: java: 5,016,041; xml: 1,191,974; cpp: 934,731; ansic: 555,697; sh: 24,299; objc: 12,703; python: 3,602; asm: 3,415; makefile: 2,772; awk: 351; sed: 172; perl: 114; jsp: 24; csh: 3
file content (138 lines) | stat: -rw-r--r-- 5,175 bytes parent folder | download | duplicates (16)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
/*
 * Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

import java.beans.BeanInfo;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.Method;

/**
 * @test
 * @bug 8196373
 * @summary behavior of this class is not specified, but it can be used to check
 *          our implementation for accidental changes
 */
public final class OverloadedSetter {

    class AAA {}
    class CCC extends AAA {}
    class BBB extends CCC {}
    class DDD extends CCC {}

    class ZZZ {}

    // DDD will be selected because it is most specific type.
    class ParentADC<T> {
        public void setValue(AAA value) {}
        public void setValue(DDD value) {}
        public void setValue(CCC value) {}
    }
    // DDD will be selected because it is most specific type.
    class ParentACD<T> {
        public void setValue(AAA value) {}
        public void setValue(CCC value) {}
        public void setValue(DDD value) {}
    }
    // DDD will be selected because it is most specific type.
    class ParentDAC<T> {
        public void setValue(DDD value) {}
        public void setValue(AAA value) {}
        public void setValue(CCC value) {}
    }
    // DDD will be selected because it is most specific type.
    class ParentDCA<T> {
        public void setValue(DDD value) {}
        public void setValue(CCC value) {}
        public void setValue(AAA value) {}
    }
    // DDD will be selected because it is most specific type.
    class ParentCAD<T> {
        public void setValue(CCC value) {}
        public void setValue(AAA value) {}
        public void setValue(DDD value) {}
    }
    // DDD will be selected because it is most specific type.
    class ParentCDA<T> {
        public void setValue(CCC value) {}
        public void setValue(DDD value) {}
        public void setValue(AAA value) {}
    }
    // DDD will be selected because it is most specific type and ZZZ will be
    // skipped because it will be placed at the end of the methods list.
    class ParentCDAZ<T> {
        public void setValue(CCC value) {}
        public void setValue(DDD value) {}
        public void setValue(AAA value) {}
        public void setValue(ZZZ value) {}
    }
    // DDD will be selected because it is most specific type which related to
    // the type of getValue(); BBB will be skipped because it is not a type or
    // subclass of the type returned by getValue();
    class ParentDACB<T> {
        public DDD getValue(){return null;}
        public void setValue(AAA value) {}
        public void setValue(DDD value) {}
        public void setValue(CCC value) {}
        public void setValue(BBB value) {}
    }

    public static void main(String[] args) throws Exception {
        test(ParentADC.class);
        test(ParentACD.class);
        test(ParentDAC.class);
        test(ParentDCA.class);
        test(ParentCAD.class);
        test(ParentCDA.class);
        test(ParentCDAZ.class);
        test(ParentDACB.class);
    }

    private static void test(Class<?> beanClass) throws Exception {
        BeanInfo beanInfo = Introspector.getBeanInfo(beanClass, Object.class);
        PropertyDescriptor[] pds = beanInfo.getPropertyDescriptors();
        if (pds.length != 1) {
            throw new RuntimeException("Wrong number of properties");
        }
        PropertyDescriptor pd = pds[0];
        String name = pd.getName();
        if (!name.equals("value")) {
            throw new RuntimeException("Wrong name: " + name);
        }

        Class<?> propertyType = pd.getPropertyType();
        if (propertyType != DDD.class) {
            throw new RuntimeException("Wrong property type: " + propertyType);
        }
        Method writeMethod = pd.getWriteMethod();
        if (writeMethod == null) {
            throw new RuntimeException("Write method is null");
        }
        Class<?>[] parameterTypes = writeMethod.getParameterTypes();
        if (parameterTypes.length != 1) {
            throw new RuntimeException("Wrong parameters " + parameterTypes);
        }
        if (parameterTypes[0] != DDD.class) {
            throw new RuntimeException("Wrong type: " + parameterTypes[0]);
        }
    }
}