1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2011 The Android Open Source Project 3*795d594fSAndroid Build Coastguard Worker * 4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*795d594fSAndroid Build Coastguard Worker * 8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*795d594fSAndroid Build Coastguard Worker * 10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*795d594fSAndroid Build Coastguard Worker * limitations under the License. 15*795d594fSAndroid Build Coastguard Worker */ 16*795d594fSAndroid Build Coastguard Worker import java.lang.reflect.*; 17*795d594fSAndroid Build Coastguard Worker import java.util.Arrays; 18*795d594fSAndroid Build Coastguard Worker 19*795d594fSAndroid Build Coastguard Worker class NarrowingTest { 20*795d594fSAndroid Build Coastguard Worker 21*795d594fSAndroid Build Coastguard Worker interface I1 { foo()22*795d594fSAndroid Build Coastguard Worker public Object foo(); 23*795d594fSAndroid Build Coastguard Worker } 24*795d594fSAndroid Build Coastguard Worker 25*795d594fSAndroid Build Coastguard Worker interface I2 extends I1 { 26*795d594fSAndroid Build Coastguard Worker // Note that this method declaration narrows the return type. 27*795d594fSAndroid Build Coastguard Worker @Override foo()28*795d594fSAndroid Build Coastguard Worker public String foo(); 29*795d594fSAndroid Build Coastguard Worker } 30*795d594fSAndroid Build Coastguard Worker main(String[] args)31*795d594fSAndroid Build Coastguard Worker public static void main(String[] args) { 32*795d594fSAndroid Build Coastguard Worker I2 proxy = (I2) Proxy.newProxyInstance(NarrowingTest.class.getClassLoader(), 33*795d594fSAndroid Build Coastguard Worker new Class<?>[] { I2.class }, 34*795d594fSAndroid Build Coastguard Worker new InvocationHandler() { 35*795d594fSAndroid Build Coastguard Worker int count = 0; 36*795d594fSAndroid Build Coastguard Worker @Override 37*795d594fSAndroid Build Coastguard Worker public Object invoke(Object proxy, Method method, 38*795d594fSAndroid Build Coastguard Worker Object[] args) throws Throwable { 39*795d594fSAndroid Build Coastguard Worker System.out.println("Invocation of " + method); 40*795d594fSAndroid Build Coastguard Worker if (count == 0) { 41*795d594fSAndroid Build Coastguard Worker count++; 42*795d594fSAndroid Build Coastguard Worker return "hello"; 43*795d594fSAndroid Build Coastguard Worker } else { 44*795d594fSAndroid Build Coastguard Worker return Integer.valueOf(1); 45*795d594fSAndroid Build Coastguard Worker } 46*795d594fSAndroid Build Coastguard Worker } 47*795d594fSAndroid Build Coastguard Worker }); 48*795d594fSAndroid Build Coastguard Worker Main.registerProxyClassName(proxy.getClass().getCanonicalName()); 49*795d594fSAndroid Build Coastguard Worker 50*795d594fSAndroid Build Coastguard Worker Method[] methods = proxy.getClass().getDeclaredMethods(); 51*795d594fSAndroid Build Coastguard Worker Arrays.sort(methods, new MethodComparator()); 52*795d594fSAndroid Build Coastguard Worker System.out.println("Proxy methods: " + 53*795d594fSAndroid Build Coastguard Worker Main.replaceProxyClassNamesForOutput(Arrays.deepToString(methods))); 54*795d594fSAndroid Build Coastguard Worker 55*795d594fSAndroid Build Coastguard Worker System.out.println("Invoking foo using I2 type: " + proxy.foo()); 56*795d594fSAndroid Build Coastguard Worker 57*795d594fSAndroid Build Coastguard Worker I1 proxyAsParent = proxy; 58*795d594fSAndroid Build Coastguard Worker System.out.println("Invoking foo using I1 type: " + proxyAsParent.foo()); 59*795d594fSAndroid Build Coastguard Worker 60*795d594fSAndroid Build Coastguard Worker try { 61*795d594fSAndroid Build Coastguard Worker proxy.foo(); 62*795d594fSAndroid Build Coastguard Worker System.out.println("Didn't get expected exception"); 63*795d594fSAndroid Build Coastguard Worker } catch (ClassCastException e) { 64*795d594fSAndroid Build Coastguard Worker // With an I2 invocation returning an integer is an exception. 65*795d594fSAndroid Build Coastguard Worker System.out.println("Got expected exception"); 66*795d594fSAndroid Build Coastguard Worker } 67*795d594fSAndroid Build Coastguard Worker 68*795d594fSAndroid Build Coastguard Worker System.out.println("Proxy narrowed invocation return type passed"); 69*795d594fSAndroid Build Coastguard Worker } 70*795d594fSAndroid Build Coastguard Worker } 71