我在运行时使用c#创建.NET 4.7.2中的TypeBuilder类。问题是,该类型的DLL存储在当前应用程序的根文件夹中。问题是,用户通常没有对本地根文件夹的写访问权限。
那么,如何设置应该保存和加载程序集的位置?什么样的用户文件夹适合这样做呢?
当前代码:
private static Type CreateRaportType(List<PropertieInformation> propertieList, string className)
{
AssemblyName assemblyName;
AssemblyBuilder assemblyBuilder;
ModuleBuilder module;
TypeBuilder typeBuilder;
FieldBuilder field;
PropertyBuilder property;
MethodAttributes GetSetAttr;
MethodBuilder currGetPropMthdBldr;
MethodBuilder currSetPropMthdBldr;
ILGenerator currGetIL;
ILGenerator currSetIL;
Type caType;
CustomAttributeBuilder caBuilder;
List<Object> objList = new List<object>();
assemblyName = new AssemblyName();
assemblyName.Name = "ReportAssembly";
assemblyBuilder = Thread.GetDomain().DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.Run);
module = assemblyBuilder.DefineDynamicModule("ReportModule");
typeBuilder = module.DefineType(className, TypeAttributes.Public | TypeAttributes.Class, typeof(GeneratedClassBase));
foreach (PropertieInformation propertieInfo in propertieList)
{
field = typeBuilder.DefineField("_" + propertieInfo.PropertieName, propertieInfo.PropertieType, FieldAttributes.Private);
property = typeBuilder.DefineProperty(propertieInfo.PropertieName, PropertyAttributes.None, propertieInfo.PropertieType, new Type[] { propertieInfo.PropertieType });
GetSetAttr = MethodAttributes.Public | MethodAttributes.HideBySig;
currGetPropMthdBldr = typeBuilder.DefineMethod("get_value", GetSetAttr, propertieInfo.PropertieType, Type.EmptyTypes);
currGetIL = currGetPropMthdBldr.GetILGenerator();
currGetIL.Emit(OpCodes.Ldarg_0);
currGetIL.Emit(OpCodes.Ldfld, field);
currGetIL.Emit(OpCodes.Ret);
currSetPropMthdBldr = typeBuilder.DefineMethod("set_value", GetSetAttr, null, new Type[] { propertieInfo.PropertieType });
currSetIL = currSetPropMthdBldr.GetILGenerator();
currSetIL.Emit(OpCodes.Ldarg_0);
currSetIL.Emit(OpCodes.Ldarg_1);
currSetIL.Emit(OpCodes.Stfld, field);
currSetIL.Emit(OpCodes.Ret);
// Last, we must map the two methods created above to our PropertyBuilder to
// their corresponding behaviors, "get" and "set" respectively.
property.SetGetMethod(currGetPropMthdBldr);
property.SetSetMethod(currSetPropMthdBldr);
caType = typeof(Reportable);
objList.Clear();
objList.Add(propertieInfo.MemberToDataBind);
objList.Add(propertieInfo.ControlToUse);
objList.Add(propertieInfo.PropertieName);
if (propertieInfo.ControlToUse == ControlToUse.SystemItemTable)
{
objList.Add(propertieInfo.PropertieInnerCollectionType);
objList.Add(propertieInfo.PropertieInnerCollectionName);
objList.Add(propertieInfo.DisplayName);
objList.Add(-1);
objList.Add(FieldListIcon.UnUsedItem);
objList.Add(propertieInfo.SystemItemKey);
objList.Add(null);
}
else if (propertieInfo.ControlToUse == ControlToUse.AttributeTable)
{
objList.Add(null);
objList.Add(null);
objList.Add(null);
objList.Add(propertieInfo.MultiAttributeId);
objList.Add(FieldListIcon.UnUsedItem);
objList.Add(null);
objList.Add(null);
}
else if (propertieInfo.ControlToUse == ControlToUse.GUIGroupTable)
{
objList.Add(propertieInfo.PropertieInnerCollectionType);
objList.Add(propertieInfo.PropertieInnerCollectionName);
objList.Add(propertieInfo.DisplayName);
objList.Add(-1);
objList.Add(FieldListIcon.UnUsedItem);
objList.Add(null);
objList.Add(propertieInfo.GuiGroupKey);
}
else
{
objList.Add(null);
objList.Add(null);
objList.Add(null);
objList.Add(-1);
objList.Add(FieldListIcon.UnUsedItem);
objList.Add(null);
objList.Add(null);
}
var conInfo = caType.GetConstructor(Type.EmptyTypes);
var conArgs = new object[] { };
var caTypeFields = caType.GetFields();
caBuilder = new CustomAttributeBuilder(conInfo, conArgs, caTypeFields, objList.ToArray());
property.SetCustomAttribute(caBuilder);
caType = typeof(DisplayNameAttribute);
if (propertieInfo.IsList)
caBuilder = new CustomAttributeBuilder(caType.GetConstructor(new Type[] { typeof(string) }), new string[] { propertieInfo.DisplayName });
else
caBuilder = new CustomAttributeBuilder(caType.GetConstructor(new Type[] { typeof(string) }), new string[] { propertieInfo.DisplayName });
property.SetCustomAttribute(caBuilder);
}
return typeBuilder.CreateType();
}发布于 2019-05-21 13:33:16
要将动态程序集保存到自定义文件夹,可以使用AppDomain的重载AppDomain方法。
AppDomain currDom = AppDomain.CurrentDomain;
AssemblyName aName = new AssemblyName();
aName.Name = "YouAssemblyName";
string moduleName = aName.Name + ".dll";
AssemblyBuilder ab = currDom.DefineDynamicAssembly(aName, AssemblyBuilderAccess.Save, @"FOLDER\TO\YOUR\ASSEMBLY");
ModuleBuilder mb = ab.DefineDynamicModule(aName.Name, moduleName);
TypeBuilder tb = mb.DefineType("Example");
tb.CreateType();
// type definition here
ab.Save(moduleName);要从自定义位置自动加载程序集,可以使用TypeResolve事件of AppDomain
AppDomain.CurrentDomain.TypeResolve += CurrentDomain_TypeResolve;
Assembly CurrentDomain_TypeResolve(object sender, ResolveEventArgs args)
{
return Assembly.LoadFrom(/*path to your saved assembly*/);
}用于此目的的好文件夹是: driveletter:\Users\AppData\Local\YourFolder。使用GetFolderPath方法获取其路径:
Environment.GetFolderPath(Environment.SpecialFolder.LocalApplicationData); 附注:我稍微修改了这个文档中的示例:https://learn.microsoft.com/en-us/dotnet/api/system.appdomain.typeresolve?view=netframework-4.8
更新:
最初的问题是如何将动态程序集保存/加载到自定义位置。但是,如果需要在一个AssemblyBuilder.DefineDynamicAssembly中定义动态模块和内存中的类型,并且不需要保存动态程序集,那么使用AssemblyBuilderAccess.Run调用AssemblyBuilderAccess.Run并使用AssemblyBuilder.DefineDynamicModule(string)方法定义瞬态动态模块就足够了。最后,您应该调用AssemblyBuilder.GetType(string,bool)来获取您的类型。
代码会是这样的
AppDomain currDom = AppDomain.CurrentDomain;
AssemblyName aName = new AssemblyName();
aName.Name = "YourAssemblyName";
AssemblyBuilder assemblyBuilder = currDom.DefineDynamicAssembly(aName, AssemblyBuilderAccess.Run); // We need Run access, don't need Save access
ModuleBuilder mb = assemblyBuilder.DefineDynamicModule(aName.Name); // Here we define transient dynamic module in assembly
TypeBuilder tb = mb.DefineType("YourTypeName");
tb.CreateType();
// Define your type definition here
// ...
Type t = assemblyBuilder.GetType("YourTypeName", true); // Load created type using our AssemblyBuilder instance
Console.WriteLine("Loaded type \"{0}\".", t);
Object o = Activator.CreateInstance(t); // Create object of loaded type
Console.WriteLine("Created object \"{0}\".", o.ToString());https://stackoverflow.com/questions/56238979
复制相似问题