`
roki
  • 浏览: 61884 次
  • 性别: Icon_minigender_1
  • 来自: 上海
文章分类
社区版块
存档分类
最新评论

《搜索引擎零距离》第三章 IRS虚拟机及编译器实现原理(3)

阅读更多
3.1.7 方法调用
例子:

foo.bar()
foo.bar
bar()
print "hello world\n"
print
语法:
[表达式 .] 标识符 [( 表达式 ... [* [表达式]] )]
若未指定被调用方法的对象,则调用 self 的方法。
方法名中除了通常的标识符以外,还可以添加“?”或“!”等后缀。通常在布尔型(返回真或伪)方法名后添加“?”,在比同名(无“!”)方法更具破坏性的方法名(例:tr 和 tr!)后添加“!”。
若最后一个参数前带“*”的话,将会先展开该参数的值,然后才传递。例如:
foo(1,*[2,3,4])   # 等同于 foo(1,2,3,4)

3.2 Java与JRuby的整合[size=x-large][/size]
当前版本的JRuby与Java的结合并不是非常完美,需要编写一些接口代码。笔者所使用的JRuby包是: jruby-complete-1.0.1.jar。
3.2.1 Java中的Ruby运行库环境
首先,Java程序中需要进行如下定义:
RubyInstanceConfig config = new RubyInstanceConfig();
Ruby runtime = Ruby.newInstance(config);

其中 RubyInstanceConfig config里可以配置JRuby运行环境的各种参数,比如
currentDirectory(当前目录),environment(环境变量)等。config作为Ruby.newInstance的参数来构造runtime ,而runtime 是JRuby的运行库环境,各种JRuby语句的执行都需要在runtime的上下文中执行 。
其次,需要在ParalleIRVirtualMachine进行如下定义:
IRubyObject rt = JavaUtil.convertJavaToRuby(runtime, this);
这句代码把ParalleIRVirtualMachine对象转换成IRubyObject对象,以便在这个对象上进行其他操作。
最后,我们需要把对象rt作为一个全局变量添加到runtime上去:
IAccessor d = new ValueAccessor(rt);
runtime.getGlobalVariables().define("$vm", d);
这步操作完成之后,runtime里就有了一个可供使用的全局对象$vm, 而$vm实际引用的是
IRubyObject对象rt, 由于rt是由ParalleIRVirtualMachine对象转化而来的,所以$vm实际引用的是ParalleIRVirtualMachine对象。
为了在$vm对象上添加方法,需要在rt.getMetaClass()返回的对象上调用defineMethod方法。defineMethod的第一个参数是函数名,第二个参数是一个org.jruby.runtime.callback.Callback对象。

例如,要在$vm上添加一个名为match的方法,可以这样写:
rt.getMetaClass().defineMethod("match", processMatch);
其中,第一个参数是函数名,第二个参数是一个org.jruby.runtime.callback.Callback对象。processMatch的定义是:
Callback processMatch = new IRSReflectionCallback(
ParalleIRVirtualMachine.class, "processMatchRuby",
new Class[] { RubyString.class }, false,// if restArg
false, // if staticArg
Arity.singleArgument(), true);// if fast
上述new IRSReflectionCallback构造函数的第一个参数ParalleIRVirtualMachine.class
是目标Java对象的类型,processMatchRuby是对象上的函数名,new Class[] { RubyString.class }是参数类型数组,参数Arity.singleArgument()指明有几个参数,single是指一个参数,其他参数一般情况下不需修改。
其中 IRSReflectionCallback类是一个自定义的类,它提供了Callback接口的实现。
Callback接口定义如下:
Callback.java:

public interface Callback {
    public IRubyObject execute(IRubyObject recv, IRubyObject[] args, Block block);
    public Arity getArity();
}
3.2.2 IRSReflectionCallback类实现


IRSReflectionCallback.java:

package com.irs..ruby;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.List;

import org.jruby.exceptions.JumpException;
import org.jruby.exceptions.MainExitException;
import org.jruby.exceptions.RaiseException;
import org.jruby.exceptions.ThreadKill;
import org.jruby.javasupport.JavaObject;
import org.jruby.runtime.Arity;
import org.jruby.runtime.Block;
import org.jruby.runtime.builtin.IRubyObject;
import org.jruby.runtime.callback.Callback;

/**
 * A wrapper for <code>java.lang.reflect.Method</code> objects which implement Ruby methods.
 */
public class IRSReflectionCallback implements Callback {
    private Method method;
    private Class type;
    private String methodName;
    private Class[] argumentTypes;
    private boolean isRestArgs;
    private Arity arity;
    private boolean isStaticMethod;
    private boolean fast;
    
/*构造函数*/
    public IRSReflectionCallback(Class type, String methodName, Class[] argumentTypes,
            boolean isRestArgs, boolean isStaticMethod, Arity arity, boolean fast) {
        this.type = type;
    	this.methodName = methodName;
    	this.argumentTypes = argumentTypes;
        this.isRestArgs = isRestArgs;
        this.isStaticMethod = isStaticMethod;
    	this.arity = arity;
        this.fast = fast;
    	
        assert type != null;
        assert methodName != null;
        assert arity != null;
        
        loadMethod(fast);
    }
    
/*
按照是否为静态方法来传递函数参数
静态方法不需要传递this对象,成员方法需要传递this对象
*/
    private void loadMethod(boolean fast) {
    	Class[] args;
    	
        if (isStaticMethod) {
            Class[] types = new Class[argumentTypes.length + 1];
            System.arraycopy(argumentTypes, 0, types, 1, argumentTypes.length);
            types[0] = IRubyObject.class;
            args = types;
        } else {
            args = argumentTypes;
        }
        
        // ENEBO: Perhaps slow but simple for now
        if (!fast) {
            Class[] types = new Class[args.length + 1];
            System.arraycopy(args, 0, types, 0, args.length);
            types[args.length] = Block.class;
            args = types;
        }
        
        try {
        	List<Class> tmpArg=Arrays.asList(args);
			//通过反射来从对象上获得方法,供Ruby引擎调用
            method = type.getMethod(methodName, args);
        } catch (NoSuchMethodException e) {
            throw new RuntimeException("NoSuchMethodException: Cannot get method \"" + methodName
                    + "\" in class \"" + type.getName() + "\" by Reflection.");
        } catch (SecurityException e) {
            throw new RuntimeException("SecurityException: Cannot get method \"" + methodName
                    + "\" in class \"" + type.getName() + "\" by Reflection.");
        }
    }
    
    /**
     * Returns an object array that collects all rest arguments in its own object array which
     * is then put into the last slot of the first object array.  That is, assuming that this
     * callback expects one required argument and any number of rest arguments, an input of
     * <code>[1, 2, 3]</code> is transformed into <code>[1, [2, 3]]</code>.  
     */
    protected final Object[] packageRestArgumentsForReflection(final Object[] originalArgs) {
        IRubyObject[] restArray = new IRubyObject[originalArgs.length - (argumentTypes.length - 1)];
        Object[] result = new Object[argumentTypes.length];
        try {
            System.arraycopy(originalArgs, argumentTypes.length - 1, restArray, 0, originalArgs.length - (argumentTypes.length - 1));
        } catch (ArrayIndexOutOfBoundsException e) {
            assert false : e;
        	return null;
        }
        System.arraycopy(originalArgs, 0, result, 0, argumentTypes.length - 1);
        result[argumentTypes.length - 1] = restArray;
        return result;
    }

    /**
	在IRubyObject recv对象上调用目标方法
     * Invokes the Ruby method. Actually, this methods delegates to an internal version
     * that may throw the usual Java reflection exceptions.  Ruby exceptions are rethrown, 
     * other exceptions throw an AssertError and abort the execution of the Ruby program.
     * They should never happen.
     */
	/**
     * Calls a wrapped Ruby method for the specified receiver with the specified arguments.
     */
    public IRubyObject execute(IRubyObject recv, IRubyObject[] oargs, Block block) {
        arity.checkArity(recv.getRuntime(), oargs);

        Object[] methodArgs = oargs;
        
    	if (isRestArgs) {
    		methodArgs = packageRestArgumentsForReflection(methodArgs);
    	}
        try {
        	JavaObject receiver = (JavaObject)recv;
        	Object target=receiver.getValue();
        	
            if (isStaticMethod) {
                Object[] args = new Object[methodArgs.length + (fast ? 1 : 2)];
                System.arraycopy(methodArgs, 0, args, 1, methodArgs.length);
                args[0] = recv;
                if (!fast) args[methodArgs.length + 1] = block;
                receiver = null;
                methodArgs = args;
            } else {
                Object[] args = new Object[methodArgs.length + (fast ? 0 : 1)];
                System.arraycopy(methodArgs, 0, args, 0, methodArgs.length);
                if (!fast) args[methodArgs.length] = block;
                methodArgs = args;
            }
            return (IRubyObject) method.invoke(target, methodArgs);
        } catch (InvocationTargetException e) {
            if (e.getTargetException() instanceof RaiseException) {
                throw (RaiseException) e.getTargetException();
            } else if (e.getTargetException() instanceof JumpException) {
                throw (JumpException) e.getTargetException();
            } else if (e.getTargetException() instanceof ThreadKill) {
            	// allow it to bubble up
            	throw (ThreadKill) e.getTargetException();
            } else if (e.getTargetException() instanceof Exception) {
                if(e.getTargetException() instanceof MainExitException) {
                    throw (RuntimeException)e.getTargetException();
                }
                recv.getRuntime().getJavaSupport().handleNativeException(e.getTargetException());
                return recv.getRuntime().getNil();
            } else {
                throw (Error) e.getTargetException();
            }
        } catch (IllegalAccessException e) {
            StringBuffer message = new StringBuffer();
            message.append(e.getMessage());
            message.append(':');
            message.append(" methodName=").append(methodName);
            message.append(" recv=").append(recv.toString());
            message.append(" type=").append(type.getName());
            message.append(" methodArgs=[");
            for (int i = 0; i < methodArgs.length; i++) {
                message.append(methodArgs[i]);
                message.append(' ');
            }
            message.append(']');
            assert false : message.toString();
            return null;
        } catch (final IllegalArgumentException e) {
/*            StringBuffer message = new StringBuffer();
            message.append(e.getMessage());
            message.append(':');
            message.append(" methodName=").append(methodName);
            message.append(" recv=").append(recv.toString());
            message.append(" type=").append(type.getName());
            message.append(" methodArgs=[");
            for (int i = 0; i < methodArgs.length; i++) {
                message.append(methodArgs[i]);
                message.append(' ');
            }
            message.append(']');*/
            assert false : e;
            return null;
        }
    }

    /**
     * Returns the arity of the wrapped Ruby method.
     */
    public Arity getArity() {
        return arity;
    }
}


3.2.3 在Java中编译执行Ruby脚本
public IRubyObject compileAndRun(String scriptTxt) {

		// Node node
		try {

			//获得全局的runtime对象(Ruby运行库环境)
			Ruby ruby = runtime;

			Script script = null;
			StringReader reader = new StringReader(scriptTxt);

			Node parsedScriptNode = ruby.parse(reader, scriptTxt,
						ruby.getCurrentContext().getCurrentScope(), 0);

			// do the compile
			StandardASMCompiler compiler = new StandardASMCompiler(
						parsedScriptNode);

			NodeCompilerFactory.getCompiler(parsedScriptNode).compile(
						parsedScriptNode, compiler);
			//加载类
			Class scriptClass = compiler.loadClass(new JRubyClassLoader());
			//生成script对象
			script = (Script) scriptClass.newInstance();

			ThreadContext tc = ruby.getCurrentContext();
			//执行脚本
			return script.run(tc, tc.getFrameSelf(), IRubyObject.NULL_ARRAY,
					Block.NULL_BLOCK);

		} catch (NotCompilableException nce) {
			System.err.println("Error -- Not compileable: " + nce.getMessage());
			return null;
		} catch (JumpException je) {
			if (je.getJumpType() == JumpException.JumpType.ReturnJump) {
				return (IRubyObject) je.getValue();
			} else {
				throw je;
			}
		} catch (ClassNotFoundException e) {
			System.err.println("Error -- Not compileable: " + e.getMessage());
			return null;
		} catch (InstantiationException e) {
			System.err.println("Error -- Not compileable: " + e.getMessage());
			return null;
		} catch (IllegalAccessException e) {
			System.err.println("Error -- Not compileable: " + e.getMessage());
			return null;
		}
	}


3.2.4 Java内嵌Ruby方法总结
结合以上各个步骤,我们就可以在Java中定义方法,然后在Ruby中调用它了,下面总结一下整个过程。
//示例Java类,其中的helloword方法将在Ruby中被调用
Class JavaSampleCls{
	public void helloworld(RubyString param){
		System.out.println(“hello ”+param.toString());
	}
}


	//初始化Ruby运行环境
	RubyInstanceConfig config = new RubyInstanceConfig();
	Ruby runtime = Ruby.newInstance(config);

	//初始化目标Java对象
	JavaSampleCls javasample=new JavaSampleCls();

	//把目标的普通Java对象转化为Ruby可识别的IRubyObject对象
	IRubyObject targetObject = JavaUtil.convertJavaToRuby(runtime,javasample);

	//定义callback对象: 类JavaSampleCls上的helloworld方法,一个String参数
	Callback  helloCallback = new IRSReflectionCallback(
				JavaSampleCls.class, "helloworld",
				new Class[] { RubyString.class }, false,// if restArg
				false, // if staticArg
				Arity.singleArgument(), true);// if fast

		//把callback定义到targetObject上
		targetObject.getMetaClass().defineMethod("helloworld", helloCallback);

		IAccessor d = new ValueAccessor(targetObject);

		//把targetObject绑定到Ruby运行库环境中
		runtime.getGlobalVariables().define("$vm", d);

		//声明Ruby脚本
		String rubycode=”$vm.helloworld(“java”)”;
		//编译执行
		compileAndRun(rubycode);

分享到:
评论

相关推荐

    IRS2453DPBF中文资料

    根据给定的文件信息,我们可以深入探讨IRS2453DPBF这款自振荡全桥驱动集成电路的关键特性、工作原理以及应用领域。 ### IRS2453DPBF概述 IRS2453DPBF是一款高性能的自振荡全桥驱动集成电路,基于广受欢迎的IR2153...

    IRS_IRS_

    2. **工作原理**:深入讨论了IRS如何通过调整其表面单元的相位,实现对入射无线信号的相位调控,进而增强信号或改变传播方向。 3. **性能优势**:阐述了IRS在无线通信中的优势,如提高信号强度、减少干扰、节省能源...

    环境卫星IRS数据预处理流程.

    辐射定标是IRS数据预处理的第一步,其目的是将原始传感器记录转换为物理意义明确的地表反射率或发射率。这一步骤对于后续的数据分析至关重要。 **扩展工具定标:** 1. **下载并安装扩展工具:** - 下载链接:...

    基于IRS2092S的逆变电源设计 (2015年)

    基于输入电压转换电路、正弦波发生电路和信号放大电路的原理设计了系统总体框架,重点探讨了该方法在硬件电路上的实现,并进行逆变电源性能测试。测试结果表明:该方法效率高,正弦波频率可通过简单的软件配置后进行调整...

    IRS2453D 自震荡全桥驱动芯片

    ### IRS2453D 自震荡全桥驱动芯片解析 #### 概述 IRS2453D是一款自震荡全桥驱动芯片,该芯片能够承受高达600伏特的电压,适用于高压环境下对功率器件进行高效驱动。该芯片继承了IR2153自震荡半桥驱动芯片的特点,...

    IRS系统使用手册.pdf

    IRS系统使用手册.pdf IRS系统使用手册.pdf是关于互联网报告系统的使用指南,本手册涵盖了网络数据挖掘、网页数据的结构与特点、网页数据挖掘的基本方法、智能网络爬虫等多方面的知识点。 网络数据挖掘 网络数据...

    irs-data trading system

    "IRS数据交易系统"是一个专为处理利率互换(Interest Rate Swap, IRS)交易而设计的平台。在构建这样的系统时,我们需要深入了解金融市场的运作机制,特别是利率互换这一金融衍生品的特性。利率互换是两个或多个实体...

    irs-server

    "irs-server"是一个基于Web服务技术构建的交易系统,它采用了Spring容器进行开发,并使用Java语言作为主要编程语言。这个系统的设计和实现充分利用了Java的强类型和面向对象特性,以及Spring框架的强大功能,旨在...

    利用DBMS与IRS实现中文全文检索得研究

    ### 利用DBMS与IRS实现中文全文检索的研究 #### 引言 在信息技术迅速发展的今天,数据库管理系统(DBMS)与信息存储检索系统(IRS)作为数据管理和检索的重要工具,各自发挥着不可或缺的作用。然而,面对海量非...

    IRS.rar_IRS_IRS classroom_IRS.rar_点名_课堂教学系统

    标题中的“IRS.rar_IRS_IRS classroom_IRS.rar_点名_课堂教学系统”指的是一个名为“IRS”的教学互动反馈系统,其相关文件被压缩在名为“IRS.rar”的压缩包内。这个系统主要用于课堂上的教学互动,如点名功能,使...

    MC34063AD+IRS2184 DC-Motor电机驱动板ALTIUM设计硬件原理图+PCB文件.zip

    在这个电机驱动板设计中,MC34063AD主要负责调节电源电压,控制电机的工作状态,而IRS2184则负责驱动电机的功率MOSFET,实现电机的正反转及速度控制。两者协同工作,确保了电机驱动板的高效运行。 硬件原理图是理解...

    IRS2334SPbF/MPbF:三相位芯片.pdf

    3. 三相位芯片(IRS2334SPbF/MPbF): - IR推出适用于反相电机驱动器的紧凑型三相位芯片,适合节能家电等应用。 - IRS2334SPbF采用SOIC20WB封装,而IRS2334MPbF采用QFN5X5封装,两种封装占用面积小,仅为25mm²。...

    199-一体化数字资源系统IRS架构及管理规范.pptx

    ### 一体化数字资源系统(IRS)架构及管理规范解析 #### 一、IRS概述 **一体化数字资源系统(IRS)**是一种先进的技术框架,旨在通过整合各类数字资源来支持高效的数字化改革进程。它作为一体化智能化公共数据平台的...

    智能反射面(IRS)在无线通信安全领域应用的论文复现-随N变化部分

    随着无线通信技术的飞速发展,智能反射面(IRS)作为该领域的一个新兴研究热点,正逐步成为提升通信安全性的重要工具。IRS,也被称作可重构智能表面,是一种通过编程调整表面单元反射相位的先进技术,能够智能操纵...

    Understanding_IRS_Math_and_pricing

    了解IRS的数学原理和定价机制对于市场参与者来说至关重要,尤其是对于涉及到IRS项目的发起人和管理人,他们需要清楚初始成本、市场价值和终止成本等相关费用。因此,本文件旨在提供一个基础的概览,帮助财资管理者和...

    B5G_6G通信的IRS技术综述.pdf

    在当今通信技术迅猛发展的背景下,第六代移动通信技术(6G)成为全球关注的焦点。随着第五代移动通信技术(5G)的普及与成熟,学术界、产业界和标准化组织已经开始对6G的愿景、需求和技术展开了深入研究。6G预计将...

    IRS图像字符识别系统

    "IRS图像字符识别系统"是一个基于VC++开发的软件,主要功能是识别图像中的中文、英文等字符。这个系统的高识别率超过90%,显示了其在光学字符识别(OCR,Optical Character Recognition)领域的优秀性能。OCR技术是...

    IRS301-10系列6轴机器人用户手册-机械篇.pdf

    **IRS301-10系列6轴机器人用户手册-机械篇** 汇川技术推出的IRS301-10系列6轴机器人专为各种工业应用设计,具备一系列显著特点,使其在众多领域表现出色。这款机器人拥有长达1422mm的最大臂长,能承载最大10kg的...

    IRS300-3系列6轴机器人用户手册-机械篇.pdf

    IRS300-3系列6轴机器人用户手册-机械篇 本文档是关于IRS300-3系列6轴机器人的用户手册-机械篇,提供了机器人的详细信息,包括安全注意事项、机器人的特点、应用场景、操作注意事项等。 一、安全注意事项 在使用...

Global site tag (gtag.js) - Google Analytics