需求:
某高校博士录取分数线录取算法是这样的:
1、 硕博连读考生为外语45分以上(含45分,下同), 综合成绩(初试总分*0.7+复试分数*0.3)不低于60分;
2、 普通考生(经济管理学院除外)为外语45分以上,专业课60分以上,综合成绩(初试总分/3*0.7+复试分数*0.3)不低于60分;
3、 经济管理学院(001)考生外语55分以上,专业课60分以上,综合成绩(初试总分/3*0.7+复试分数*0.3)不低于60分。
这段算法简单、清楚,用java实现是a piece of cake,但是考虑到我们的招生系统是为全国很多高校服务的,每个学校的录取算法可能有不同,不能在代码里面写死,最好是能让用户自己配置这段算法。这么简单的算法(高校研究生招生明规则还是比较简单的,潜规则就不清楚了)如果用规则引擎,显然是杀鸡用牛刀了。在目前的需求而言,用脚本语言来处理可能是更好的选择。
java里面对脚本语言的支持很好,以前的话beanshell流行一点(印象中shark、jbpm里面都支持beanshell,不过很久没跟踪了,不知道现在这两大workflow engine的情况怎样了),现在groovy是越来越流行了,最新的groovy 1.6更号称性能有了巨大的改进。( http://www.infoq.com/articles/groovy-1-6 ),所以这里先尝试groovy。
g了一下,发现在应用里嵌入groovy有三种方式(参考 http://groovy.codehaus.org/Embedding+Groovy )
第一种是比较传统的,通过GroovyShell的方式,跟用beanshell的方式差不多。
第二种方式是通过GroovyClassLoader来动态加载groovy类,然后直接使用groovy类。这点得以实现的原因是每一个groovy类编译后都是合法的java类。这种方式没有太多意思,还不如直接集成java编译器,编译java类后动态加载java类?呵呵。
第三种方式是使用GroovyScriptEngine。如果要在应用中提供最完整的脚本支持,GroovyScriptEngine是不二之选。GSE会做依赖性检测,即某脚本依赖的脚本修改过了,整个脚本树都会重新编译、重新加载。
我们的应用暂时只需要简单的执行一段脚本就可以了,所以选用了第一种方式。将embeddable目录下的groovy-all-1.6.0.jar扔进类路径,在界面配置好脚本(见附图
):
后台判断博士考生是否上线的方法:
简单说明:我痛恨变量命名采用拼音缩写,但是高校研*部很多的数据结构都是用拼音缩写来命名,所以这里的脚本只能跟他们接轨。脚本最终是要给老师改的。
完工!
执行效率还是不错的,不知道groovy evaluate同样的脚本后是否会用类似执行sql的方式缓存。在我们这里数据也比较少,只有几百条,毕竟博士考生还是比较少的,一般就几百人。不过呢,可能经过几年硕士就业率也不行了,又鼓励大家继续考博,那博士考生会不会上到几千呢?到时看要不要再优化程序吧,呵呵。
某高校博士录取分数线录取算法是这样的:
1、 硕博连读考生为外语45分以上(含45分,下同), 综合成绩(初试总分*0.7+复试分数*0.3)不低于60分;
2、 普通考生(经济管理学院除外)为外语45分以上,专业课60分以上,综合成绩(初试总分/3*0.7+复试分数*0.3)不低于60分;
3、 经济管理学院(001)考生外语55分以上,专业课60分以上,综合成绩(初试总分/3*0.7+复试分数*0.3)不低于60分。
这段算法简单、清楚,用java实现是a piece of cake,但是考虑到我们的招生系统是为全国很多高校服务的,每个学校的录取算法可能有不同,不能在代码里面写死,最好是能让用户自己配置这段算法。这么简单的算法(高校研究生招生明规则还是比较简单的,潜规则就不清楚了)如果用规则引擎,显然是杀鸡用牛刀了。在目前的需求而言,用脚本语言来处理可能是更好的选择。
java里面对脚本语言的支持很好,以前的话beanshell流行一点(印象中shark、jbpm里面都支持beanshell,不过很久没跟踪了,不知道现在这两大workflow engine的情况怎样了),现在groovy是越来越流行了,最新的groovy 1.6更号称性能有了巨大的改进。( http://www.infoq.com/articles/groovy-1-6 ),所以这里先尝试groovy。
g了一下,发现在应用里嵌入groovy有三种方式(参考 http://groovy.codehaus.org/Embedding+Groovy )
第一种是比较传统的,通过GroovyShell的方式,跟用beanshell的方式差不多。
- Binding binding = new Binding();
- binding.setVariable("foo", new Integer(2));
- GroovyShell shell = new GroovyShell(binding);
- Object value = shell.evaluate("println 'Hello World!'; x = 123; return foo * 10");
- assert value.equals(new Integer(20));
- assert binding.getVariable("x").equals(new Integer(123));
第二种方式是通过GroovyClassLoader来动态加载groovy类,然后直接使用groovy类。这点得以实现的原因是每一个groovy类编译后都是合法的java类。这种方式没有太多意思,还不如直接集成java编译器,编译java类后动态加载java类?呵呵。
- ClassLoader parent = getClass().getClassLoader();
- GroovyClassLoader loader = new GroovyClassLoader(parent);
- Class groovyClass = loader.parseClass(new File("src/test/groovy/script/HelloWorld.groovy"));
- // let's call some method on an instance
- GroovyObject groovyObject = (GroovyObject) groovyClass.newInstance();
- Object[] args = {};
- groovyObject.invokeMethod("run", args);
第三种方式是使用GroovyScriptEngine。如果要在应用中提供最完整的脚本支持,GroovyScriptEngine是不二之选。GSE会做依赖性检测,即某脚本依赖的脚本修改过了,整个脚本树都会重新编译、重新加载。
- import groovy.lang.Binding;
- import groovy.util.GroovyScriptEngine;
- String[] roots = new String[] { "/my/groovy/script/path" };
- GroovyScriptEngine gse = new GroovyScriptEngine(roots);
- Binding binding = new Binding();
- binding.setVariable("input", "world");
- gse.run("hello.groovy", binding);
- System.out.println(binding.getVariable("output"));
我们的应用暂时只需要简单的执行一段脚本就可以了,所以选用了第一种方式。将embeddable目录下的groovy-all-1.6.0.jar扔进类路径,在界面配置好脚本(见附图

):
- result=false;
- if(kslym=='12')
- {
- zhcj=cszf*0.7+fscj*0.3;
- if(wgy>=45&&zhcj>=60)result=true;
- }
- else{
- if(yxsm!='001'){
- zhcj=cszf/3*0.7+fscj*0.3;
- if(wgy>=45&&ywk1>=60&&ywk2>=60&&zhcj>=60)result=true;
- }
- else{
- zhcj=cszf/3*0.7+fscj*0.3;
- if(wgy>=55&&ywk1>=60&&ywk2>=60&&zhcj>=60)result=true;
- }
- }
后台判断博士考生是否上线的方法:
- public Boolean canPassed(DoctorRecruitScoreObj score)
- {
- Binding binding=new Binding();
- binding.setVariable("kslym",score.getDoctorResignup().getKslym());
- binding.setVariable("wgy",score.getEnglish());
- binding.setVariable("zzll",score.getZzll());
- binding.setVariable("ywk1",score.getCourseA());
- binding.setVariable("ywk2",score.getCourseB());
- binding.setVariable("cszf",score.getEnglish()+score.getZzll()+score.getCourseA()+score.getCourseB());
- binding.setVariable("fscj",score.getCourseC());
- binding.setVariable("yxsm",score.getDoctorResignup().getBkyxsm());
- GroovyShell shell=new GroovyShell(binding);
- DoctorCuttingscore cuttingScore=getCuttingScore();
- if(cuttingScore!=null)
- {
- shell.evaluate(cuttingScore.getRule());
- return (Boolean)binding.getVariable("result");
- }
- return false;
- }
简单说明:我痛恨变量命名采用拼音缩写,但是高校研*部很多的数据结构都是用拼音缩写来命名,所以这里的脚本只能跟他们接轨。脚本最终是要给老师改的。
完工!
执行效率还是不错的,不知道groovy evaluate同样的脚本后是否会用类似执行sql的方式缓存。在我们这里数据也比较少,只有几百条,毕竟博士考生还是比较少的,一般就几百人。不过呢,可能经过几年硕士就业率也不行了,又鼓励大家继续考博,那博士考生会不会上到几千呢?到时看要不要再优化程序吧,呵呵。