- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
我正在尝试使用 typescript 编译器 API 编写 typescript 编译器转换。然而,当创建新的标识符节点时,即使节点被发送到最终的 .js 文件,它们似乎缺少符号绑定(bind)信息,因此最终输出不正确。
假设我有以下程序:
A.ts
export class A {
static myMethod() {
return 'value';
}
}
索引.ts
import { A } from './A';
export function main() {
const value1 = 'replaceMe';
const value2 = A.myMethod();
const equals = value1 == value2;
}
假设我尝试用下面的转换器编译上面的程序:
function transformer(program: ts.Program): ts.TransformerFactory<ts.SourceFile> {
return (context: ts.TransformationContext) => (file: ts.SourceFile) => transformFile(program, context, file);
}
function transformFile(program: ts.Program, context: ts.TransformationContext, file: ts.SourceFile): ts.SourceFile {
const transformedFile = ts.visitEachChild(file, child => visit(child, context), context);
console.log(ts.createPrinter().printFile(transformedFile));
return transformedFile;
}
function visit(node: ts.Node, context: ts.TransformationContext): ts.Node {
if (ts.isStringLiteral(node) && node.text == 'replaceMe') {
return ts.createCall(
ts.createPropertyAccess(
ts.createIdentifier('A'),
'myMethod'),
[],
[]);
}
return ts.visitEachChild(node, child => visit(child, context), context);
}
中间 AST 在 pretty-print 时实际上看起来是正确的:
import { A } from './A';
export function main() {
const value1 = A.myMethod();
const value2 = A.myMethod();
const equals = value1 == value2;
}
但输出的 javascript 不正确:
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
var A_1 = require("./A");
function main() {
var value1 = A.myMethod();
var value2 = A_1.A.myMethod();
var equals = value1 == value2;
}
exports.main = main;
我理解这可能是因为通过使用 ts.createIdentitier('A')
创建一个新标识符,这个新标识符没有绑定(bind)到与另一个 A< 相同的符号
同一文件中的标识符。
有没有办法使用公共(public)编译器 API 将新标识符绑定(bind)到现有符号?
最佳答案
Typescript 编译分阶段进行(解析、绑定(bind)、类型检查、发出,更详细一点 here )。您可以使用前一阶段的信息,但通常不能更改它。您可以在发射阶段执行的转换旨在允许您将 AST 从 Typescript 带到 Javascript,而不是重构代码。
实现目标的一种方法是创建程序,应用转换,然后使用修改后的代码创建一个新程序,尽可能多地重用原始程序(重用相同的SourceFile
没有发生变化)
function transformFile(program: ts.Program, file: ts.SourceFile): ts.SourceFile {
let empty = ()=> {};
// Dummy transformation context
let context: ts.TransformationContext = {
startLexicalEnvironment: empty,
suspendLexicalEnvironment: empty,
resumeLexicalEnvironment: empty,
endLexicalEnvironment: ()=> [],
getCompilerOptions: ()=> program.getCompilerOptions(),
hoistFunctionDeclaration: empty,
hoistVariableDeclaration: empty,
readEmitHelpers: ()=>undefined,
requestEmitHelper: empty,
enableEmitNotification: empty,
enableSubstitution: empty,
isEmitNotificationEnabled: ()=> false,
isSubstitutionEnabled: ()=> false,
onEmitNode: empty,
onSubstituteNode: (hint, node)=>node,
};
const transformedFile = ts.visitEachChild(file, child => visit(child, context), context);
return transformedFile;
}
function visit(node: ts.Node, context: ts.TransformationContext): ts.Node {
if (ts.isStringLiteral(node) && node.text == 'replaceMe') {
return ts.createCall(
ts.createPropertyAccess(
ts.createIdentifier('A'),
'myMethod'),
[],
[]);
}
return ts.visitEachChild(node, child => visit(child, context), context);
}
let host = ts.createCompilerHost({});
let program = ts.createProgram(["toTrans.ts"], {}, host)
let transformed = program.getSourceFiles()
.map(f=> ({ original: f, transformed: transformFile(program, f) }))
.reduce<{ [name: string] : {original: ts.SourceFile, transformed: ts.SourceFile }}>((r, f)=> { r[f.original.fileName] = f; return r; }, {});
let originalGetSourceFile = host.getSourceFile;
let printer = ts.createPrinter();
// Rig the host to return the new verisons of transformed files.
host.getSourceFile = function(fileName, languageVersion, onError, shouldCreateNewSourceFile){
let file = transformed[fileName];
if(file){
if(file.original != file.transformed){
// Since we need to return a SourceFile it is tempting to return the transformed source file and not parse it again
// The compiler doe not support Synthesized nodes in the AST except during emit, and it will check node positions
// (which for Synthesized are -1) and fail. So we need to reparse
return ts.createSourceFile(fileName, printer.printFile(file.transformed), languageVersion);
} else {
// For unchanged files it should be safe to reuse the source file
return file.original;
}
}
return originalGetSourceFile(fileName, languageVersion, onError, shouldCreateNewSourceFile);
}
// Recreate the program, we pass in the original to
program = ts.createProgram(["toTrans.ts"], {}, host, program);
var result = program.emit();
另一种方法是使用语言服务并通过语言服务应用这些更改,但老实说我对编译器的那部分没有经验,它似乎比这种方法更复杂。
关于typescript - 如何在 typescript 中的编译器转换器上将标识符与现有符号绑定(bind)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49139601/
我知道这在 Linux 上是可能的。我尝试使用 open("E:", 0); 和 open("E:\\", 0); 但它返回为 -1。我想将 DVD 作为一个大文件来读取,而不是将其用作文件系统。 最
我正在尝试编译一个包含 CUDA 代码的小型库。 我已成功将其编译为共享库,但我真正需要的是静态库。 我有两个源文件: main.c:包含一个用C编写的测试函数。我用gcc编译这个文件 mai
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎与 help center 中定义的范围内的编程无关。 . 关闭 9 年前。 Improve
我正在使用 MingW 在 Windows 上编写 C 程序,并希望使用 EXPAT XML 库。我想静态编译我的程序,所以我需要静态 .a 库。 有什么方法可以将 EXPAT 编译成 Windows
我想将结果限制为 KEY_HOMEID 等于 journalId 的结果。我已经研究了几天,如有任何帮助,我们将不胜感激。 public Cursor fetchAllNotes(String jou
我一直在寻找这个信息,但是由于可以通过 homebrew 和 pip 安装额外的包和 python 版本,我感觉我的环境很乱向上。此外,很久以前,我用 sudo pip install 和 sudo
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a softwar
我正在尝试合并目录中的所有 *.pdf : gswin64c -q -dNOPAUSE -sDEVICE=pdfwrite -sOutputFile=Total_Files.pdf -dBATCH *
所以我有一个简单的图像缩略图着陆页,例如: 在我的网站上,4 个并排显示在桌面上。如何强制它们在移动设备 View 中成对出现在一行中? 所以:桌面: #### 手机: ## ## 最佳答案
我正在使用 Ubuntu 21.04。我已删除 /usr/bin/python3和 /usr/lib/python3/因为某些软件包在二进制文件中出现错误。我的意思是重新安装python3到一个新的状
是否可以在 Windows 上使用 gyp 将 googles v8 构建为共享库(msvc 2012)?我试过的一切都不起作用。我试过的: python build\gyp_v8 -Dcompone
我需要将 rubygems 从 1.3.5 更新到 1.4.2 但显然 rubygems update 只是将您更新到最新版本 如何更新到 1.4.2? 最佳答案 您可以使用 RVM 安装特定
我还没有找到太多关于它的信息,但我看到了一些提示,表明可以在 iPhone 应用程序中使用 NSTask。如果可能的话,我将如何去做? 我不想越狱我的 iPhone,但我正在开发的应用程序仅供内部使用
我在 UIWebView 中有一个 map 图像。它默认加载在左上角。我希望它在 UIWebView 的中心初始化。 有人知道怎么做吗? 谢谢! 最佳答案 如果 map 图像是页面中唯一的内容,它是否
如何公开 NodePort 类型的服务上网没有 使用类型 LoadBalancer ?我发现的每个资源都是通过使用负载均衡器来完成的。但我不希望负载平衡对我的用例来说既昂贵又不必要,因为我正在运行 p
是否可以将 View 变成可编辑的,例如 this image ? 我知道我可以使用 GridView 来做到这一点。但是,我正在尝试使用 TableRows 来做到这一点,这可能吗? 编辑:我真正想
假设我已将 Heroku 应用程序扩展为 1 个工作进程,但如何指定具有特定名称的 rake 任务应作为工作进程运行? 最佳答案 在你的项目中创建一个 Procfile,然后像这样将 rake 任务放
目前,我在 GitHub 上一个项目的 README.md 文件中使用此 Markdown 文本: See the docs of [testthat][3] on how to write unit
我正在尝试使用一些到 uint8_t 的转换将 IPv4 转换为 IPv6。我知道 IPv4 有 4 个字节,IPv6 有 2 个字节的 16 个无符号整数,但我找不到它们转换的方法。 #includ
我是编程新手,目前正在学习 C。您能帮我解决以下案例吗? 一个例子是,如果用户输入“cbamike”,我想将其分成两个字符串:cba 和 mike。 我尝试了下面的代码,但它不起作用: #includ
我是一名优秀的程序员,十分优秀!