受保护的(protected)——声明该成员的类的子类可以访问这个类的成员(但有一定的限制),并且,声明该成员的包内部的任何类也可以访问这个成员

protected修饰符参考:http://www.360doc.com/content/12/0529/09/10042054_214421414.shtml  

今天在看《Effective java》(第二版)的发现一个困扰我的问题,条目:第17条  要么为继承而设计,并提供文档说明,要么就禁止继承

看到java.util.ArrayList中的removeRange(int fromIndex,int toIndex)方法

removeRange(int fromIndex,int toIndex)方法的定义是从此列表中移除索引在fromIndex(包括)和toIndex(不包括)之间的所有元素。

于是自己就写了一个测试,如下:

import java.util.*;
public class TestRemoveRange1{
public static void main(String args[]){
ArrayList<Integer> list = new ArrayList<>();
for(int i=0;i<10;i++){
list.add(i);
}
//删除0 - 6(不包括6)的元素
list.removeRange(0,6);
System.out.println(list);
}
}

但是运行时报错。

于是查看API,发现removeRange(int fromIndex,int toIndex)方法是protected ,我心想难道是list.removeRange(2,4)调用有错。

解决方法一:写了下一版本,如下:

import java.util.*;
public class TestRemoveRange2<E> extends ArrayList<E>{
public static void main(String args[]){
TestRemoveRange2<Integer> list = new TestRemoveRange2<>();
for(int i=0;i<10;i++){
list.add(i);
}
//删除0 - 6(不包括6)的元素,编译能过
list.removeRange(0,6);
System.out.println(list);
}
}

输出的结果如我所料:[6,7,8,9]

解决方法二:用subList(int fromIndex,int toIndex).clear();代码如下:

import java.util.*;
public class TestRemoveRange3{
public static void main(String args[]){
ArrayList<Integer> list = new ArrayList<>();
for(int i=0;i<10;i++){
list.add(i);
}
//删除0 - 6(不包括6)的元素
list.subList(0,6).clear();
System.out.println(list);
}
}

输出结果也是:[6,7,8,9]

于是我对protected关键字有了很大的疑惑,学了那么久的java,我从理论上知道protected是一个包内部的类,成员变量,方法可访问。但是现在遇到这个问题,我不知道怎么解释它是怎么形成的了。这里就说明protected修饰符还是有一些微妙的地方。在网上找了一篇比较好的博文了解了一下protected这个类的微妙...

com.cwnu.test1包中有:

package com.cwnu.test1;
public class SuperClass {
protected void method(){
System.out.println("This is SuperClass method");
}
}
package com.cwnu.test1;
/**
* 同一个包下,父类protected方法对同一包中的类是可见的
*/
public class SubClass1 extends SuperClass{
public static void main(String[] args) {
SuperClass sc = new SuperClass();
sc.method(); //success
SubClass1 sc1 = new SubClass1();
sc1.method(); //success
SubClass2 sc2 = new SubClass2();
sc2.method(); //success
}
}
class SubClass2 extends SuperClass{
}

如果修改了代码,在两个不同的包中测试protected的访问权限,例子如下:

package com.cwnu.test1;
public class SuperClass {
protected void method(){
System.out.println("This is SuperClass method");
}
}

SuperClass类还是不变

package com.cwnu.test2;

import com.cwnu.test1.SuperClass;

public class SubClass2 extends SuperClass{
public static void main(String[] args) {
SuperClass sc = new SuperClass();
/**
* sc.method();
* 上面的方法不能编译通过,因为SuperClass类的实例在其他包(package com.cwnu.test2)
* 中调用包test1中自己定义的由protected修饰的method()方法,方法是不可见的
**/
SubClass2 sc2 = new SubClass2();
sc2.method();
SubClass3 sc3 = new SubClass3();
sc3.method();
}
}
class SubClass3 extends SuperClass{
/**
* 重写了父类SuperClass的method()方法
* 如果不重写父类的method()方法,用sc3.method()调用编译不会通过,方法也是不可见的
*/
protected void method(){
System.out.println("This is SubClass3 method()");
}
}

我想这也就解释了上面为什么ArrayList的实例list想调用removeRange(int fromIndex,int toIndex)方法不能编译通过的原因——不可见的

也解释了通过继承ArrayList而子类能访问removeRange(int fromIndex,int toIndex)这个方法的原因(详细见TestRemoveRange2)

打开ArrayList源码发现,调用SubList(int fromIndex,int toIndex)最终还是调用了removeRange(int fromIndex,int toIndex)的实现。

removeRange(int fromIndex,int toIndex)内部的删除还是通过移位这一经典的删除方法实现的。

