- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
给出下面的代码片段,对于 POJO 类的每个字段,有没有办法检查类型是否是整数列表?这里真正的问题是类型参数,因为通过 instanceof
或 isAssignableFrom
很容易检查它是否是一个列表。
main
中的行是我目前所发现的,但它不适用于那些具有更复杂类层次结构的类型。
public class Test {
public static void main(String[] args) throws NoSuchFieldException {
// to check if field f1 is an integer list
Field field = Pojo.class.getField("f1");
if (List.class.isAssignableFrom(field.getType())) {
Type type = field.getGenericType();
if (type instanceof ParameterizedType) {
ParameterizedType pt = (ParameterizedType) type;
if (pt.getActualTypeArguments()[0].equals(Integer.class)) {
System.out.println("Yes, it's an integer list");
}
}
}
}
public static class Pojo {
public List<Integer> f1;
public ArrayList<Integer> f2;
public C1 f3;
public C2<Integer> f4;
public C3<Integer> f5; // notice, this is not an integer list, it's a string list
public C4<Integer> f6;
public C5<String, Integer> f7;
}
public static class C1 implements List<Integer> {...}
public static class C2<T> implements List<T> {...}
public static class C3<T> implements List<String> {...}
public static class C4<T> implements List<Integer> {...}
public static class C5<T1, T2> implements List<T2> {...}
}
最佳答案
你是对的,因为删除,这不是一个容易的问题。但是,我认为这并非无法解决。
基本原则应该是一直到List<E>
都遵循并用实际参数类型替换类型变量。 .
考虑这两个类:
class B<T> extends AbstractList<T>{}
class A<E> extends B<E>{}
B
声明中使用的类型变量名是T
,而当它从 A
中扩展时, E
用来。因此,我们不能依赖声明的名称。
Field
开始直到我们点击 List<E>
. (我不确定当 List 通过不止一条路径进入层次结构时会发生什么。因此,下面没有处理这种情况。)E
时在 List
最后,我们可以得出结论,如果它是 List<Integer>
还是不是。这是执行此操作的代码。这是相当多的行,但适用于您添加的案例(以及我这边的更多案例)。实现过程中可能存在一些缺陷,但这种方法应该有效。
public class ParameterizedTypeHierarchy{
/* When a conclusion is reached - negative or positive - about a field, one of these exceptions is thrown to
* quickly return to the main(). */
private static class NotListOfIntegerException extends RuntimeException{
private Type elementType;
public NotListOfIntegerException( Type elementType ){
this.elementType = elementType;
}
}
private static class IsAListOfIntegerException extends RuntimeException{}
public static void main( String[] args ) throws NoSuchFieldException{
List<String> fields = Arrays.asList( "f0", "f1", "f2", "c1", "c2", "c3", "c4",
"c5", "c5Neg1", "c6", "c6Neg1", "c7", "c7Neg1", "c7Neg2" );
for( String f : fields ){
// to check if field f1 is an integer list
Field field = Pojo.class.getField( f );
try{
check( field );
}
catch( IsAListOfIntegerException e ){
System.out.println( f + " (" + field.getType().getSimpleName() + ") is a List<Integer>" );
}
catch( NotListOfIntegerException e ){
if( e.elementType == null ) System.out.println( f + " (" + field.getType().getSimpleName() + ") is NOT a List." );
else System.out.println( f + " (" + field.getType().getSimpleName() + ") is NOT a List<Integer>. It is List<" + e.elementType.getTypeName() + ">." );
}
catch( Exception e ){
e.printStackTrace();
}
}
}
private static boolean check( Field f ){
Type type = f.getGenericType();
if( type instanceof ParameterizedType ){
/* Parameterized type field. */
if( isList( (ParameterizedType) type, null, null ) ) return true;
}
else if( type instanceof Class ){
/* Field's type takes no parameters. */
return fromClass( (Class<?>) type );
}
return false;
}
private static boolean fromClass( Class<?> type ){
/* For class there are ways in which a List could be one of its parents: interface implementations
* or the super class indirectly being a child of List. */
Type[] intfs = type.getGenericInterfaces();
if( intfs != null && intfs.length > 0 ){
for( Type intf : intfs ){
if( intf instanceof ParameterizedType && isList( (ParameterizedType) intf, type, null ) ) return true;
}
}
Type st = type.getGenericSuperclass();
/* If there is no super class or the super is Object, we can conclude in the negative. */
if( st == null || Object.class == st ) throw new NotListOfIntegerException( null );
if( st instanceof ParameterizedType ){
if( isList( (ParameterizedType) st, type, null ) ) return true;
}
return false;
}
private static boolean isList( ParameterizedType pt, Class<?> extendingEntity, Type[] types ){
/* This parameterized type needs to be a list. Else return. */
Type raw = pt.getRawType();
if( raw instanceof Class && !List.class.isAssignableFrom( (Class<?>) raw ) ) return false;
Type[] listParamTypes = pt.getActualTypeArguments();
/* If this is directly List.class, then we can decide here itself. */
if( raw == List.class ) return listParamTypes[ 0 ] == Integer.class;
/* This is a parameterized class that implements List. However, the extending class's parameter(s) need not be for
* List. Hence, we have get the RIGHT parameters for the parameterized type and check against them. */
Type[] replaced = replaceTypeVars( pt, extendingEntity, types );
Class<?> c = (Class<?>) raw;
return classExtList( c, replaced );
}
private static Type[] replaceTypeVars( ParameterizedType pt, Class<?> impl, Type[] types ){
Map<String, Type> repl = replacements( impl, types );
Type raw = pt.getRawType();
Type[] rawTypeParams = null;
if( raw instanceof Class<?> ){
Class<?> c = (Class<?>) raw;
Type[] actual = pt.getActualTypeArguments();
if( actual == null || actual.length == 0 ) return null;
rawTypeParams = new Type[ actual.length ];
for( int i = 0; i < actual.length; i++ ){
Type tv = actual[ i ];
Type val = null;
if( !( tv instanceof TypeVariable ) ){
rawTypeParams[ i ] = actual[ i ];
}
else{
if( ( val = repl.get( ( (TypeVariable<?>) tv ).getName() ) ) == null )
rawTypeParams[ i ] = tv;
else
rawTypeParams[ i ] = val;
}
}
}
return rawTypeParams;
}
/* Replaces the type variables declared on a class with the actual parameters passed. */
private static Map<String, Type> replacements( Class<?> c, Type[] types ){
if( c == null ) return Collections.emptyMap();
TypeVariable<?>[] tps = c.getTypeParameters();
if( tps == null || tps.length == 0 ) return Collections.emptyMap();
Map<String, Type> map = new HashMap<>();
for( int i = 0; i < tps.length; i++ ){
/* Skip TypeVariable instances. We want only replaced ones. */
if( types[ i ] instanceof TypeVariable ) continue;
TypeVariable<?> tv = tps[ i ];
map.put( tv.getName(), types[ i ] );
}
return map;
}
private static boolean classExtList( Class<?> c, Type[] types ){
TypeVariable<?>[] params = c.getTypeParameters();
Type[] intfs = c.getGenericInterfaces();
for( Type intf : intfs ){
/* If this interface is List and is taking Integer as param, great. */
if( intf instanceof ParameterizedType ){
ParameterizedType pt = (ParameterizedType) intf;
if( pt.getRawType() == List.class ){
/* If the type argument is Integer.class, we are done. */
if( pt.getActualTypeArguments()[ 0 ] == Integer.class ) throw new IsAListOfIntegerException();
/* Type argument is the type variable itself. We have to check which type variable
* was passed to List. */
for( int i = 0; i < params.length; i++ ){
if( params[ i ].getName().equals( pt.getActualTypeArguments()[ 0 ].getTypeName() ) ){
if( types[ i ] == Integer.class ) throw new IsAListOfIntegerException();
throw new NotListOfIntegerException( types[ i ] );
}
}
}
}
}
Type st = c.getGenericSuperclass();
if( st instanceof ParameterizedType ) return isList( (ParameterizedType) st, c, types );
return false;
}
public static class Pojo {
public Object f0;
public List<Integer> f1;
public ArrayList<Integer> f2;
public C1 c1;
public C2<Integer> c2;
public C3<Integer> c3; // notice, this is not an integer list, it's a string list
public C4<Integer> c4;
public C5<String, Integer> c5;
public C5<String, String> c5Neg1;
public C6<String, Integer> c6;
public C6<String, String> c6Neg1;
public C7<String, Integer> c7;
public C7<String, String> c7Neg1;
public C7<String, ?> c7Neg2;
}
private static interface A<E>{}
private static class B<T> extends AbstractList<T> {
@Override
public T get( int index ){ return null; }
@Override
public int size(){ return 0; }
}
public static class C1 extends B<Integer> {}
public static class C2<T> extends B<T> {}
public static class C3<T> extends B<String> {}
public static class C4<T> extends B<Integer> {}
public static class C5<T1, T2> extends B<T2> {}
public static class C6<T1, T2> implements List<T2> {...}
public static class C7<T1, T2> implements List<T2>, A<T1>{...}
}
关于java - 如何检查一个 ParameterizedType 是否代表另一个 ParameterizedType 的子类型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/70105878/
我正在尝试在我的 UITableView 上调用 reloadData。我在没有界面生成器的情况下制作我的应用程序。 此代码适用于 Interface Builder,但似乎没有。 当我将我的代码与我
有人可以解释 UIAlertView 的委托(delegate)是如何工作的吗?它是自动调用还是我必须调用它?例如: - (void)alertView:(UIAlertView *)alertVie
编辑:好吧,根据其他人的建议,我创建了一个最小的示例......并且它有效,因此我将在未来与任何人分享它。这是工作代码: #include #include using namespace std
unicode 是 ,它被用在 XML 文档中。 最佳答案 查看图表:unicodelookup.com 换行符。 关于html - unicode字符是什么 代表?,我们在Stack Overflo
我有一个应用程序,可以以编程方式在配置的 Facebook 页面上发帖。我的应用程序显然已批准管理页面和发布页面权限,并且我正在使用页面访问 token 从现在开始一切正常,但最近当我在页面提要上发布
代表 NCAA 男子篮球分组的最佳数据库模式是什么?如果您不熟悉,请点击以下链接:http://www.cbssports.com/collegebasketball/mayhem/brackets/
所以我一直在阅读这个关于如何使用 Frida 的教程:https://www.frida.re/docs/functions/我遇到过以下情况: $ ./client 127.0.0.1 connec
委托(delegate)函数返回之前是否需要调用replyHandler?我需要进行几次 Web 服务 API 调用才能回复,以下实现正确吗? func session(_ session: WCSe
下面提到的是我的 textField 委托(delegate)方法,我正在使用 IQKeyBoardSwift 作为智能键盘。我尝试移除我的键盘,但我仍然没有收到任何关于接受“开始触摸”的方法的调用
我有一个表格 View ,其中几乎没有用于数据输入的文本字段和弹出窗口。我想将其中一些表示为强制性的。我不知道如何讨厌星号。任何帮助将不胜感激。 最佳答案 我认为你可以使用自定义 UITableVie
例如,我知道如何使用 numpy 对数组进行切片 v[1, :, :] 现在我想要一个函数将切片 (1,1,None) 作为输入并返回 v[1,:,:] 问题是我不知道如何表示省略号 最佳答案 您可以
修订... 应用程序的关键是与数据库服务器通信。服务器对应用程序的响应都是 XML 格式的。有几个屏幕。例如,屏幕 1 列出了用户的信息,屏幕 2 列出了用户过去的交易,允许新交易,等等。 这是我的
我想知道映射/表示内存的最佳方式是什么。我的意思是,例如,如何描述一个结构及其所有字段都被序列化。 我正在创建一个 RPC 库,它将使用 dwarf 调试数据创建客户端和服务器,因此我需要创建一个函数
如果我有一个实现了两个协议(protocol)的 View Controller : @interface CustomerOperationsViewController : UIViewContr
在 Objective-C 中我可以做这样的事情: @property (nonatomic, weak) id someObject; 如何在swift中做到这一点?我试过这个: let someO
我成功地使用了相当棒的 connection:didReceiveAuthenticationChallenge: NSURLConnectionDelegate 委托(delegate)方法。很酷。
我正在寻找原始数据类型的 @NonNull 等效 Java 注释。我知道原始数据不能为 null,但我找不到替代方法。 我想要实现的在逻辑上等同于: int mPageNumber; public v
我正在学习 Git,如果我能描述代表 Git 存储库的数学结构,那就太好了。例如:它是一个有向无环图;它的节点代表提交;它的节点有代表分支等的标签(每个节点最多一个标签,没有标签使用两次)。(我知道这
我看过很多与委托(delegate)相关的帖子,我想知道引用它们的正确方法。假设我有一个声明如下的对象: @interface MyViewController : UITableViewContro
我有这个类: public class Order { int OrderId {get; set;} string CustomerName {get; set;} } 我也声明下面的变
我是一名优秀的程序员,十分优秀!