Java编程思想学习笔记_4(异常机制,容器)
一、finally语句注意的细节:
当涉及到break和continue语句的时候,finally字句也会得到执行。
public class Test7 {
public static void main(String[] args) {
int i = 0;
while (true) {
try {
i++;
if (i == 3)
break;
} finally {
if (i == 3)
System.out.println("hi");//输出hi
}
}
}
}
二、异常在继承的时候需要注意的细节:
1、异常限制对构造器不起作用,子类构造器可以抛出任意异常,而不必理会基类所抛出的异常(这与方法不同),但是由于基类构造器必须以这样或那样的方式调用,子类构造器的异常说明必须包含基类构造器的异常说明。
class Dad {
public Dad() throws AException {
}
}
class Son extends Dad {
public Son() throws AException,BException/*AException必须包含在声明*/ {
super();
}
}
2、子类方法可以不抛出任何异常,即使是基类定义的异常,因为假使基类的方法抛出了异常,这样做也不会破坏已有的程序,所以也没有问题。
interface AInterface {
public void f() throws AException;
}
interface BInterface {
public void f() throws BException;
}
class Impl implements AInterface,BInterface {
@Override
public void f() {
}
}
3、如果子类对象向上转型为基类的引用,那么在调用方法的时候,编译器就会要求捕获基类可能抛出的异常。同样是上例,在main方法里向上转型为AInterface,那么编译器会要求处理异常:
public class Test8 {
public static void main(String[] args) throws AException{
AInterface imp=new Impl();
imp.f();
}
}
interface AInterface {
public void f() throws AException;
}
interface BInterface {
public void f() throws BException;
}
class Impl implements AInterface,BInterface {
@Override
public void f() {
}
}
4、异常说明本身不属于方法类型的一部分,方法类型是有方法的名字和参数的类型组成的。此外一个出现在基类方法的异常说明中的异常不一定会出现在子类中。
三、getCause方法的运用:(利用getCause处理利用异常链包装进其他异常的异常)
public class Test6 {
public static void main(String[] args) {
try {
new A().f();
}catch(RuntimeException e) {
System.out.println(e.getCause());//输出three.AException
}
}
}
class A {
public void f() {
try {
throw new AException();
} catch (AException e) {
throw new RuntimeException(e);
}
}
}
四、异常处理的原则:注意能处理的处理,但是一定不能将重要的异常通过catch语句进行捕捉造成隐藏。
五、优先级队列(PriorityQueue)的介绍:
优先级队列始终保持队列顶部的元素始终是最小的,而其他元素无法保证,因此当调用队列的remove方法的时候,每次都是获得当前队列里面的最小的元素。PriorityQueue是怎么保证在添加或者删除元素的时候,顶部的元素始终是最小的呢?实际上,查看源码可以发现,其内部是一个二叉树的结构:
private void siftUpComparable(int k, E x) {
Comparable<? super E> key = (Comparable<? super E>) x;
while (k > 0) {
int parent = (k - 1) >>> 1;
Object e = queue[parent];
if (key.compareTo((E) e) >= 0)
break;
queue[k] = e;
k = parent;
}
queue[k] = key;
}
while循环进行的工作是,当欲加入的元素小于其父节点时,就将两个节点的位置交换。这个算法保证了如果只执行add操作,那么queue这个二叉树是有序的:该二叉树中的任意一个节点都小于以该节点为根节点的子数中的任意其它节点。这也就保证了queue[0],即队顶元素总是所有元素中最小的。
六、Map的一个简单的实现
通过对于Map的一个简单的实现,可以更好的理解Map的内部实现机制,这里出于简单,Key和Value使用了ArrayList,但是实际上这种操作效率极低。
class MyMap<K,V> {
private List<K> kList=new ArrayList<K>();
private List<V> vList=new ArrayList<V>();
private Set<MyEntry> entries=new HashSet<>();
public V put(K key, V value) {
for(K k:kList) {
//如果找到了key.那么替换key对应的value,返回旧的value
if(k.equals(key)) {
int index=kList.indexOf(k);
V oldValue=vList.get(index);
vList.set(index, value);
return oldValue;
}
}
kList.add(key);
vList.add(value);
return null;
}
public Object get(Object key) {
for(K k:kList) {
//如果找到了key.返回key对应的value
if(k.equals(key)) {
int index=kList.indexOf(k);
V value=vList.get(index);
return value;
}
}
return null;
}
public class MyEntry implements Map.Entry<K, V> {
int index;
public MyEntry(int index) {
this.index=index;
}
@Override
public K getKey() {
return kList.get(index);
}
@Override
public V getValue() {
return vList.get(index);
}
@Override
public V setValue(V value) {
return vList.set(index, value);
}
}
public Set<MyEntry> entrySet() {
for(int i=0;i<kList.size();i++) {
entries.add(new MyEntry(i));
}
return entries;
}
}
public class Demo5 {
public static void main(String[] args) {
MyMap<String,String> map=new MyMap<>();
map.put("1", "哈士奇");
map.put("2", "柯基");
map.put("3", "金毛");
map.put("4", "博美");
for(Map.Entry<String, String> en:map.entrySet()) {
System.out.println(en.getKey()+"==>"+en.getValue());
}
}
}
七、LinkedHashMap
可以在LinkedHashMap的构造方法中传入参数,使之采用基于访问的最近最少使用的算法。于是没有被访问过的元素就会出现在前面。下面是例子:
public class Demo6 {
public static void main(String[] args) {
LinkedHashMap<String, String> lh=new LinkedHashMap<>(5, 0.75f, true);
lh.put("1", "小狗");
lh.put("2", "小兔子");
lh.put("3", "小猫");
lh.put("4", "小老鼠");
lh.put("5", "小鸡");
System.out.println(lh.keySet());
String test1=lh.get("2");
String test2=lh.get("3");
System.out.println(lh.keySet());
}
}
输出为(其中最近访问的排在后面):
[1, 2, 3, 4, 5]
[1, 4, 5, 2, 3]
在这类HashMap的内部实现中,查询一个值的情况是,先计算散列码,然后根据散列码查询内部的数组,(数组可以看做桶,而数组中的一个个元素称之为桶位。)数组不保存值,而是保存值的List,然后对List中的值使用equals方法进行线性的查询。
LinkedHashMap的构造器参数中的float参数是负载因子。代表当负载情况(尺寸/容量)>负载因子的时候,容器将自动增加容量(桶位数),实现的情况是容量大致加倍,并且将现有的对象分配到新的桶位中。(称为再散列)
Java编程思想学习笔记_4(异常机制,容器)的更多相关文章
- JAVA编程思想读书笔记(二)--容器
接上篇JAVA编程思想读书笔记(一) 第八章.对象的容纳 No1: java提供了四种类型的集合类:Vector(矢量).BitSet(位集).Stack(堆栈).Hashtable(散列表) No2 ...
- [Java编程思想-学习笔记]第3章 操作符
3.1 更简单的打印语句 学习编程语言的通许遇到的第一个程序无非打印"Hello, world"了,然而在Java中要写成 System.out.println("He ...
- Java编程思想 学习笔记12
十二.通过异常处理错误 Java的基本理念是“结构不佳的代码不能运行”. Java中的异常处理的目的在于通过使用少于目前数量的代码来简化大型.可靠的程序的生成,并且通过这种方式可以使你更加自信:你的 ...
- Java编程思想 学习笔记10
十.内部类 可以将一个类的定义放在另一个类的定义内部,这就是内部类. 内部类是一种非常有用的特性,因为它允许你把一些逻辑相关的类组织在一起,并控制位于内部的类的可视性.然而必须要了解,内部类和组合是 ...
- [Java编程思想-学习笔记]第1章 对象导论
1.1 抽象过程 Java是一门面向对象的语言,它的一个优点在于只针对待解问题抽象,而不用为具体的计算机结构而烦心,这使得Java有完美的移植性,也即Java的口号"Write Once, ...
- Java编程思想 学习笔记11
十一.持有对象 通常,程序总是根据运行时才知道的某些条件去创建新对象.在此之前,不会知道所需对象的数量,甚至不知道确切的类型. Java实用库还提供了一套相当完整的容器类来解决这个问题,其中基本的类 ...
- Java编程思想 学习笔记1
一.对象导论 1.抽象过程 Alan Kay曾经总结了第一个成功的面向对象语言.同时也是Java所基于的语言之一的Smalltalk的五个基本特性,这些特性表现了纯粹的面向对象程序设计方式 1)万物皆 ...
- Java编程思想学习笔记——类型信息
前言 运行时类型信息(RTTI:Runtime Type Information)使得我们可以在程序运行时发现和使用类型信息. Java在运行时识别对象和类的信息的方式: (1)一种是RTTI,它假定 ...
- Java编程思想 学习笔记7
七.复用类 1.组合语法 在新的类中产生现有类的对象.由于新的类是由现有类的对象所组成,所以这种方法叫做组合. 类中域为基本类型时能够自动被初始化为零.对象引用被初始化为null. 编译器不是简单地为 ...
随机推荐
- 通过HtmlEmail 发送邮件
今天第一次来上海市虹口图书馆上自习,感觉还是很爽的.自己撸代码学会了发送邮件.啥都不说了,直接撸代码吧! 首先 必须引进来三个jar包: compile 'javax.mail:mail:1.4.7' ...
- JSTL,自定义一个标签的功能案例
1.自定义一个带有两个属性的标签<max>,用于计算并输出两个数的最大值: 2.自定义一个带有一个属性的标签<lxn:readFile src=“”>,用于输出指定文件的内容 ...
- MVC模式(Model View Controller)下实现数据库的连接,对数据的删,查操作
MVC模式(Model View Controller): Model:DAO模型 View:JSP 在页面上填写java代码实现显示 Controller:Servlet 重定向和请求的转发: 若 ...
- c#中委托和事件(续)(转)
本文将讨论委托和事件一些更为细节的问题,包括一些大家常问到的问题,以及事件访问器.异常处理.超时处理和异步方法调用等内容. 为什么要使用事件而不是委托变量? 在 C#中的委托和事件 中,我提出了两个为 ...
- 关于sql状态值
C = CHECK 约束D = 默认值或 DEFAULT 约束F = FOREIGN KEY 约束L = 日志FN = 标量函数IF = 内嵌表函数P = 存储过程PK = PRIMARY KEY 约 ...
- 【转】【调试技巧】Linux环境下段错误的产生原因及调试方法小结
本文转自:http://www.cnblogs.com/panfeng412/archive/2011/11/06/segmentation-fault-in-linux.html 1. 段错误是什么 ...
- jar 文件
使用类库中的类: 1.当前应用程序所在目录中没有包名的类,直接用. 2.java库. import语句导入. 3.当前目录的子孙目录有报名的类,set classpath,之前有写. 4.扩展ext文 ...
- flash压力测试
涉及目录: vendor/mediatek/proprietary/bootable/bootloader/preloader/platform/mt6735/src/drivers/inc/dram ...
- InputStream和Reader区别
InputStream,OutputStream 前者为字节输入流,后者为字节输出流.Reader Writer 前者为字符输入流,后者为字符输出流. 四个均为抽象类.fileInputStr ...
- 进程间通信之打开关闭一个exe文件
一.打开 1.WinExec("F:\\QQ\\Bin\\QQ.exe",SW_SHOW); 注意:'\'需要改成'\\',阻塞 2. STARTUPINFO si; PROCES ...