####java类里的public,protected,private
1. public:public表明该数据成员、成员函数是对所有用户开放的,所有用户都可以直接进行调用 
2、private:private表示私有,私有的意思就是除了class自己之外,任何人都不可以直接使用,私有财产神圣不可侵犯嘛,即便是子女,朋友,都不可以使用。 

3、protected:protected对于子女、朋友来说,就是public的,可以自由使用,没有任何限制,而对于其他的外部class,
      protected就变成private。 

     与c++的public,private,protected不同,java是以是否在同一个package为基准 
         作用域          当前类    同一package   子孙类     其他package 

          public           √               √                       √             √ 

          protected     √             √                       √           × 

          friendly        √                √                       ×           × 

          private          √                ×                       ×           ×
          
  ####java反射

反射是什么

    Reflection(反射)就是Java程序在运行时可以动态得到程序内部所有类的内部信息,并能动态调用任意对象的内部属性和方法。

 

为什么需要反射

 

    我们为什么要用反射,这主要是反射的动态性决定的,由于反射可以实现动态创建对象,这就很大程度发挥了java的灵活性,降低了程序调用的耦合性,使系统可以更加的灵活,可以更好的应对变化。

 

反射应用

运行反射我们可以做到:

 

    在运行时判断任意一个对象所属的类

    在运行时构造任意一个类的对象

    在运行时判断任意一个类所具有的成员变量和方法

    在运行时调用任意一个对象的成员变量和方法

    生成动态代理

 

跟反射相关的常用类主要有:

 

    java.lang.Class:代表一个类

    java.lang.reflect.Method:代表类的方法

    java.lang.reflect.Field:代表类的成员变量

    java.lang.reflect.Constructor:代表类的构造方法

 


Class类

 

    Class类是反射机制的源头,实际上所谓的反射从程序的运行结果来看就是,通过对象反射获取类的名称。

 

反射其实是类正常方式使用的反过程就像物理学中的反射一样:

 

Java类里的private变量 private在java中的含义_System

 

 

    通过反射我们可以得到的信息:某个类的属性、方法、构造器、类实现了哪些接口、父类是什么。对于每个类而言,JRE都为其保留了一个不变的Class类型的对象,一个Class对象包含了特定某个类的有关信息。

 

    Class本身也是一个类,每个类在JVM中只有有一个Class实例,每个Class对象对应的是一个加载到JVM中的一个.class文件,每个类的实例都会记得自己是由哪个Class实例所生成的。

 


通过反射得到Class类对象,四种方式

 

    我们现在有一个建立好的类Person,类路径是package com.tgb.reflect.common;

 

    1.已知具体Person类,通过Person类的class属性获取,该方法最为安全可靠,程序性能最高。

1. @Test
2. public void
3. Classclass1 = Person.class;  
4. System.out.println(class1);  
5. }

 

 

    2.已知Person类的实例,调用该实例的getClass()方法获取Class对象

1. @Test
2. public void
3. Personp = new
4. Classclass1 =p.getClass();  
5. System.out.println(class1);  
6. }

 

    3.已知Person类的全类名,且该类在类路径下,可通过Class类的静态方法forName()获取。

1. @Test
2. public void test3()throws
3. StringclassName = "com.tgb.reflect.common.Person";  
4. Classclass1 = Class.forName(className);  
5. System.out.println(class1);  
6. }

    4.通过类加载器来获取Class对象。

1. @Test
2. public void test4()throws
3. StringclassName= "com.tgb.reflect.common.Person";  
4. ClassLoaderclassLoader =this.getClass().getClassLoader();  
5. Classclass1=classLoader.loadClass(className);  
6. System.out.println(class1);  
7. }

    以上四种方式得到Class对象的代码经运行得到的结果是一样的,运行结果是"classcom.tgb.reflect.common.Person"; 

 

 

类加载器

 

    当我们程序主动使用某个类时,如果该类还未被加载到内存中,则系统会通过如下三个步骤来对该类进行初始化。

 

