反射与线程间通讯

版权声明:原创作品,转载请联系公众号阿Q说 https://blog.csdn.net/Qingai521/article/details/88398422

反射

一、在运行状态中,对于任意一个类,都能够获取到这个类的所有属性和方法,对于任意一个对象,都能够调用它的任意一个方法和属性(包括私有的方法和属性),这种动态获取的信息以及动态调用对象的方法的功能就称为java语言的反射机制。通俗点讲,通过反射,该类对我们来说是完全透明的,想要获取任何东西都可以。

注:反射的步骤

A:首先学习字节码文件

B:第二学习获取字节码文件的构造方法,并创建对象

C:有了对象了,就要学会获取对象中的成员变量## 多线程与同步代码块详解

D:能获取对象中的成员变量了,那么还要学习成员方法

二、1)字节码文件的三种获取方式

①Object类的getClass()方法:对象.getClass()

Person p = new Person();
Class c = p.getClass();

注意所有同一个类的字节码文件对象(实例)都是同一个(因为一个类只有唯一的字节码文件)

比如:

Person p = new Person();
Class c = p.getClass();

Person p2 = new Person();
Class c2 = p2.getClass();

System.out.println(p == p2);// false
System.out.println(c == c2);// true

②数据类型的静态的class属性:类名.class;例如:Class c3 = Person.class;

③通过Class类的静态方法forName(String className)(一般只用最后一种,前面两种了解即可)

Class c4 = Class.forName(“com.itcast.Person”);//通过包路径获取类字节码文件

2)反射获取类的构造方法

  • public Constructor<?>[] getConstructors():所有公共构造方法
  • public Constructor<?>[] getDeclaredConstructors():所有构造方法,包括私有
  • public Constructor getConstructor(Class<?>… parameterTypes):获取单个构造方法

比如:

  Class clazz = Class.forName("com.itcast.Person");
  Constructor c = clazz.getConstructor(String.class,int.class);//获取有参构造
  Person p = (Person) c.newInstance("张三",23);	//通过有参构造创建对象
  System.out.println(p);

3)反射获取类的成员变量

  • Field[] fields = c.getFields();// 获取所有公共的成员变量
  • Field[] fields = c.getDeclaredFields();// 获取所有的成员变量
  • Field field = c.getField(“age”);// 获取单个的成员变量

比如:

Class clazz = Class.forName("com.itcast.Person");
Constructor c = clazz.getConstructor(String.class,int.class);	//获取有参构造
Person p = (Person) c.newInstance("张三",23);						//通过有参构造创建对象	//修改姓名的值
Field f = clazz.getDeclaredField("name");						//暴力反射获取字段
f.setAccessible(true);											//去除私有权限
f.set(p, "李四");	
System.out.println(p);

4)反射获取类的成员方法

  • Method[] methods = c.getMethods();// 所有公共方法,包括父类的
  • Method[] methods = c.getDeclaredMethods();// 本类的所有方法
Class clazz = Class.forName("com.heima.bean.Person");
Constructor c = clazz.getConstructor(String.class,int.class);	//获取有参构造
Person p = (Person) c.newInstance("张三",23);						//通过有参构造创建对象

Method m = clazz.getMethod("eat");								//获取eat方法
m.invoke(p);

Method m2 = clazz.getMethod("eat", int.class);					//获取有参的eat方法
m2.invoke(p, 10);

三、案例:动态创建某个配置文件中提供的一个类的名字的这个类的对象,并执行其方法

public class DemoClass {
    public void run() {
        System.out.println("welcome to heima!");
    }
}
//创建输入流关联xxx.properties
BufferedReader br = new BufferedReader(new FileReader("file.properties"));	
Class clazz = Class.forName(br.readLine());	//读取配置文件中类名,获取字节码对象

DemoClass dc = (DemoClass) clazz.newInstance();	//通过字节码对象创建对象
dc.run();//执行方法

四、泛型只是在编译期间(或者是你在写代码期间)强制要求你的做的一些规范,等编译通过生成了字节码文件后,泛型就会被擦除,运行期间就没有泛型了。
五、public void setProperty(Object obj, String propertyName, Object value){},此方法可将obj对象中名为propertyName的属性的值设置为value。

public class Tool {
    //此方法可将obj对象中名为propertyName的属性的值设置为value。
    public void setProperty(Object obj, String propertyName, Object value) throws Exception {
        Class clazz = obj.getClass();					//获取字节码对象
        Field f = clazz.getDeclaredField(propertyName);	//暴力反射获取字段
        f.setAccessible(true);							//去除权限
        f.set(obj, value);
    }
}

public static void main(String[] args) throws Exception {
    Student s = new Student("张三", 23);
    System.out.println(s);

    Tool t = new Tool();
    t.setProperty(s, "name", "李四");
    System.out.println(s);
}

线程间通讯

多个线程并发执行时,在默认情况下CPU是随机切换线程的,如果我们希望他们有规律的执行, 就可以使用通信。两个线程就用wait() 和notify();多个线程就用wait() 和notifyAll()。

注意事项:

1,这两个方法必须在同步代码中执行, 并且使用同步锁对象来调用。

2,为什么wait方法和notify方法定义在Object这类中?因为锁对象可以是任意对象,Object是所有的类的基类,所以wait方法和notify方法需要定义在Object这个类中。

3,sleep方法和wait方法的区别?

sleep方法必须传入参数,参数就是时间,时间到了自动醒来;wait方法可以传入参数也可以不传入参数,传入参数就是在参数的时间结束后等待,不传入参数就是直接等待。sleep方法在同步函数或同步代码块中,不释放锁,睡着了也抱着锁睡;wait方法在同步函数或者同步代码块中释放锁。

线程组
Java中使用ThreadGroup来表示线程组,它可以对一批线程进行分类管理,Java允许程序直接对线程组进行控制。默认情况下,所有的线程都属于主线程组。

public final ThreadGroup getThreadGroup();//通过线程对象获取他所属于的组

public final String getName();//通过线程组对象获取他组的名字

我们也可以给线程设置分组

1,ThreadGroup(String name) 创建线程组对象并给其赋值名字

2,创建线程对象

3,Thread(ThreadGroup group, Runnable target, String name)

4,设置整组的优先级或者守护线程

多线程程序实现方式

这种方式的好处,可以通过线程方法来返回一个结果,而run方法不能返回结果

public class MyCallable implements Callable<Integer> {
	private int number;

	public MyCallable(int number) {
		this.number = number;
	}
    @Override
    public Integer call() throws Exception {
    	int sum = 0;
    	for (int x = 1; x <= number; x++) {
        	sum += x;
    	}
    	return sum;
    }	
}

// 创建线程池对象
ExecutorService pool = Executors.newFixedThreadPool(2);

// 可以执行Runnable对象或者Callable对象代表的线程
Future<Integer> f1 = pool.submit(new MyCallable(100));
Future<Integer> f2 = pool.submit(new MyCallable(200));

// V get()
Integer i1 = f1.get();
Integer i2 = f2.get();

System.out.println(i1);
System.out.println(i2);

// 结束
pool.shutdown();

线程的生命周期:新建,就绪,运行,阻塞,死亡

好了今天就先说到这了,明天继续。想了解更多学习知识,请关注微信公众号“阿Q说”,获取更多学习资料吧!你也可以后台留言说出你的疑惑,阿Q将会在后期的文章中为你解答。每天学习一点点,每天进步一点点。

猜你喜欢

转载自blog.csdn.net/Qingai521/article/details/88398422