当我运行我的 xsl 样式表时,我需要知道是否存在某些文件。因为 xslt 没有本地方法来测试文件是否存在 (ASFAIK),所以我想用 javascript 来做这件事。通常我使用的是 saxon 9 HE,但一些同事告诉我 HE 版本不支持 javascript 函数。有没有办法用 saxon 9 HE 执行 java 脚本函数?
我试过这个例子:如何在 xslt 中包含 javaScript 文件,但氧气给了我这个错误:“找不到匹配的 1-argument 函数命名”。
当我运行我的 xsl 样式表时,我需要知道是否存在某些文件。因为 xslt 没有本地方法来测试文件是否存在 (ASFAIK),所以我想用 javascript 来做这件事。通常我使用的是 saxon 9 HE,但一些同事告诉我 HE 版本不支持 javascript 函数。有没有办法用 saxon 9 HE 执行 java 脚本函数?
我试过这个例子:如何在 xslt 中包含 javaScript 文件,但氧气给了我这个错误:“找不到匹配的 1-argument 函数命名”。
您可以使用 XPath 函数doc-available
https://www.w3.org/TR/xpath-functions/#func-doc-available和unparsed-text-available
https://www.w3.org/TR/xpath-functions/#func-unparsed- text-available检查是否存在 XML 文档或非 XML 文本文档。
不支持在 Saxon 9 中使用 Javascript(并且一个简单的 ECMAScript/Javascript 引擎通常不包含任何文件 IO 功能)但是有一个称为集成扩展功能的功能http://saxonica.com/html/documentation/extensibility /integratedfunctions/您可以在 Saxon 9(所有版本)中使用来调用 Java 代码。在 Saxon 9 PE 和 EE 中,您还可以使用自反扩展函数直接从 XSLT 代码调用 Java 代码。
再次感谢马丁,你真的帮助了我!我用集成的扩展函数特性编写了自己的自定义 xslt 函数。该函数调用一个 java 方法,该方法测试文件是否存在于给定目录中并返回 true 或 false。对于那些需要“集成扩展功能”功能的工作示例,或者甚至想用 saxon-9-HE 测试文件是否存在的人,我将分享我的简单解决方案。
当调用 xslt 函数时,定义 xslt 函数名称、参数和返回类型并包含要调用的 java 方法的 Java 类:
package de.mypackage.xsltfunctions;
import java.io.File;
import net.sf.saxon.expr.XPathContext;
import net.sf.saxon.lib.ExtensionFunctionCall;
import net.sf.saxon.lib.ExtensionFunctionDefinition;
import net.sf.saxon.om.Sequence;
import net.sf.saxon.om.StructuredQName;
import net.sf.saxon.trans.XPathException;
import net.sf.saxon.value.BooleanValue;
import net.sf.saxon.value.SequenceType;
import net.sf.saxon.value.StringValue;
public class FileExists extends ExtensionFunctionDefinition {
@Override
public StructuredQName getFunctionQName() {
return new StructuredQName("file", "http://mydomain.de/xslt/filesystem", "file-exists");
}
@Override
public SequenceType[] getArgumentTypes() {
return new SequenceType[] { SequenceType.SINGLE_STRING, SequenceType.SINGLE_STRING };
}
@Override
public SequenceType getResultType(final SequenceType[] suppliedArgumentTypes) {
return SequenceType.SINGLE_BOOLEAN;
}
@Override
public ExtensionFunctionCall makeCallExpression() {
return new ExtensionFunctionCall() {
@Override
public Sequence call(final XPathContext context, final Sequence[] arguments)
throws XPathException {
String searchDir = ((StringValue) arguments[0]).getStringValue();
String fileName = ((StringValue) arguments[1]).getStringValue();
if (!new File(searchDir).isDirectory()) {
throw new XPathException(
"First argument \"" + searchDir + "\" is not a directory or cannot be found!");
}
return BooleanValue.get(new File(searchDir + fileName).exists());
}
};
}
}
为 saxon 处理器注册自定义 xslt 函数的代码片段:
import java.io.StringWriter;
import de.mypackage.xsltfunctions.FileExists;
import net.sf.saxon.TransformerFactoryImpl;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.stream.StreamResult;
import javax.xml.transform.stream.StreamSource;
StringWriter xmlResultResource = new StringWriter();
System.setProperty("javax.xml.transform.TransformerFactory","net.sf.saxon.TransformerFactoryImpl");
TransformerFactory factory = TransformerFactory.newInstance();
TransformerFactoryImpl tFactoryImpl = (TransformerFactoryImpl) factory;
net.sf.saxon.Configuration saxonConfig = tFactoryImpl.getConfiguration();
saxonConfig.registerExtensionFunction(new FileExists());
Transformer transformer = factory.newTransformer(new StreamSource(getXslFile()));
transformer.transform(new StreamSource(xmlFileInput), new StreamResult(xmlResultResource));
String result = xmlResultResource.getBuffer().toString();