【初学者常见问题】一脚踏入protected埋下的陷阱的更多相关文章

  1. JDK动态代理给Spring事务埋下的坑!

    一.场景分析 最近做项目遇到了一个很奇怪的问题,大致的业务场景是这样的:我们首先设定两个事务,事务parent和事务child,在Controller里边同时调用这两个方法,示例代码如下: 1.场景A ...

  2. (转)面试必备技能:JDK动态代理给Spring事务埋下的坑!

    一.场景分析 最近做项目遇到了一个很奇怪的问题,大致的业务场景是这样的:我们首先设定两个事务,事务parent和事务child,在Controller里边同时调用这两个方法,示例代码如下: 1.场景A ...

  3. 处理器在 protected mode 下的 protection

    前言 实模式,保护模式,分段,分页,虚拟内存,内核态,用户态,如果你对这些术语之间的关系非常熟悉,那就不用继续看了.这篇主要记录我对用户态/内核态的一些理解,如有不对还请指教. 下述说明均为 x86- ...

  4. 关于sails 初学者常见问题汇总

    http://sailsdoc.swift.ren/ 这里有 sails中文文档 一.安装时: 先装nodejs,成功标志 node -v 安装sails 全局安装 node install sail ...

  5. mysql中now()函数的使用,还有oracle的sysdate,可能埋下的坑

    mysql中now()函数的使用,还有oracle的sysdate 在需求中如果系统中药添加当前操作的时间那么很简单的一个操作在写sql的时候直接在这个字段对应的位置写上now()函数就可以了,这样就 ...

  6. 路由页面缓存开启 以及 keep-alive 给你埋下的坑

    为什么要用keep-alive呢, 因为这个会缓存dom模板, 下次再回到这个页面, 就可以利用这个已经渲染好的dom结构了, 如果数据不一样, 也会启用 virtualDoM 进行diff更新, 这 ...

  7. webpack学习笔记(二)-- 初学者常见问题及解决方法

    这篇文章是webpack学习第二篇,主要罗列了本人在实际操作中遇到的一些问题及其解决方法,仅供参考,欢迎提出不同意见. 注:本文假设读者已有webpack方面相关知识,故文中涉及到的专有名词不做另外解 ...

  8. 【持续更新】.Net 开发中给自己埋下的坑!

    1.文件“XXX”正在由另一进程使用,因此该进程无法访问此文件. 原因剖析:文件在主线程操作,在子线程中读写操作文件,刚开始没有意识到程序的问题所在,总是在FileStream中报错,google后常 ...

  9. [开发笔记]--把input框设置成font-size:0埋下的坑。

    vue项目上开发了一个输入pin码的组件.在某些安卓机器上.用户点击键盘后会自动失去焦点自动收起键盘.经过排查,发现是css属性font-size设置成0导致的.

随机推荐

  1. GBK编码和UTF-8编码互转的大坑

    这几天遇到一个BUG,问题很简单,解决却花了3.4天,特意记录下来. linux环境下,将默认编码设置为GBK以后,运行GBK编码的脚本,调用一个Java的jar包,然后总jar包中返回GBK字符串. ...

  2. Linux shell入门基础(三)

    三.输入输出重定向及管道 01.过滤器   Linux过滤器分三种:     1.过滤器(重定向只对过滤器有作用) #gzip a(将a作为输入源,涉及到输入输出)     2.编辑器     3.交 ...

  3. mysql02

    -- 查询课程名称 和年级的名称 -- 非等值连接查询 SELECT subjectname,gradeName FROM `subject`,grade -- 等值连接查询 SELECT subje ...

  4. 刚接触js感觉好吃力啊

    我是一个新手,最近刚刚开始学习js这门语言,感觉好难,有一种无从下手的感觉,不知道应该从哪里学习,虽然也看了很多的书,但是对于一个没有计算机基础的人来说,真的是一种煎熬,每一个名词都要去查.万事开头难 ...

  5. javascript基础之变量和函数声明

    1.变量的声名 window.name = 'gjlin' ; //全局变量  直接name = 'gjlin'  也表示全局变量,但是建议使用window.name = 'gjlin' 这种形式表示 ...

  6. C++线性序列容器<vector>简单总结

    C++线性序列容器<vector>简单总结 vector是一个长度可变的数组,使用的时候无须声明上限,随着元素的增加,Vector的长度会自动增加:Vector类提供额外的方法来增加.删除 ...

  7. (转)用eclipse创建一个j2ee的web工程后,左面projects窗口中的项目如何没有显示webRoot文件夹,除了src的文件夹,其他都不显示

    左边目录窗口的右上方,有个向下的箭头,点里面的filters,把所有的勾都去掉看看

  8. Android Studio中常用设置与快捷键

    常用设置: 1.Tab不用4个空格Code Style->Java->Tabs and Indents->Use tab characterCode Style->Genera ...

  9. Swift2.0异常处理

    // 在抛出异常之前,我们需要在函数或方法的返回箭头 -> 前使用 throws 来标明将会抛出异常 func myMethodRetrunString() throws -> Strin ...

  10. cocos2d-x创建新项目模板

    1.起因 长期使用项目中自带的HelloWorldScene来创建模板工程,不知大家有木有感到厌烦? 我是个懒人,所以就弄了个新的模板工程.这样最起码可以不用每次都把HelloWorldScene删掉 ...