欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 新闻 > 社会 > 修改MVCActiveRecord支持匿名函数(用于动态决定数据库连接)

修改MVCActiveRecord支持匿名函数(用于动态决定数据库连接)

2024/12/27 6:19:40 来源:https://blog.csdn.net/redhat588/article/details/144223318  浏览:    关键词:修改MVCActiveRecord支持匿名函数(用于动态决定数据库连接)

要修改 TMVCActiveRecordMiddleware 以直接接受一个匿名函数(用于动态决定数据库连接)以及一个配置文件名,你需要对构造函数进行一些调整。这可以通过重载构造函数以接收另一个参数——匿名函数来实现。

构造函数修改步骤

假设你的目标是允许传入一个匿名函数,用于在运行时选择数据库连接名称。对于 FireDAC 的 ActiveRecord 支持,你可能需要一个逻辑,在中间件初始化时通过连接配置文件动态加载连接定义。

以下是如何修改和添加新的构造函数:

  1. 定义合适的匿名函数类型:假设你希望使用 TFunc<TWebContext, string> 类型的函数来动态提供数据库连接定义名称。
  2. 修改构造函数以接受匿名函数:添加新的构造函数。

修改后的代码

type// 新增的匿名函数类型,用于获取连接定义名称TConnectionDefNameResolver = reference to function(AContext: TWebContext): string;TMVCActiveRecordMiddleware = class(TInterfacedObject, IMVCMiddleware)privatefDefaultConnectionDefName: string;fConnectionDefFileName: string;fConnectionLoaded: Boolean;fResolver: TConnectionDefNameResolver; // 新增成员protectedprocedure EnsureConnection(AContext: TWebContext);// ...public// 新的构造函数,接收匿名函数constructor Create(const Resolver: TConnectionDefNameResolver;const ConnectionDefFileName: string = 'FDConnectionDefs.ini'); overload;// ...end;constructor TMVCActiveRecordMiddleware.Create(const Resolver: TConnectionDefNameResolver;const ConnectionDefFileName: string);
begininherited Create;fConnectionLoaded := False;fResolver := Resolver; // 赋值匿名函数fConnectionDefFileName := ConnectionDefFileName;
end;procedure TMVCActiveRecordMiddleware.EnsureConnection(AContext: TWebContext);
varConnectionDefName: string;
beginif fConnectionLoaded thenbeginExit;end;TMonitor.Enter(Self);tryif fConnectionLoaded thenbeginExit;end;if TInterlocked.CompareExchange(gCONNECTION_DEF_FILE_LOADED, 1, 0) = 0 thenbeginFDManager.ConnectionDefFileAutoLoad := False;FDManager.ConnectionDefFileName := fConnectionDefFileName;if not FDManager.ConnectionDefFileLoaded thenbeginFDManager.LoadConnectionDefFile;end;end;// 使用匿名函数动态获取连接定义名称if Assigned(fResolver) thenbeginConnectionDefName := fResolver(AContext);if FDManager.IsConnectionDef(ConnectionDefName) thenbeginActiveRecordConnectionsRegistry.AddDefaultConnection(ConnectionDefName);endelsebeginraise EMVCConfigException.CreateFmt('ConnectionDefName "%s" not found in config file "%s"',[ConnectionDefName, FDManager.ActualConnectionDefFileName]);end;end;fConnectionLoaded := True;finallyTMonitor.Exit(Self);end;
end;procedure TMVCActiveRecordMiddleware.OnBeforeRouting(AContext: TWebContext; var AHandled: Boolean);
begin// 用于确保每次请求都动态评价连接名EnsureConnection(AContext);AHandled := False;
end;

在 Web Module 中的配置

使用这个新的构造函数,可以在 WebModuleCreate 中注册中间件:

procedure TMyWebModule.WebModuleCreate(Sender: TObject);
beginFMVC := TMVCEngine.Create(Self,procedure(Config: TMVCConfig)begin// 通用配置end);// 注册中间件并传递匿名函数FMVC.AddMiddleware(TMVCActiveRecordMiddleware.Create(function(AContext: TWebContext): stringvarDBIdentifier: string;begin// 身份认证的检查if not AContext.LoggedUser.IsValid thenraise EMVCException.Create(HTTP_STATUS.Unauthorized, 'Unauthorized access');// 从请求中确定数据库名称DBIdentifier := AContext.Request.BodyParam('database_name');if DBIdentifier = '' thenraise EMVCException.Create(HTTP_STATUS.BadRequest, 'Database name must be provided');if not IsDatabaseAllowed(DBIdentifier) thenraise EMVCException.Create(HTTP_STATUS.Forbidden, 'Access to this database is not allowed');Result := DBIdentifier;  // 返回有效的连接定义名称end,'FDConnectionDefs.ini'  // 连接定义文件));// 其他中间件注册和配置
end;

注意事项

  • 错误处理:确保在匿名函数中和配置检查中有适当的错误处理机制。
  • 线程安全:考虑多线程环境下对连接的访问和修改是否安全。
  • 性能:确保在高并发环境下性能最佳,尽量减少锁定时间和资源占用。

通过这些修改,你可以让 TMVCActiveRecordMiddleware 更加灵活地根据请求上下文动态选择数据库连接。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com