Java类里的private变量 private在java中的含义_java_02

 

 

    ClassLoader(类加载器)是用来把类(.class)装在进入内存的,JVM在运行时会产生3个类加载器组成的初始化加载器层次结构 ,如下图所示:

 

Java类里的private变量 private在java中的含义_类加载器_03

 

 

    我们通过代码来看下:

1. @Test
2. public void test1()throws
3.    
4. //1.获取一个系统类加载器
5. ClassLoaderloader1  =ClassLoader.getSystemClassLoader();  
6. System.out.println("系统类加载器为:"+loader1);  
7.    
8. //2.获取系统类加载器的父类加载器,即扩展类加载器
9. ClassLoaderloader2 = loader1.getParent();  
10. System.out.println("扩展类加载器:"+loader2);  
11.    
12. //3.获取扩展类加载器的父类加载器,即引导类加载器
13. ClassLoaderloader3 = loader2.getParent();  
14. System.out.println("引导类加载器:"+loader3);  
15.    
16. //4.测试当前类由哪个类加载器进行加载
17. Classclass1 = Person.class;  
18. ClassLoaderloader4= class1.getClassLoader();  
19. System.out.println("当前类的类加载器:"+loader4);  
20.    
21. //5.测试JDK提供的String类由哪个类加载器加载
22. StringclassName= "java.lang.String";  
23. Classclass2 = Class.forName(className);  
24. ClassLoaderloader5= class2.getClassLoader();  
25. System.out.println("JDK提供的String类加载器:"+loader5);  
26.    
27. }

运行结果为:


    系统类加载器为:sun.misc.Launcher$AppClassLoader@60e53b93

    扩展类加载器:sun.misc.Launcher$ExtClassLoader@5acf9800

    引导类加载器:null

    当前类的类加载器:sun.misc.Launcher$AppClassLoader@60e53b93

    JDK提供的String类加载器:null

 

    由代码运行结果我们可以看到,我们自己定义的类是由系统类加载器加载的、系统类加载器上一级是扩展类加载器、扩展类加载器上一级系统不允许我们拿到,但是扩张类加载器上一级确实是引导类加载器,这类加载器主要是用来加载jvm自带的类库的。

 

    关于类加载器有时候我们可以用来读取系统中的配置文件,下面我们就个简单的小例子,我们将一个Test.properties文件放到我们的类目录中,我们用类加载器来读取其中的内容。

 

我们配置文件的内容很简单:


    user=root

    password=123456

 

    代码如下:

1. @Test
2. public void test2()throws
3. //获得类加载器
4. ClassLoaderclassLoader = this.getClass().getClassLoader();  
5.    
6. //运用类加载器读取property文件到输入流中
7. StringfilePath = "com\\tgb\\reflect\\test1\\Test.properties";  
8. InputStreaminputStream = ClassLoader.getSystemResourceAsStream(filePath);  
9.    
10. //获取文件内容
11. Propertiesproperties =new
12. properties.load(inputStream);  
13. Stringuser=properties.getProperty("user");  
14. System.out.println("user:"+user);  
15. Stringpassword = properties.getProperty("password");  
16. System.out.println("password:"+password);  
17.    
18. }  
19.

运行结果是:


    user:root

    password:123456

 

后记

    这篇文章介绍了反射的基本知识,还有反射的源头Class类,并讲述了得到Class类对象的四种方式,同时还介绍了类加载器的一些知识,关于反射这篇文章就先介绍这么多,下篇文章我们继续。


#############################################################################

#java super关键字

public class java_test {
    java_test(){
    System.out.println("ccdd");
}
public static void main(String args[]){
    java_test1 a = new java_test1();
    a.a(1);
}
   void printfx(){
   System.out.println("aabb");
   }
}
class java_test1 extends java_test{
    java_test1(){
    super();  //用super调用基类的构造函数,必须是在子类的构造函数里边调用
}
  void a(int a){
    System.out.println("java_test1");
    super.printfx(); //调用父类的成员函数,就不必像调用基类的构造函数一样在构造函数里边调用
  }
}