学习java很长时间了,但是比较惭愧,很少运用java比较神秘的反射机制来开发项目,就是平时的娱乐也没用到过。但是现在我终于可以摆脱这个魔咒了,哈哈。现在也跟大家一起分享一下,java反射的奥妙。

 JAVA反射机制定义

  JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

  Java反射机制主要提供了以下功能: 在运行时判断任意一个对象所属的类;在运行时构造任意一个类的对象;在运行时判断任意一个类所具有的成员变量和方法;在运行时调用任意一个对象的方法;生成动态代理。

  有时候我们说某个语言具有很强的动态性,有时候我们会区分动态和静态的不同技术与作法。我们朗朗上口动态绑定(dynamic binding)、动态链接(dynamic linking)、动态加载(dynamic loading)等。然而“动态”一词其实没有绝对而普遍适用的严格定义,有时候甚至像对象导向当初被导入编程领域一样,一人一把号,各吹各的调。

  一般而言,开发者社群说到动态语言,大致认同的一个定义是:“程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言”。从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C#不是动态语言。

 

  尽管在这样的定义与分类下Java不是动态语言,它却有着一个非常突出的动态相关机制:Reflection。这个字的意思是“反射、映象、倒影”,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods1。这种“看透class”的能力(the ability of the program to examine itself)被称为introspection(内省、内观、反省)。Reflection和introspection是常被并提的两个术语。

 

现在我们不来虚的了,来点实际东西让大家体验一下java反射机制的应用。

首先搞个数据库,用mysql数据库哦:

CREATE DATABASE test
USE `test`;
/*Table structure for table `sync_config` */
DROP TABLE IF EXISTS `sync_config`;
CREATE TABLE `sync_config` (
  `rkey` varchar(255) NOT NULL DEFAULT '',
  `rvalue` varchar(255) NOT NULL DEFAULT '',
  `desc` varchar(255) DEFAULT NULL
) ENGINE=MyISAM DEFAULT CHARSET=latin1;
/*Data for the table `sync_config` */
insert  into `sync_config`(`rkey`,`rvalue`,`desc`) values ('ServerMaxCopyTaskCount','15',NULL),('NormalJobHarddiskRate','90',NULL),('ClientJobHarddiskRate','99',NULL),('ServerMaxDelTaskCount','50',NULL),('ClientMaxDelTaskCount','50',NULL),('ClientMaxCopyTaskCount','50',NULL),('MinSrcCount','3',NULL),('TouchTimeInSecond','5',NULL),('sync_version','0.11',NULL),('http_443_version','0',NULL),('http_80_version','1.10.1124.0',NULL),('NormalJobDeleteRate','1',NULL),('isStarted_CheckTimeoutServerCron','1',NULL),('isStarted_DelFileCron','1',NULL),('isStarted_EditorHotFileCron','1',NULL),('isStarted_NewFileCron','1',NULL),('isStarted_ServerInfoCron','1',NULL),('isStarted_ServerNodeSpeedCron','1',NULL),('isStarted_SyncCountCron','1',NULL),('isStarted_TaskCron','1',NULL),('isStarted_WorkerThread','1',NULL),('isStarted_WorkerServletGiveTaskToClient','1','send task to client');

 

数据库搞好了 咱们开始写代码。这里我先说明使用反射机制的原因。因为代码中的一个类ConfigInDB中的每一个属性是和数据库的字段一一对应的,并且初始化了值。项目启动一开始就会去初始化这个ConfigInDB中的属性。为了动态方便地初始化如此多的属性,我们有理由用反射机制去动态的萃取其数据库你的对应字段的值。代码如下所示:

public class ConfigInDB implements BasicInterface {
	public static volatile int isStarted_CheckTimeoutServerCron = 1; // 1.set '1' to start cron  2.set '2' to stop cron
	public static volatile int isStarted_DelFileCron = 1; // 1.set '1' to start cron  2.set '2' to stop cron
	public static volatile int isStarted_EditorHotFileCron = 1; // 1.set '1' to start cron  2.set '2' to stop cron
	public static volatile int isStarted_NewFileCron = 1; // 1.set '1' to start cron  2.set '2' to stop cron
	public static volatile int isStarted_ServerInfoCron = 1; // 1.set '1' to start cron  2.set '2' to stop cron
	public static volatile int isStarted_ServerNodeSpeedCron = 1; // 1.set '1' to start cron  2.set '2' to stop cron
	public static volatile int isStarted_SyncCountCron = 1; // 1.set '1' to start cron  2.set '2' to stop cron
	public static volatile int isStarted_TaskCron = 1; // 1.set '1' to start cron  2.set '2' to stop cron
	public static volatile int isStarted_ReadVirtualNodeXmlCron = 1; // 1.set '1' to start cron  2.set '2' to stop cron
	public static volatile int isStarted_WorkerThread = 1; // 1.set '1' to start   2.set '2' to stop 
	public static volatile int isStarted_WorkerServletGiveTaskToClient = 1; // 1.set '1' to start   2.set '2' to stop 
	public static volatile String virtualNodeXmlPath = System.getProperty("user.home") + "/virtualNode.xml";
	public static volatile String kaFilePath = System.getProperty("user.home") + "/kaFile.txt";
	public static String ServerVersion;
	public static volatile int //
			ClientJobHarddiskRate = 99, //
			EmergecyJobHarddiskRate = 98, //
			NormalJobHarddiskRate = 95, //
			NormalJobDeleteRate = 80; //
	public static volatile long //
			//not main Cron
			ConfigUpdateCron = Constant.Minute, //
			ServerNodeSpeedCron = 1 * Constant.Minute, //
			ServerInfoCron = 10 * Constant.Second, //
			TaskCron = 10 * Constant.Second, //
			ReadVirtualNodeXmlCron = 10 * Constant.Minute, //
			//main Cron
			SyncCountCron = 10 * Constant.Second, //SyncCountCron
			EditorHotFileCron = 1 * Constant.Minute, //EditorHotFileCron
			NewFileCron = 10 * Constant.Second, //NewFileCron
			DelFileCron = 15 * Constant.Minute, //DelFileCron
			CheckTimeoutServerCron = 1 * Constant.Minute, //CheckTimeoutServerCron
			CronSleepTime = 5 * Constant.Second; //
	public static int MinSrcCount = 2;
	public static int DBPoolSize = 10;
	public static int CdnDBPoolSize = 2;
	public static int EditorDBPoolSize = 1;
	public static int ToodouDBPoolSize = 1;
	public static int DelTimeoutInSecond = 60;
	public static int MinCopyTimeoutInSecond = 60 * 10;
	public static int MaxCopyTimeoutInSecond = 60 * 60 * 2;
	public static int CopyMinSpeed = 10 * 1024;
	public static int ClientMaxCopyTaskCount = 100;
	public static int ServerMaxCopyTaskCount = 50;
	public static int ClientMaxDelTaskCount = 100;
	public static int ServerMaxDelTaskCount = 50;
	public static int TouchTimeInSecond = 10;
	public static int ClientCopyThreadCount = 3;
	public static int TaskQueueBusyLength = 10000;
	public static int CdnFileCacheSize = 100;
	public static int CheckTimeoutServer = 1;
	public static String http_443_version = "asd";
	public static String http_80_version = "asd";
	public static String sync_version = "asd";
	public static long TimeJobCheckCron = 6 * Constant.Hour;
}

