由于反射加载不再适用于.NET核心,所以我在System.Reflection.Metadata上实现了一个包装器,允许我扫描程序集中包含的类型,而不需要完整的程序集加载。
虽然MetadataReader基本上是工作的,但我实现它的步骤如下:
private unsafe MetadataReader LoadMetadataReader(
string filename,
MetadataReaderOptions options = MetadataReaderOptions.Default,
MetadataStringDecoder decoder = null)
{
buffer = File.ReadAllBytes(filename);
pinnedHandle = GCHandle.Alloc(buffer, GCHandleType.Pinned);
var headers = new PEHeaders(new MemoryStream(buffer));
var startOffset = headers.MetadataStartOffset;
var metaDataStart = (byte*)pinnedHandle.AddrOfPinnedObject() + startOffset;
return new MetadataReader(metaDataStart, headers.MetadataSize, options, decoder);
}这意味着每次我都要加载程序集的所有字节。
我的问题是:是否只有从读取元数据所需的文件中识别和加载字节子集的方法??
发布于 2018-02-17 19:11:38
感谢汉斯·帕桑特的评论,我想出了如下结论:
以内存映射文件的形式加载程序集
private static MemoryMappedFile LoadAssembly(string filename, out long length, out MemoryMappedFileAccess access)
{
// Setup parameters to pass in
return MemoryMappedFile.CreateFromFile(filename, mode, mapName, length, access);
}using (var file = LoadAssembly(filename, out var length, out var access)) {
using (var stream = file.CreateViewStream(0, length, access)) {通过PEHeaders对象从流中获取标头的尺寸
var headers = new PEHeaders(stream);然后,我能够从流中获得指向文件开始的实际指针,并将其传递到MetadataReader构造函数中。
var start = (byte*)0;
stream.SafeMemoryMappedViewHandle.AcquirePointer(ref start);
var size = headers.MetadataSize;
var reader = new MetadataReader(start + headers.MetadataStartOffset, size, options, decoder);https://stackoverflow.com/questions/48842294
复制相似问题