泛型(一)

集合中可以存储任意类型对象,但是在取出时,如果要使用具体对象的特有方法时,需要进行向下转型,如果存储的对象类型不一致,在转型过程中就会出现ClassCastException异常。这样就给程序带来了不安全性

在jdk1.5以后就有了解决方案——泛型技术:在存储元素时,就不允许存储不同类型的元素。存储了就编译失败。 所以就需要在存储元素时,在容器上明确具体的元素类型,这其实和数组定义很像。


好处:

1)将运行时期的ClassCastException异常转移到了编译时期,进行检查,并以编译失败来体现。 这样有利于程序员尽早解决问题

2)避免了向下转型(强转)的麻烦。




package cn.hncu.generic.one;

import java.util.ArrayList;
import java.util.List;

public class MySet<E> {
	private List<E> list;//必须用Object
	public MySet() {
		list=new ArrayList<E>(0);
	}
	public boolean contain(E e){
		if(list.size()==0){
			return false;
		}
		Object objs[]=list.toArray();
		for(Object obj:objs){
			if(obj.equals(e)){//注意要写equal方法
				return true;
			}
		}
		return false;
	}
	public boolean add(E e){
		if(contain(e)){
			return true;
		}
		list.add(e);
		return false;
	}
	public E popLast(){
		if(list.size()==0){
			return null;
		}
		return list.remove(list.size()-1);
	}
	public E popFirst(){
		if(list.size()==0){
			return null;
		}
		return list.remove(0);
	}
	public List<E> getAll(){
		return list;
	}
}



package cn.hncu.generic.one;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

public class Demo1 {

	public static void main(String[] args) {
//		test();
//		test2();
		test3();
	}
	//演示泛型的好处之一:没采用泛弄,对元素的处理不安全。泛型可以把运行期的错误提前到编译期
	private static void test() {
//		Map map = new HashMap();//无泛型
		Map<Integer, Integer> map = new HashMap<Integer, Integer>();//无泛型
		map.put(1,2);
		map.put(1,2);
		map.put(1,2);//允许重复
		map.put(1,2);
		map.put(1,2);
//		map.put("3", "###");//无泛型时,此处不会报错,但有泛型时会报错。
		Iterator it=map.entrySet().iterator();
//		Iterator<Entry<Integer, Integer>> it=map.entrySet().iterator();
		while(it.hasNext()){
//			Object obj=it.hasNext();
//			String str=(String)obj;
//			str=str.substring(2);
//			System.out.println(str);
			System.out.println(it.next());
		}
		
	}
	
	//演示泛型的好处之二:从集合中读取的数据不需要强转,能够自动识别
	private static void test2() {
		//演示泛型的好处之二:从集合中读取的数据不需要强转,能够自动识别
		List<String> list =new ArrayList<String>();
		list.add("!!");
		list.add("!!!");
		list.add("!");
		list.add("0000");
//		Iterator it = list.iterator();
		Iterator<String> it = list.iterator();
		while(it.hasNext()){
//			String str=(String) it.next();
			String str=(String) it.next();//List有泛型,并且Iterator也加泛型,那么这里就不用强转
			System.out.println(str);
		}
	}
	
	
	//我们自己在MySet中定义泛型,然后在此处进行调用测试
	private static void test3() {//什么元素都可以放
		MySet<String> set=new MySet<String>();
		set.add("###");
		set.add("#!!!");
		set.add("wang");
		set.add("wja");
		System.out.println(set.popFirst());
		System.out.println(set.popLast());
		
		MySet<Integer> set2 = new MySet<Integer>();
		set2.add(100);
		int x=set2.popLast();
		System.out.println(x);
	}

}



在 Flutter 中,种非常强大的特性,允许开发者编写可以处理多种数据类的代码,同时保持类安全性。可以应用于类、方法以及数据结构中,以提高代码的复用性和灵活性。 ### 定义和使用类 可以通过在类名后添加 `<T>` 来定义一个类,其中 `T` 是类参数。例如,可以定义一个简单的类来存储和检索任意类的值: ```dart class Cache<T> { T? _value; void cacheValue(T value) { _value = value; } T? getCachedValue() { return _value; } } ``` 使用时,可以指定具体的类,如 `String` 或 `int`: ```dart void main() { Cache<String> stringCache = Cache<String>(); stringCache.cacheValue("Tom"); print("获取的缓存内容为 ${stringCache.getCachedValue()}"); // 输出: 获取的缓存内容为 Tom Cache<int> intCache = Cache<int>(); intCache.cacheValue(18); print("获取的缓存内容为 ${intCache.getCachedValue()}"); // 输出: 获取的缓存内容为 18 } ``` ### 定义和使用方法 除了类,还可以定义方法。方法允许在方法级别上使用类参数。例如,可以定义一个通用的打印方法: ```dart void printValue<T>(T value) { print("获取的值为 $value"); } ``` 调用时可以传递任意类的参数: ```dart printValue<String>("Tom"); // 输出: 获取的值为 Tom printValue<int>(18); // 输出: 获取的值为 18 ``` ### 与特定类约束 有时,可能需要限制参数的类范围,例如只接受 `Person` 类的子类。这可以通过 `T extends Person` 来实现: ```dart class Person { String name; Person(this.name); } class CacheWithConstraint<T extends Person> { T? _person; void cachePerson(T person) { _person = person; } String? getPersonName() { return _person?.name; } } ``` 使用时,只能传递 `Person` 或其子类的实例: ```dart void main() { CacheWithConstraint<Person> personCache = CacheWithConstraint<Person>(); personCache.cachePerson(Person("Tom")); print("获取的 T extends Person 的 name 字段为 ${personCache.getPersonName()}"); // 输出: 获取的 T extends Person 的 name 字段为 Tom } ``` ### 使用解析 JSON 数据 在 Flutter 中,使用来解析 JSON 数据是种常见的做法,尤其是在处理 API 响应时。通过 `json_serializable` 包,可以轻松实现支持。定义一个类 `BaseBean<T>`,并在注解中启用 `genericArgumentFactories`: ```dart import 'package:json_annotation/json_annotation.dart'; part 'base_bean.g.dart'; @JsonSerializable(genericArgumentFactories: true) class BaseBean<T> { T? data; int? status; String? message; BaseBean({this.data, this.status, this.message}); factory BaseBean.fromJson(Map<String, dynamic> json, T Function(dynamic json) fromJsonT) => _$BaseBeanFromJson(json, fromJsonT); Map<String, dynamic> toJson(Object? Function(T value) toJsonT) => _$BaseBeanToJson(this, toJsonT); } ``` 在解析 JSON 时,可以传递一个具体的类转换函数,例如 `fromJson` 方法: ```dart void main() { final jsonMap = { 'data': {'name': 'Tom', 'age': 20}, 'status': 200, 'message': 'Success', }; final baseBean = BaseBean.fromJson( jsonMap, (json) => Person.fromJson(json), ); print("解析的 JSON 数据中,data 的 name 字段为 ${baseBean.data.name}"); // 输出: 解析的 JSON 数据中,data 的 name 字段为 Tom } ``` ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值