接下来是动态的赋值代码:

 

public class ConfigUpdateCron extends ICronJob implements BasicInterface {
	public boolean Arun(Connection conn) throws SQLException {
		List<StringPair> list = null;
		Statement stmt = conn.createStatement();
		ResultSet srs = stmt.executeQuery(SqlString.SelectConfigTable);
		while (srs.next()) {
			if (list == null)
				list = new ArrayList<StringPair>();
			list.add(new StringPair(srs.getString(1), srs.getString(2)));
		}
		srs.close();
		stmt.close();
		if (list != null) {
			return SetValue(list);
		}
		return true;
	}
	private boolean SetValue(List<StringPair> list) {
		Class<ConfigInDB> c = ConfigInDB.class;
		Field f;
		String s;
		int i;
		long l;
		byte b;
		float d;
		for (StringPair pair : list) {
			try {
				f = c.getField(pair.key);
				s = f.getType().getName();
				if (s.compareTo("int") == 0) {
					try {
						i = Integer.parseInt(pair.value);
					} catch (Exception ex) {
						log.error("bad value of " + s, ex);
						continue;
					}
					if (f.getInt(c) != i) {
						log.info("Config." + pair.key + " /"" + f.getInt(c) + "/" to /"" + i + "/"");
						f.setInt(c, i);
					}
				} else if (s.compareTo("byte") == 0) {
					try {
						b = Byte.parseByte(pair.value);
					} catch (Exception ex) {
						log.error("bad value of " + s, ex);
						continue;
					}
					if (f.getByte(c) != b) {
						log.info("Config." + pair.key + " /"" + f.getByte(c) + "/" to /"" + b + "/"");
						f.setInt(c, b);
					}
				} else if (s.compareTo("float") == 0) {
					try {
						d = Float.parseFloat(pair.value);
					} catch (Exception ex) {
						log.error("bad value of " + s, ex);
						continue;
					}
					if (f.getFloat(c) != d) {
						log.info("Config." + pair.key + " /"" + f.getFloat(c) + "/" to /"" + d + "/"");
						f.setFloat(c, d);
					}
				} else if (s.compareTo("[I") == 0)//int[]
				{
					String[] ss = pair.value.split(",");
					int[] old = (int[]) f.get(c);
					if (old.length != ss.length) {
						//error
						continue;
					}
					for (int n = 0; n < ss.length; n++) {
						i = Integer.parseInt(ss[n]);
						if (i != old[n]) {
							old[n] = i;
							//
						}
					}
				} else if (s.compareTo("long") == 0) {
					try {
						l = Long.parseLong(pair.value);
					} catch (Exception ex) {
						log.error("bad value of " + s, ex);
						continue;
					}
					if (f.getLong(c) != l) {
						log.info("Config." + pair.key + " /"" + f.getLong(c) + "/" to /"" + l + "/"");
						f.setLong(c, l);
					}
				} else if (s.compareTo("java.lang.String") == 0) {
					if (((String) f.get(c)).compareTo(pair.value) != 0) {
						log.info("Config." + pair.key + " /"" + f.get(c) + "/" to /"" + pair.value + "/"");
						f.set(c, pair.value);
					}
				} else {
					Nagios.Alert("unsupport type:" + s + " on Config." + pair.key);
				}
			} catch (Exception e) {
				log.error("ex", e);
				return false;
			}
		}
		return true;
	}
	private ConfigUpdateCron() {
	}
	private static ConfigUpdateCron instance = new ConfigUpdateCron();
	public static ConfigUpdateCron GetInstance() {
		return instance;
	}
	public boolean Init() {
		this.initLogger();
		return true;
	}
}

 

大家我的数据库连接就不写了,大家可以补充一下我的数据库连接,就可以运行代码了。