当前位置: 首页 > news >正文

学习设计模式《十》——代理模式

一、基础概念

        代理模式的本质【控制对象访问】;

        代理模式的定义:为其他对象提供一种代理以控制对这个对象的访问;

        代理模式的功能:代理模式是通过创建一个代理对象,用这个代理对象去代表真实的对象;客户端得到这个代理对象后,对客户端没有什么影响,就跟得到了真实对象一样来使用【当客户端操作整个代理对象的时候,实际上功能最终还是会由真实的对象来完成,只不过是由通过代理操作的,也就是客户端操作代理,代理操作真正的对象】正是因为有代理对象夹在客户端和被代理的真实对象中间,相当于一个中转,那么在中转的时候就有很多花招可以使用了(如:判断权限,若没有足够权限就不给你中转等等)。

代理的分类
序号代理分类说明
1虚代理根据需要来创建开销很大的对象,该对象只有在需要的时候才会被真正创建
2远程代理用来在不同的地址空间上代表同一个对象,这个不同的地址空间可以是在本机,也可以在其他机器上
3Copy-on-Write代理在客户端操作的时候,只有对象改变了,才会真的拷贝(或克隆)一个目标对象,算是虚代理的一个分支
4保护代理控制对原始对象的访问,如果有需要,可以给不同的用户提供不同的访问权限,以控制他们对原始对象的访问
5Cache代理为那些昂贵操作的结果提供临时的存储空间,以便多个客户端可以共享这些结果
6防火墙代理保护对象不被恶意用户访问和操作
7同步代理使多个用户能够同时访问目标对象而没有冲突
8智能指引在访问对象时执行一些附加操作(如:对指向对象的引用计数、第一次引用一个持久对象时,将它装入内存等)

        何时选用代理模式?      
                1、需要为一个对象在不同的地址空间提供局部代表的时候,可以使用远程代理;
                2、需要按照需要创建开销很大的对象的时候,可以使用虚代理;
                3、需要控制对原始对象的访问的时候,可以使用保护代理;
                4、需要在访问对象执行一些附加操作的时候,可以使用智能指引代理。

二、代理模式示例

        业务需求:在HR项目中,客户提出当选择一个部门或分公司的时候,要把该部门或分公司下的所有员工都显示出来(且只需要显示用户的名称即可),而且不要翻页,方便他们进行业务处理;但是当点击某个员工时,可查看该员工的详细信息。

  2.1、不使用任何模式的示例

        直接使用sql语句将指定部门下关联的所有员工信息都获取出来即可:

        2.1.1、准备工作:

        (为了方便获取某个部门或者某个分公司下的所有员工信息,设计部门编号的时候,就是按照层级来进行编码,如:母公司编码为01,下面的分公司就是0101、0102、0103以此类推,在下一级的公司部门编号就是:010101、010102、010103,...;010201、010202、010203,...这样的部门编码设计虽然不优雅,但是实用,像这种获取某个部门或某个分公司下的所有员工信息功能,就不用递归查找,直接使用like匹配部门编号开头即可)。

在sqlserver中创建部门表与用户信息表:

--创建部门表
CREATE TABLE [dbo].[Depment]([ID] [nvarchar](10) PRIMARY KEY, NOT NULL,[DepmentName] [nvarchar](10) NOT NULL,[DepmentDesc] [nvarchar](100) NULL
);INSERT INTO [dbo].[Depment]([ID], [DepmentName], [DepmentDesc]) VALUES (N'01', N'总公司', N'这是公司总部,管理旗下所有公司');
INSERT INTO [dbo].[Depment]([ID], [DepmentName], [DepmentDesc]) VALUES (N'0101', N'一分公司', N'这是公司在上海的第一个分公司 ');
INSERT INTO [dbo].[Depment]([ID], [DepmentName], [DepmentDesc]) VALUES (N'0102', N'二分公司', N'这是公司在苏州的第二个分公司');
INSERT INTO [dbo].[Depment]([ID], [DepmentName], [DepmentDesc]) VALUES (N'0103', N'三分公司', N'这是公司在深圳的第三个分公司');
INSERT INTO [dbo].[Depment]([ID], [DepmentName], [DepmentDesc]) VALUES (N'010101', N'信息部', N'这是公司在上海的第一个分公司的下属信息部,主要复杂公司技术相关的所有事物(如软件产品研发、实施部署运维等相关工作)');
INSERT INTO [dbo].[Depment]([ID], [DepmentName], [DepmentDesc]) VALUES (N'010102', N'人力资源部', N'这是公司在上海的第一分公司的下属人力资源部门,主要负责公司所需人才的招聘、管理、薪酬、岗位职责、培训、离职等相关工作');
INSERT INTO [dbo].[Depment]([ID], [DepmentName], [DepmentDesc]) VALUES (N'010201', N'行政部', N'这是公司在苏州的第二个分公司的下属行政部门,主要负责公司的行政事务(如:商务接待、会议安排、食堂管理、安保管理等相关工作)');
INSERT INTO [dbo].[Depment]([ID], [DepmentName], [DepmentDesc]) VALUES (N'010202', N'销售部', N'这是公司在苏州的第二个分公司的下属销售部门,主要负责公司产品的销售推广(如:对外推广公司产品、扩大公司产品的受众、让公司产品打开销路等相关工作)');
--创建用户信息表
CREATE TABLE [dbo].[UserInfo]([ID] [nvarchar](10) PRIMARY KEY,NOT NULL,[UserID] [nchar](18) NOT NULL,[UserName] [varchar](50) NOT NULL,[UserSex] [char](2) NOT NULL,[UserAge] [int] NOT NULL,[UserTelnumber] [nchar](11) NOT NULL,[UserHeight] [int] NULL,[UserWeight] [int] NULL,[UserAddress] [nchar](30) NULL,[DepmentId] [varchar](50) NOT NULL
);INSERT INTO [dbo].[UserInfo]([ID], [UserID], [UserName], [UserSex], [UserAge], [UserTelnumber], [UserHeight], [UserWeight], [UserAddress], [DepmentId]) VALUES (N'001', N'522020199001124561', '张三', '男', 35, N'14236598541', 168, 70, NULL, '010101');
INSERT INTO [dbo].[UserInfo]([ID], [UserID], [UserName], [UserSex], [UserAge], [UserTelnumber], [UserHeight], [UserWeight], [UserAddress], [DepmentId]) VALUES (N'002', N'522020199101124452', '李四', '男', 34, N'14238765541', 170, 72, NULL, '010101');
INSERT INTO [dbo].[UserInfo]([ID], [UserID], [UserName], [UserSex], [UserAge], [UserTelnumber], [UserHeight], [UserWeight], [UserAddress], [DepmentId]) VALUES (N'003', N'00201020000606345X', '王茜', '女', 25, N'13522687451', 166, 50, NULL, '010102');
INSERT INTO [dbo].[UserInfo]([ID], [UserID], [UserName], [UserSex], [UserAge], [UserTelnumber], [UserHeight], [UserWeight], [UserAddress], [DepmentId]) VALUES (N'004', N'00201020010507234X', '思雨', '女', 24, N'14587653241', 165, 50, NULL, '010201');
INSERT INTO [dbo].[UserInfo]([ID], [UserID], [UserName], [UserSex], [UserAge], [UserTelnumber], [UserHeight], [UserWeight], [UserAddress], [DepmentId]) VALUES (N'005', N'222021199506013756', '王五', '男', 30, N'18623597461', 169, 71, NULL, '010202');

              2.1.2、正式不用任何模式实现业务示例

1、创建用户信息对象

using System;
using System.Collections.Generic;
using System.Linq;
using System.Security;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern
{/// <summary>/// 用户对象/// </summary>internal class UserModel{//用户编号public string? Id { get; set; }//用户身份证编号public string? UserId { get; set; }//用户名称public string? UserName { get; set; }//用户性别public string? UserSex { get; set; }//用户年龄public string? UserAge { get; set; }//联系电话public string? UserTelnumber { get; set; }//用户身高public string? UserHeight { get; set; }//用户体重public string? UserWeight { get; set; }//用户地址public string? UserAddress { get; set; }//用户所属部门编号public string? DepmentId { get; set; }public override string ToString(){string str = $"用户的身份证编号【{UserId}】姓名【{UserName}】性别【{UserSex}】年龄【{UserAge}】" +$"联系电话【{UserTelnumber}】身高【{UserHeight}】体重【{UserWeight}】地址【{UserAddress}】所属部门编号【{DepmentId}】";return str;}}//Class_end
}

2、创建用户管理对象操作数据库中的用户信息

/***
*	Title:"WinFormClient" 项目
*		主题:通用的SQLServer数据库操作类
*	Description:
*		功能:实现数据库的基础增、删、查、改操作
*	Date:2025
*	Version:0.1版本
*	Author:Coffee
*	Modify Recoder:
*/using System;
using System.Collections.Generic;
using System.Data;
using System.Data.SqlClient;namespace ProxyPattern
{public class SqlServerHelper{private string connectionString;/// <summary>/// 数据库连接定义/// </summary>//private SqlConnection dbConnection;private SqlConnection dbConnection;/// <summary>/// SQL命令定义/// </summary>private SqlCommand dbCommand;/// <summary>/// 数据读取定义/// </summary>private SqlDataReader dataReader;/// <summary>/// 设置数据库连接字符串/// </summary>public string ConnectionString{set { connectionString = value; }}/// <summary>/// 构造函数/// </summary>/// <param name="connectionString">数据库连接字符串</param>public SqlServerHelper(string connectionString){this.connectionString = connectionString;}/// <summary>/// 执行一个查询,并返回结果集/// </summary>/// <param name="sql">要执行的查询SQL文本命令</param>/// <returns>返回查询结果集</returns>public DataTable ExecuteDataTable(string sql){return ExecuteDataTable(sql, CommandType.Text, null);}/// <summary>/// 执行一个查询,并返回查询结果/// </summary>/// <param name="sql">要执行的SQL语句</param>/// <param name="commandType">要执行的查询语句的类型,如存储过程或者SQL文本命令</param>/// <returns>返回查询结果集</returns>public DataTable ExecuteDataTable(string sql, CommandType commandType){return ExecuteDataTable(sql, commandType, null);}/// <summary>/// 执行一个查询,并返回查询结果/// </summary>/// <param name="sql">要执行的SQL语句</param>/// <param name="commandType">要执行的查询语句的类型,如存储过程或者SQL文本命令</param>/// <param name="parameters">Transact-SQL 语句或存储过程的参数数组</param>/// <returns></returns>public DataTable ExecuteDataTable(string sql, CommandType commandType, SqlParameter[] parameters){DataTable data = new DataTable();//实例化DataTable,用于装载查询结果集using (SqlConnection connection = new SqlConnection(connectionString)){using (SqlCommand command = new SqlCommand(sql, connection)){command.CommandType = commandType;//设置command的CommandType为指定的CommandType//如果同时传入了参数,则添加这些参数if (parameters != null){foreach (SqlParameter parameter in parameters){command.Parameters.Add(parameter);}}//通过包含查询SQL的SqlCommand实例来实例化SqlDataAdapterSqlDataAdapter adapter = new SqlDataAdapter(command);adapter.Fill(data);//填充DataTablecommand.Dispose();}connection.Dispose();}return data;}/// <summary>/// /// </summary>/// <param name="sql">要执行的查询SQL文本命令</param>/// <returns></returns>public SqlDataReader ExecuteReader(string sql){return ExecuteReader(sql, CommandType.Text, null);}/// <summary>/// /// </summary>/// <param name="sql">要执行的SQL语句</param>/// <param name="commandType">要执行的查询语句的类型,如存储过程或者SQL文本命令</param>/// <returns></returns>public SqlDataReader ExecuteReader(string sql, CommandType commandType){return ExecuteReader(sql, commandType, null);}/// <summary>/// /// </summary>/// <param name="sql">要执行的SQL语句</param>/// <param name="commandType">要执行的查询语句的类型,如存储过程或者SQL文本命令</param>/// <param name="parameters">Transact-SQL 语句或存储过程的参数数组</param>/// <returns></returns>public SqlDataReader ExecuteReader(string sql, CommandType commandType, SqlParameter[] parameters){SqlConnection connection = new SqlConnection(connectionString);SqlCommand command = new SqlCommand(sql, connection);//如果同时传入了参数,则添加这些参数if (parameters != null){foreach (SqlParameter parameter in parameters){command.Parameters.Add(parameter);}}//0表示永久,默认是30command.CommandTimeout = 240;connection.Open();//CommandBehavior.CloseConnection参数指示关闭Reader对象时关闭与其关联的Connection对象return command.ExecuteReader(CommandBehavior.CloseConnection);}/// <summary>/// /// </summary>/// <param name="sql">要执行的查询SQL文本命令</param>/// <returns></returns>public Object ExecuteScalar(string sql){return ExecuteScalar(sql, CommandType.Text, null);}/// <summary>/// /// </summary>/// <param name="sql">要执行的SQL语句</param>/// <param name="commandType">要执行的查询语句的类型,如存储过程或者SQL文本命令</param>/// <returns></returns>public Object ExecuteScalar(string sql, CommandType commandType){return ExecuteScalar(sql, commandType, null);}/// <summary>/// /// </summary>/// <param name="sql">要执行的SQL语句</param>/// <param name="commandType">要执行的查询语句的类型,如存储过程或者SQL文本命令</param>/// <param name="parameters">Transact-SQL 语句或存储过程的参数数组</param>/// <returns></returns>public Object ExecuteScalar(string sql, CommandType commandType, SqlParameter[] parameters){object result = null;using (SqlConnection connection = new SqlConnection(connectionString)){using (SqlCommand command = new SqlCommand(sql, connection)){command.CommandType = commandType;//设置command的CommandType为指定的CommandType//如果同时传入了参数,则添加这些参数if (parameters != null){foreach (SqlParameter parameter in parameters){command.Parameters.Add(parameter);}}//0表示永久,默认是30command.CommandTimeout = 240;connection.Open();//打开数据库连接result = command.ExecuteScalar();command.Dispose();}connection.Dispose();}return result;//返回查询结果的第一行第一列,忽略其它行和列}/// <summary>/// 对数据库执行增删改操作/// </summary>/// <param name="sql">要执行的查询SQL文本命令</param>/// <returns></returns>public int ExecuteNonQuery(string sql){return ExecuteNonQuery(sql, CommandType.Text, null);}/// <summary>/// 对数据库执行增删改操作/// </summary>/// <param name="sql">要执行的SQL语句</param>/// <param name="commandType">要执行的查询语句的类型,如存储过程或者SQL文本命令</param>/// <returns></returns>public int ExecuteNonQuery(string sql, CommandType commandType){return ExecuteNonQuery(sql, commandType, null);}/// <summary>/// 对数据库执行增删改操作/// </summary>/// <param name="sql">要执行的SQL语句</param>/// <param name="commandType">要执行的查询语句的类型,如存储过程或者SQL文本命令</param>/// <param name="parameters">Transact-SQL 语句或存储过程的参数数组</param>/// <returns></returns>public int ExecuteNonQuery(string sql, CommandType commandType, SqlParameter[] parameters){int count = 0;using (SqlConnection connection = new SqlConnection(connectionString)){using (SqlCommand command = new SqlCommand(sql, connection)){command.CommandType = commandType;//设置command的CommandType为指定的CommandType//如果同时传入了参数,则添加这些参数if (parameters != null){foreach (SqlParameter parameter in parameters){command.Parameters.Add(parameter);}}//0表示永久,默认是30command.CommandTimeout = 240;connection.Open();//打开数据库连接count = command.ExecuteNonQuery();command.Dispose();}connection.Dispose();}return count;//返回执行增删改操作之后,数据库中受影响的行数}/// <summary>/// 返回当前连接的数据库中所有由用户创建的数据库/// </summary>/// <returns></returns>public DataTable GetTables(){DataTable data = null;using (SqlConnection connection = new SqlConnection(connectionString)){connection.Open();//打开数据库连接data = connection.GetSchema("Tables");connection.Dispose();}return data;}/// <summary>/// 执行多条SQL语句,实现数据库事务。/// </summary>/// <param name="SQLStringList">多条SQL语句</param>        public int ExecuteSqlTran(List<String> SQLStringList){using (SqlConnection connection = new SqlConnection(connectionString)){connection.Open();SqlCommand cmd = new SqlCommand();cmd.Connection = connection;SqlTransaction tx = connection.BeginTransaction();cmd.Transaction = tx;try{int count = 0;for (int n = 0; n < SQLStringList.Count; n++){string strsql = SQLStringList[n];if (strsql.Trim().Length > 1){cmd.CommandText = strsql;count += cmd.ExecuteNonQuery();}}tx.Commit();return count;}catch{tx.Rollback();return 0;}finally{cmd.Dispose();connection.Dispose();}}}/// <summary>/// 执行带一个存储过程参数的的SQL语句。/// </summary>/// <param name="SQLString">SQL语句</param>/// <param name="content">参数内容,比如一个字段是格式复杂的文章,有特殊符号,可以通过这个方式添加</param>/// <returns>影响的记录数</returns>public int ExecuteSql(string SQLString, string content){using (SqlConnection connection = new SqlConnection(connectionString)){SqlCommand cmd = new SqlCommand(SQLString, connection);SqlParameter myParameter = new SqlParameter("@content", SqlDbType.NText);myParameter.Value = content;cmd.Parameters.Add(myParameter);try{connection.Open();int rows = cmd.ExecuteNonQuery();return rows;}catch (Exception e){throw e;}finally{cmd.Dispose();connection.Dispose();}}}/// <summary>/// 执行带一个存储过程参数的的SQL语句。/// </summary>/// <param name="SQLString">SQL语句</param>/// <param name="content">参数内容,比如一个字段是格式复杂的文章,有特殊符号,可以通过这个方式添加</param>/// <returns>影响的记录数</returns>public object ExecuteSqlGet(string SQLString, string content){using (SqlConnection connection = new SqlConnection(connectionString)){SqlCommand cmd = new SqlCommand(SQLString, connection);SqlParameter myParameter = new SqlParameter("@content", SqlDbType.NText);myParameter.Value = content;cmd.Parameters.Add(myParameter);try{connection.Open();object obj = cmd.ExecuteScalar();if ((Object.Equals(obj, null)) || (Object.Equals(obj, System.DBNull.Value))){return null;}else{return obj;}}catch (Exception e){throw e;}finally{cmd.Dispose();connection.Dispose();}}}/// <summary>/// 向数据库里插入图像格式的字段(和上面情况类似的另一种实例)/// </summary>/// <param name="strSQL">SQL语句</param>/// <param name="fs">图像字节,数据库的字段类型为image的情况</param>/// <returns>影响的记录数</returns>public int ExecuteSqlInsertImg(string strSQL, byte[] fs){using (SqlConnection connection = new SqlConnection(connectionString)){SqlCommand cmd = new SqlCommand(strSQL, connection);SqlParameter myParameter = new SqlParameter("@fs", SqlDbType.Image);myParameter.Value = fs;cmd.Parameters.Add(myParameter);try{connection.Open();int rows = cmd.ExecuteNonQuery();return rows;}catch (Exception e){throw e;}finally{cmd.Dispose();connection.Dispose();}}}/// <summary>/// 执行查询语句,返回DataSet/// </summary>/// <param name="SQLString">查询语句</param>/// <returns>DataSet</returns>public DataSet Query(string SQLString){using (SqlConnection connection = new SqlConnection(connectionString)){DataSet ds = new DataSet();try{connection.Open();SqlDataAdapter command = new SqlDataAdapter(SQLString, connection);command.Fill(ds, "ds");command.Dispose();}catch (Exception ex){throw new Exception(ex.Message);}finally{connection.Dispose();}return ds;}}}//Class_end
}
using System;
using System.Collections.Generic;
using System.Data;
using System.Data.SqlClient;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern
{/// <summary>/// 用户管理对象/// </summary>internal class UserManager{public static List<UserModel> GetUserByDepmentId(string depmentId){string conStr = $"server=.;database=Test;uid=test;pwd=123456";SqlServerHelper sqlServerHelper = new SqlServerHelper(conStr);string sql = $"select uo.* from UserInfo uo left join Depment dt on uo.depmentId=dt.ID where dt.ID like '{depmentId}%'";DataTable dt = sqlServerHelper.ExecuteDataTable(sql);使用参数内容时周围不能有特殊修符号如单引号等//SqlParameter[] sqlParameters ={//    new SqlParameter("@dd",SqlDbType.VarChar){Value=depmentId}//}//;//DataTable dt = sqlServerHelper.ExecuteDataTable(sql, System.Data.CommandType.Text, sqlParameters);if (dt.Rows.Count>0){List<UserModel> userModels = new List<UserModel>(); foreach (DataRow dr in dt.Rows){UserModel userModel = new UserModel();userModel.Id = dr["ID"].ToString();userModel.UserId = dr["UserID"].ToString();userModel.UserName = dr["UserName"].ToString();userModel.UserSex = dr["UserSex"].ToString();userModel.UserAge = dr["UserAge"].ToString();userModel.UserTelnumber = dr["UserTelnumber"].ToString();userModel.UserHeight = dr["UserHeight"].ToString();userModel.UserWeight = dr["UserWeight"].ToString();userModel.UserAddress = dr["UserAddress"].ToString();userModel.DepmentId = dr["DepmentId"].ToString();userModels.Add(userModel);}return userModels;}return null;}}//Class_end
}

3、编写客户端示例

namespace ProxyPattern
{internal class Program{static void Main(string[] args){GetDepmentUsersTest();Console.ReadLine();}/// <summary>/// 测试获取部门人员/// </summary>private static void GetDepmentUsersTest(){Console.WriteLine("---测试获取部门人员---");string depmentId = "0101";Console.WriteLine($"{depmentId} 部门的所有人员如下");List<UserModel> userModels = UserManager.GetUserByDepmentId(depmentId);foreach (UserModel userModel in userModels){Console.WriteLine(userModel.ToString());}depmentId = "0102";Console.WriteLine($"\n{depmentId} 部门的所有人员如下");List<UserModel> userModels2 = UserManager.GetUserByDepmentId(depmentId);foreach (UserModel userModel in userModels2){Console.WriteLine(userModel.ToString());}}}//Class_end
}

4、运行结果

如上所示我们已经不使用任何模式实现了获取指定部门的所有用户信息;但是当我们一次性访问的数据条数很多,且每条数据量很大的情况下,那么会十分消耗我们宝贵的内存空间;并且从客户使用的角度来说,查看用户信息具有很大的随机性,客户有可能访问每一条数据,也有可能一条都不访问;也就是说,一次性访问很多条数据,消耗了大量内存,但是很可能是浪费的,因为客户根本不会查看这么多数据;对于每条数据用户只查看姓名而已【那么,我们该怎么实现,才能既把多条用户数据的姓名展示出来,而又能节省内存空间?只有当用户想要查看指定用户更多数据的时候才显示对应用户的详细数据?】

  2.2、使用代理模式的示例

        使用代理模式就能解决如上的问题:代理模式解决的思路是:①由于客户开始访问的时候只查看用户姓名,,因此开始的时候就只从数据库中查询返回所有的用户编号与姓名数据;②当客户要查看某个用户相信信息的时候,再根据该用户编号从数据库中查询到该用户的详细数据信息;这样一来再满足用户需求的前提下,还减少了对内存的消耗,只是每次需要重新查询一下数据库【可以看做是以时间换空间的做法】。

1、定义用户数据对象接口

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern.Proxy
{/// <summary>/// 用户数据对象接口/// </summary>internal interface IUserModel{//用户编号string? Id { get; set; }//用户身份证编号public string? UserId { get; set; }//用户名称public string? UserName { get; set; }//用户性别public string? UserSex { get; set; }//用户年龄public string? UserAge { get; set; }//联系电话public string? UserTelnumber { get; set; }//用户身高public string? UserHeight { get; set; }//用户体重public string? UserWeight { get; set; }//用户地址public string? UserAddress { get; set; }//用户所属部门编号public string? DepmentId { get; set; }}//Interface_end
}

2、用户对象模型

using System;
using System.Collections.Generic;
using System.Linq;
using System.Security;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern.Proxy
{/// <summary>/// 用户对象/// </summary>internal class UserModel{//用户编号public string? Id { get; set; }//用户身份证编号public string? UserId { get; set; }//用户名称public string? UserName { get; set; }//用户性别public string? UserSex { get; set; }//用户年龄public string? UserAge { get; set; }//联系电话public string? UserTelnumber { get; set; }//用户身高public string? UserHeight { get; set; }//用户体重public string? UserWeight { get; set; }//用户地址public string? UserAddress { get; set; }//用户所属部门编号public string? DepmentId { get; set; }}//Class_end
}

3、创建代理对象继承接口并实现

using System;
using System.Collections.Generic;
using System.Data;
using System.Data.SqlClient;
using System.Linq;
using System.Reflection.Metadata.Ecma335;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern.Proxy
{internal class Proxy : IUserModel{//持有被代理的具体目标对象private UserModel userModel;//是否已经重新装载过数据标识private bool loaded = false;/// <summary>/// 构造函数/// </summary>/// <param name="userModel">被代理的具体目标对象</param>public Proxy(UserModel userModel){this.userModel = userModel;}public string? Id { get=>this.userModel.Id ; set =>this.userModel.Id=value; }public string? UserId { get => this.userModel.UserId; set => this.userModel.UserId=value; }public string? UserName { get => this.userModel.UserName; set => this.userModel.UserName=value; }public string? UserSex { get => this.userModel.UserSex; set => this.userModel.UserSex=value; }public string? UserAge { get => this.userModel.UserAge; set => this.userModel.UserAge=value; }public string? UserTelnumber { get => this.userModel.UserTelnumber; set => this.userModel.UserTelnumber=value; }public string? UserHeight { get => this.userModel.UserHeight; set => this.userModel.UserHeight=value; }public string? UserWeight { get => this.userModel.UserWeight; set => this.userModel.UserWeight=value; }public string? UserAddress { get => this.userModel.UserAddress; set => this.userModel.UserAddress=value; }public string? DepmentId {get {if (!this.loaded){//从数据库重写加载数据Reload();//重新设置加载标识为truethis.loaded = true;}return this.userModel.DepmentId;}set {this.userModel.DepmentId = value;}     }private void Reload(){Console.WriteLine($"重新查询数据库获取完整的用户数据,UserId={userModel.UserId}");string conStr = $"server=.;database=Test;uid=test;pwd=123456";SqlServerHelper sqlServerHelper = new SqlServerHelper(conStr);string sql = "select ID,UserAge,UserSex,UserTelnumber,UserHeight,UserWeight,UserAddress,DepmentId from UserInfo where UserID=@userId";//使用参数内容时周围不能有特殊修符号如单引号等SqlParameter[] sqlParameters ={new SqlParameter("@userId",SqlDbType.VarChar){Value=userModel.UserId}};DataTable dt = sqlServerHelper.ExecuteDataTable(sql, System.Data.CommandType.Text, sqlParameters);if (dt.Rows.Count > 0){foreach (DataRow dr in dt.Rows){userModel.Id = dr["ID"].ToString();userModel.UserSex = dr["UserSex"].ToString();userModel.UserAge = dr["UserAge"].ToString();userModel.UserTelnumber = dr["UserTelnumber"].ToString();userModel.UserHeight = dr["UserHeight"].ToString();userModel.UserWeight = dr["UserWeight"].ToString();userModel.UserAddress = dr["UserAddress"].ToString();userModel.DepmentId = dr["DepmentId"].ToString();}}}public override string ToString(){string str = $"完整用户信息——用户的身份证编号【{UserId}】姓名【{UserName}】性别【{UserSex}】年龄【{UserAge}】" +$"联系电话【{UserTelnumber}】身高【{UserHeight}】体重【{UserWeight}】地址【{UserAddress}】所属部门编号【{DepmentId}】";return str;}}//Class_end
}

4、创建用户管理对象

《1》此时的用户管理对象查询的时候不需要全部获取数据,只需要查询用户编号与姓名就可以了;

《2》从数据库获取到用户编号与姓名数据后也只用将这两个内容转换为用户对象的属性赋值,其他内容不用设置。

using System;
using System.Collections.Generic;
using System.Data;
using System.Data.SqlClient;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern.Proxy
{/// <summary>/// 用户管理对象/// </summary>internal class UserManager{/// <summary>/// 根据部门编号来获取该部门下的所有人员/// </summary>/// <param name="depmentId">部门编号</param>/// <returns>返回该部门下的所有人员</returns>public List<IUserModel> GetUserByDepmentId(string depmentId){string conStr = $"server=.;database=Test;uid=test;pwd=123456";SqlServerHelper sqlServerHelper = new SqlServerHelper(conStr);//只需要查询UserId与UserName两个值就可以了string sql = $"select uo.UserId,uo.UserName from UserInfo uo left join Depment dt on uo.depmentId=dt.ID where dt.ID like '{depmentId}%'";DataTable dt = sqlServerHelper.ExecuteDataTable(sql);if (dt.Rows.Count>0){List<IUserModel> userModels = new List<IUserModel>();foreach (DataRow dr in dt.Rows){//这里是只创建代理对象,而不是直接创建UserModel对象Proxy proxy = new Proxy(new UserModel());proxy.UserId = dr["UserID"].ToString();proxy.UserName = dr["UserName"].ToString();userModels.Add(proxy);}return userModels;}return null;}}//Class_end
}

5、创建客户端测试

using ProxyPattern.Proxy;namespace ProxyPattern
{internal class Program{static void Main(string[] args){GetUserInfoProxyTest();Console.ReadLine();}/// <summary>/// 测试用户代理客户端/// </summary>private static void GetUserInfoProxyTest(){Proxy.UserManager userManager = new Proxy.UserManager();List<IUserModel> userModels = userManager.GetUserByDepmentId("0101");//若只是显示用户名称,则不需要重新查询数据库foreach (var item in userModels){string str = $"用户的身份证编号是【{item.UserId}】姓名是【{item.UserName}】";Console.WriteLine(str);}Console.WriteLine();//若要访问非用户身份证编号和姓名外的属性内容,那就需要重新查询数据库foreach (var item in userModels){string str = $"用户的身份证编号是【{item.UserId}】姓名是【{item.UserName}】部门是【{item.DepmentId}】";Console.WriteLine(str);Console.WriteLine(item.ToString());}}}//Class_end
}

6、运行结果

        如上的代理模式实现了开始获取所有的用户编号与用户姓名数据;只有当访问到这两个数据外的数据时才需要重新查询数据获得完整用户数据信息。但如上示例也存在一个问题,就是如果客户对每条数据都要求查看详细数据的话,那么总的查询数据库的次数会达到【1+N】次;这种代理模式最合适的场景是:大多数情况下只查看用户编号与姓名数据,只有少量情况查看个人详情数据。

 2.3、保护代理

保护代理是一种控制对原始对象访问的代理,多用于对象应该有不同的访问权限的情况。

        业务需求:现有一个订单系统业务,要求一旦订单被创建,只有订单的创建人才可以修改订单数据,其他人则不能修改。

1、创建订单接口规范行为

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern.ProtectProxy
{/// <summary>/// 订单对象接口/// </summary>internal interface IOrder{//获取订单订购的产品名称string GetProductName();//设置订单订购的产品名称与人员void SetProductName(string productName,string user);//获取订购订单的数量int GetOrderNumber();//设置订购订单的数量与人员void SetOrderNumber(int orderNumber,string user);//获取创建订单的人员string GetOrderUser();//设置创建订单的人员与人员void SetOrderUser(string orderUser, string user);}//Interface_end
}

2、创建订单对象

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern.ProtectProxy
{/// <summary>/// 订单对象/// </summary>internal class Order : IOrder{//订单订购的产品名称private string productName=string.Empty;//订单的订购数量private int orderNumber=0;//创建订单的人员private string orderUser=string.Empty;/// <summary>/// 构造函数/// </summary>/// <param name="productName">订单订购的产品名称</param>/// <param name="orderNumber">订单数量</param>/// <param name="orderUserName">创建订单的人员</param>public Order(string productName,int orderNumber,string orderUser){this.productName = productName;this.orderNumber = orderNumber;this.orderUser=orderUser;}public int GetOrderNumber(){return this.orderNumber;}public string GetOrderUser(){return this.orderUser;}public string GetProductName(){return this.productName;}public void SetOrderNumber(int orderNumber, string user){this.orderNumber=orderNumber;}public void SetOrderUser(string orderUser, string user){this.orderUser=orderUser;}public void SetProductName(string productName, string user){this.productName=productName;}}//Class_end
}

3、创建订单对象的代理并继承接口实现具体功能

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern.ProtectProxy
{/// <summary>/// 订单代理对象/// </summary>internal class OrderProxy : IOrder{//持有被代理的具体目标对象private Order order = null;/// <summary>/// 构造函数/// </summary>/// <param name="realSubject">被代理的具体目标对象</param>public OrderProxy(Order realSubject){this.order = realSubject;   }public int GetOrderNumber(){return this.order.GetOrderNumber();}public string GetOrderUser(){return this.order.GetOrderUser();}public string GetProductName(){return this.order.GetProductName();}public void SetOrderNumber(int orderNumber, string user){//控制访问权限,只有创建订单的人员才能够修改if (!string.IsNullOrEmpty(user) && user.Equals(this.GetOrderUser())){order.SetOrderNumber(orderNumber, user);}else{Console.WriteLine($"抱歉【{user}】,您无权修改订单中的产品数量");}}public void SetOrderUser(string orderUser, string user){//控制访问权限,只有创建订单的人员才能够修改if (!string.IsNullOrEmpty(user) && user.Equals(this.GetOrderUser())){order.SetOrderUser(orderUser, user);}else{Console.WriteLine($"抱歉【{user}】,您无权修改订单中的创建人员");}}public void SetProductName(string productName, string user){//控制访问权限,只有创建订单的人员才能够修改if (!string.IsNullOrEmpty(user) && user.Equals(this.GetOrderUser())){order.SetProductName(productName, user);}else{Console.WriteLine($"抱歉【{user}】,您无权修改订单中的产品名称");}}public override string ToString(){string str = $"订单订购的产品名称是【{GetProductName()}】数量是【{GetOrderNumber()}】创建订单的人员是【{GetOrderUser()}】";return str;}}//Class_end
}

4、客户端测试

using ProxyPattern.ProtectProxy;
using ProxyPattern.Proxy;namespace ProxyPattern
{internal class Program{static void Main(string[] args){OrderProxyTest();Console.ReadLine();}/// <summary>/// 订单代理测试/// </summary>private static void OrderProxyTest(){//张三先登录系统创建一个订单IOrder order = new OrderProxy(new Order("设计模式",666,"张三"));Console.WriteLine($"初始创建订单的信息是\n{order.ToString()}\n\n");//李四想要修改,此时应该有报错提示order.SetOrderNumber(999,"李四");Console.WriteLine($"李四尝试修改订单数量后,订单信息是\n{order.ToString()}");//创建者张三修改订单,可正常修改且不会报错order.SetOrderNumber(888,"张三");Console.WriteLine($"\n\n张三修改订单数量后,订单信息是\n{order.ToString()}");}}//Class_end
}

5、运行结果

对于代理模式某些情况下还可以使用继承的方式取代掉接口:

1、创建订单对象并将需要设置权限控制的方法设置为虚方法

using ProxyPattern.ProtectProxy;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern.ModifyProxy
{/// <summary>/// 订单对象/// </summary>internal class Order : IOrder{//订单订购的产品名称private string productName=string.Empty;//订单的订购数量private int orderNumber=0;//创建订单的人员private string orderUser=string.Empty;/// <summary>/// 构造函数/// </summary>/// <param name="productName">订单订购的产品名称</param>/// <param name="orderNumber">订单数量</param>/// <param name="orderUserName">创建订单的人员</param>public Order(string productName,int orderNumber,string orderUser){this.productName = productName;this.orderNumber = orderNumber;this.orderUser=orderUser;}public int GetOrderNumber(){return orderNumber;}public string GetOrderUser(){return orderUser;}public string GetProductName(){return productName;}public virtual void SetOrderNumber(int orderNumber, string user){this.orderNumber=orderNumber;}public virtual void SetOrderUser(string orderUser, string user){this.orderUser=orderUser;}public virtual void SetProductName(string productName, string user){this.productName=productName;}}//Class_end
}

2、创建一个订单代理对象继承该订单对象重新虚方法

using ProxyPattern.ProtectProxy;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace ProxyPattern.ModifyProxy
{/// <summary>/// 订单代理对象/// </summary>internal class OrderProxy : Order{public static OrderProxy instance;//单例public OrderProxy(string productName, int orderNumber, string orderUser) : base(productName, orderNumber, orderUser){if (instance == null){instance = this;}else{Console.WriteLine($"{GetType()}/OrderProxy()函数不允许重复实例化!!!");}}public override void SetOrderNumber(int orderNumber, string user){//控制访问权限,只有创建订单的人员才能够修改if (!string.IsNullOrEmpty(user) && user.Equals(this.GetOrderUser())){base.SetOrderNumber(orderNumber, user);}else{Console.WriteLine($"抱歉【{user}】,您无权修改订单中的产品数量");}}public override void SetOrderUser(string orderUser, string user){//控制访问权限,只有创建订单的人员才能够修改if (!string.IsNullOrEmpty(user) && user.Equals(this.GetOrderUser())){base.SetOrderUser(orderUser, user);}else{Console.WriteLine($"抱歉【{user}】,您无权修改订单中的创建人员");}}public override void SetProductName(string productName, string user){//控制访问权限,只有创建订单的人员才能够修改if (!string.IsNullOrEmpty(user) && user.Equals(this.GetOrderUser())){base.SetProductName(productName, user);}else{Console.WriteLine($"抱歉【{user}】,您无权修改订单中的产品名称");}}public override string ToString(){string str = $"订单订购的产品名称是【{GetProductName()}】数量是【{GetOrderNumber()}】创建订单的人员是【{GetOrderUser()}】";return str;}}//Class_end
}

 3、客户端测试

using ProxyPattern.ProtectProxy;
using ProxyPattern.Proxy;namespace ProxyPattern
{internal class Program{static void Main(string[] args){OrderProxyTest2();Console.ReadLine();}/// <summary>/// 订单代理测试【继承方式】/// </summary>private static void OrderProxyTest2(){//张三先登录系统创建一个订单ModifyProxy.Order order = new ModifyProxy.OrderProxy("设计模式", 666, "张三");Console.WriteLine($"初始创建订单的信息是\n{order.ToString()}\n\n");//李四想要修改,此时应该有报错提示order.SetOrderNumber(999, "李四");Console.WriteLine($"李四尝试修改订单数量后,订单信息是\n{order.ToString()}");//创建者张三修改订单,可正常修改且不会报错order.SetOrderNumber(888, "张三");Console.WriteLine($"\n\n张三修改订单数量后,订单信息是\n{order.ToString()}");}}//Class_end
}

 4、运行结果

三、项目源码工程

kafeiweimei/Learning_DesignPattern: 这是一个关于C#语言编写的基础设计模式项目工程,方便学习理解常见的26种设计模式https://github.com/kafeiweimei/Learning_DesignPattern

相关文章:

学习设计模式《十》——代理模式

一、基础概念 代理模式的本质【控制对象访问】&#xff1b; 代理模式的定义&#xff1a;为其他对象提供一种代理以控制对这个对象的访问&#xff1b; 代理模式的功能&#xff1a;代理模式是通过创建一个代理对象&#xff0c;用这个代理对象去代表真实的对象&#xff1b;客户端得…...

RestFul操作ElasticSearch:索引与文档全攻略

RestFul方式操作ES 索引库操作 创建索引库 PUT /索引库名称 {"mappings":{"properties":{"字段名":{"type":"字段类型","analyzer":"分词器","index":"是否参与搜索(布尔值)"},…...

OpenCV 图像读取与显示

一、知识点: 1、读取图像 (1)、Mat imread( const String & filename, int flags IMREAD_COLOR_BGR ); (2)、返回值: Mat&#xff0c;返回读取的图像。 若读取图像失败&#xff0c;则返回一个空的对象&#xff0c;对象.empty()为true。 (3)、参数filename: String是…...

Django快速入门篇

Django官网 https://docs.djangoproject.com/zh-hans/4.2/ 官方介绍 官方版本 推荐LTS版本&#xff0c;python3.9/3.10 djongo 每两年会出一个LTS版本 关于环节djongo&#xff0c;conda直接安装即可 conda create -n myenv python3.9 conda activate myenv pip install dj…...

C++23 新增扁平化关联容器详解

文章目录 一、引言已有关联容器回顾新容器的引入原因 二、std::flat_set定义与特性代码示例适用场景 三、std::flat_multiset定义与特性代码示例适用场景 四、std::flat_map定义与特性代码示例适用场景 五、std::flat_multimap定义与特性代码示例适用场景 六、与其他容器的比较…...

当PLC遇上电焊机器人:EtherCAT转CANopen上演工业级“语言翻译官”

在汽车自动化产线中&#xff0c;PLC与电焊机器人的高效协同是提升生产效率的关键。但PLC常用的EtherCAT协议与电焊机器人采用的CANopen协议存在通信壁垒&#xff0c;JH-ECT009疆鸿智能EtherCAT转CANopen技术成为打破这一障碍的核心方案。 应用拓扑图 EtherCAT是高速工业以太网协…...

LeetCode 1345. 跳跃游戏 IV(困难)

题目描述 给你一个整数数组 arr &#xff0c;你一开始在数组的第一个元素处&#xff08;下标为 0&#xff09;。 每一步&#xff0c;你可以从下标 i 跳到下标 i 1 、i - 1 或者 j &#xff1a; i 1 需满足&#xff1a;i 1 < arr.lengthi - 1 需满足&#xff1a;i - 1 …...

Linux bash shell的循环命令for、while和until

1、for命令 for命令&#xff0c;允许你创建一个遍历一系列值的循环&#xff0c;每次迭代都使用其中一个 值来执行已定义好的一组命令。 for var in list do commands done # 在list参数中&#xff0c;你需要提供迭代中要用到的一系列值。 # 可以通过几种不同的方法指定列表中的…...

三、【数据建模篇】:用 Django Models 构建测试平台核心数据

【数据建模篇】&#xff1a;用 Django Models 构建测试平台核心数据 前言我们要设计哪些核心数据&#xff1f;准备工作&#xff1a;创建 Django App开始设计数据模型 (Models)1. 通用基础模型 (可选但推荐)2. 项目模型 (Project)3. 模块模型 (Module)4. 测试用例模型 (TestCase…...

Mac如何允许安装任何来源软件?

打开系统偏好设置-安全性与隐私&#xff0c;点击右下角的解锁按钮&#xff0c;选择允许从任何来源。 如果没有这一选项&#xff0c;请到打开终端&#xff0c;输入命令行&#xff1a;sudo spctl --master-disable, 输入命令后回车&#xff0c;输入电脑的开机密码后回车。 返回“…...

云原生主要架构模式

云原生(Cloud Native)是一种利用云计算的优势来构建和运行可扩展、弹性和高效应用程序的方法。它不仅仅是技术的集合,更是一种架构和设计理念。本文将围绕你提出的几部分,深入探讨云原生主要的架构模式,帮助你理解如何利用这些模式构建现代化的应用。 1. 服务化架构模式(…...

Neon数据库:让Postgres更智能的选择!

Neon&#xff1a;革新的Serverless PostgreSQL解决方案 在当今快速发展的技术世界&#xff0c;数据库的效率和灵活性成为众多开发者关注的重中之重。Neon&#xff0c;以其独特的serverless架构&#xff0c;正引领着这一变革。本文将深入探讨Neon的独特构架、应用场景以及具体的…...

《Metasploit框架核心模块解析与安全防护实践》​

目录 ​​一、框架模块化设计与安全验证价值​​ ​​1. 漏洞验证模块&#xff08;Exploit Modules&#xff09;​​ ​​2. 安全评估模块&#xff08;Auxiliary Modules&#xff09;​​ ​​3. 安全响应模块&#xff08;Post-Exploitation&#xff09;​​ ​​4. 载荷安全…...

C#:多线程Task使用

一.Task与Thread Task是架构在Thread之上的&#xff0c;也就是说任务最终还是要抛给线程去执行。Task跟Thread不是一对一的关系&#xff0c;比如开10个任务并不是说会开10个线程&#xff0c;这一点任务有点类似线程池&#xff0c;但是任务相比线程池有很小的开销和精确的控制。…...

Nginx笔记

一、概述 Nginx一个具有高性能的【HTTP】和【反向代理】的【WEB服务器】&#xff0c;同时也是一个电子邮件代理服务器。正向代理服务的是客户端&#xff08;比如VPN&#xff09;&#xff0c;反向代理服务的是服务端。Nginx是多进程的&#xff0c;有一个Master进程控制多个Worke…...

小米便签源码部署流程

一、准备环境 1. 安装必要工具 Android Studio&#xff1a;最新稳定版&#xff08;需支持 Kotlin 和 Jetpack Compose&#xff09;。 JDK&#xff1a;建议 JDK 11 或更高&#xff08;通过 sdkman 或 brew 安装&#xff09;。 Git&#xff1a;用于克隆源码。 2. 配置国内镜像源&…...

DAY 30 超大力王爱学Python

知识点回顾&#xff1a; 导入官方库的三种手段导入自定义库/模块的方式导入库/模块的核心逻辑&#xff1a;找到根目录&#xff08;python解释器的目录和终端的目录不一致&#xff09; 作业&#xff1a;自己新建几个不同路径文件尝试下如何导入 步骤 1&#xff1a;创建项目结构 …...

左右边界策略

这是一套完整的交易逻辑策略,涵盖了从函数定义、指标计算、信号生成到资金和仓位管理、加仓和减仓逻辑、以及止损和止盈逻辑的各个方面。 以下对该交易系统进行详细分析: 交易逻辑思路 1. 函数定义 - DZSell 和 DZBuy 函数:这两个函数用于计算卖出和买入的价格区间。它…...

iOS苹果和Android安卓测试APP应用程序的区别差异

在当今这个移动互联网时代&#xff0c;iOS和Android作为两大主流操作系统&#xff0c;它们在测试应用程序时存在哪些差异呢&#xff1f;这不仅是一个技术问题&#xff0c;也是一个市场策略问题。让我们从一个实际案例开始探讨。 假设我们有一个新的社交应用需要在iOS和Android…...

【Python装饰器深潜】从语法糖到元编程的艺术

目录 🌟 前言🏗️ 技术背景与价值🩹 当前技术痛点🛠️ 解决方案概述👥 目标读者说明🧠 一、技术原理剖析📊 核心概念图解💡 核心作用讲解🔧 关键技术模块说明⚖️ 技术选型对比🛠️ 二、实战演示⚙️ 环境配置要求💻 核心代码实现案例1:基础计时装饰器案…...

Kubernetes中微服务JVM监控与自动发现的解决方案

以下是针对 Kubernetes 中微服务 JVM 监控与自动发现的解决方案,结合 Prometheus 的动态发现机制和 Spring Boot 的监控能力,解决 Pod IP 动态变化和当前微服务监控数据暴露匿名随意访问的安全问题。 一、微服务端配置(Spring Boot 微服务) 1. 依赖配置(pom.xml) <…...

mapbox进阶,纯前端geojson转shape,并将shape相关文件压缩成zip压缩包并下载

👨‍⚕️ 主页: gis分享者 👨‍⚕️ 感谢各位大佬 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍⚕️ 收录于专栏:mapbox 从入门到精通 文章目录 一、🍀前言1.1 ☘️mapboxgl.Map 地图对象1.2 ☘️mapboxgl.Map style属性1.3 ☘️line线图层样式二、🍀纯前端geojson转…...

Oracle ASM Rebalance Power 了解

Oracle ASM Rebalance Power 了解 基本概念 ASM (Automatic Storage Management) 的 Rebalance Power 参数控制磁盘组重新平衡操作的速度和资源使用。当添加、删除或调整 ASM 磁盘时&#xff0c;ASM 会自动执行重新平衡操作来重新分布数据。 ASM_POWER_LIMIT 参数 作用 控…...

sqlite的拼接字段的方法(sqlite没有convert函数)

我在sqlserver 操作方式&#xff1a; /// <summary>///获取当前门店工资列表/// </summary>/// <param name"wheres">其他条件</param>/// <param name"ThisMendian">当前门店</param>/// <param name"IsNotU…...

深入解析 OpenManus:开源 AI 智能体框架的技术原理与实践

深入解析 OpenManus&#xff1a;开源 AI 智能体框架的技术原理与本地部署指南 在当今人工智能快速发展的时代&#xff0c;智能体&#xff08;Agent&#xff09;技术正逐渐成为推动自动化和智能化的关键力量。OpenManus&#xff0c;由 MetaGPT 团队开发的开源 AI 智能体框架&am…...

[面试精选] 0001. 两数之和

文章目录 1. 题目链接2. 题目描述3. 题目示例4. 解题思路5. 题解代码6. 复杂度分析 1. 题目链接 1. 两数之和 - 力扣&#xff08;LeetCode&#xff09; 2. 题目描述 给定一个整数数组 nums 和一个整数目标值 target&#xff0c;请你在该数组中找出 和为目标值 target 的那 两个…...

CAP分布式理论

CAP分布式理论 文章目录 CAP分布式理论事务本地事务定义适用场景优点缺点 分布式事务一、分布式事务的定义二、分布式事务的标准与 CAP 理论的关系1. CAP 理论的核心内容2. CAP 理论对分布式事务的指导意义3. 分布式事务的其他关键标 三、总结&#xff1a;CAP 理论与分布式事务…...

如何管理和优化内核参数

Linux 系统中的内核参数&#xff08;Kernel Parameters&#xff09;对系统的性能、安全性和稳定性有很大影响。通过调整内核参数&#xff0c;可以优化系统性能、增强网络能力、调整内存管理等。管理和优化内核参数通常涉及以下几个步骤&#xff1a; 1. 查看当前内核参数 在 L…...

sgRNA的靶基因基因组如何获得? for 下游的 T7E1验证

愿武艺晴小朋友一定得每天都开心&#xff01; 1&#xff09;在基因组&#xff0c;靶标区域上下游&#xff0c;设计引物&#xff08;以Zfp532基因为例&#xff09;&#xff1a; a. NCBI&#xff08;Home - Gene - NCBI&#xff09;Gene 页面上输入&#xff1a;Zfp532。 b. 在新…...

人工智能+:职业价值的重构与技能升级

当“人工智能”成为产业升级的标配时&#xff0c;一个令人振奋的就业图景正在展开——不是简单的岗位替代&#xff0c;而是职业价值的重新定义。这场变革的核心在于&#xff0c;AI并非抢走工作机会&#xff0c;而是创造了人类与技术协作的全新工作范式。理解这一范式转换的逻辑…...

【前端开发】Uniapp日期时间选择器:实现分钟动态步长设置

技术栈 Uniapp Vue3 uView年份显示前后一年&#xff0c;分钟动态设置间隔 效果图 主体显示 <view class"uni-row selector" click"openPicker"><uni-icons color"#c0c4cc" type"calendar" size"22"></uni-…...

目标检测DN-DETR(2022)详细解读

文章目录 gt labels 和gt boxes加噪query的构造attention maskIS&#xff08;InStability&#xff09;指标 在DAB-Detr的基础上&#xff0c;进一步分析了Detr收敛速度慢的原因&#xff1a;二分图匹配的不稳定性&#xff08;也就是说它的目标在频繁地切换&#xff0c;特别是在训…...

榕壹云上门家政系统:基于Spring Boot+MySQL+UniApp的全能解决方案

一、项目背景&#xff1a;家政服务行业的数字化升级需求 随着城市化进程加快&#xff0c;居民对上门家政服务的需求持续增长。传统家政行业存在服务效率低、信息不透明、管理成本高的问题&#xff0c;亟需通过数字化工具实现服务标准化、流程自动化。榕壹云上门家政系统应运而…...

目标检测DINO-DETR(2023)详细解读

文章目录 对比去噪训练混合查询选择look forward twice 论文全称为&#xff1a;DETR with Improved DeNoising Anchor Boxes for End-to-End Object Detection 提出了三个新的方法&#xff1a; 首先&#xff0c;为了改进一对一的匹配效果&#xff0c;提出了一种对比去噪训练方法…...

LLM笔记(十)vLLM(1)PagedAttention论文笔记

文章目录 PagedAttention论文笔记论文摘要概览1. 引言问题背景提出的解决方案主要贡献和成果 2. 背景2.1 基于Transformer的大语言模型2.2 LLM服务与自回归生成2.3 LLM的批处理技术 3. LLM服务中的内存挑战3.1 现有系统中的内存管理 4. 方法: PagedAttention 和 vLLMvLLM 系统概…...

5.20 note

不用if else实现递归 原代码 class Solution { public: int mechanicalAccumulator(int target) { if (target 0) return 0; // 终止条件&#xff1a;基准情况 return target mechanicalAccumulator(target - 1); // 递归累加 } }; 变形 class Solution { public: int…...

vuejs处理后端返回数字类型精度丢失问题

标题问题描述 后端返回数据有5.00和3.30这种数据&#xff0c;但是前端展示的时候返回对应分别为5和3.0&#xff0c;小数点后0都丢失了。 接口返回数据展示network-Response&#xff1a; 接口返回数据展示network-Preview&#xff1a; 错误数据效果展示 发现问题 浏览器接口…...

指令烧录ORIN NANO操作系统

1 概述 模组为ORIN NANO 4GB版本 Ubuntu系统为18.04虚拟机 说明&#xff1a;刷机过程会有重新连接USB的操作&#xff0c;烧写过程需要注意虚拟机提示&#xff0c;官方不建议使用虚拟机&#xff0c;建议直接使用ubuntu操作系统的机器。 2 下载烧录所需文件 进入到下载网址&am…...

每日算法刷题Day11 5.20:leetcode不定长滑动窗口求最长/最大6道题,结束不定长滑动窗口求最长/最大,用时1h20min

6. 1695.删除子数组的最大得分(中等) 1695. 删除子数组的最大得分 - 力扣&#xff08;LeetCode&#xff09; 思想 1.给你一个正整数数组 nums &#xff0c;请你从中删除一个含有 若干不同元素 的子数组**。**删除子数组的 得分 就是子数组各元素之 和 。 返回 只删除一个 子…...

List介绍

什么是List 在集合框架中&#xff0c;List是一个接口&#xff0c;继承自Collection Collection也是一个接口&#xff0c;该接口中规范了后序容器中常用的一些方法 Iterable也是一个接口&#xff0c;表示实现该接口的类是可以逐个元素进行遍历的&#xff0c;具体如下&#xff1…...

List更简洁的编码构建

List的链式构建需求 jdk1.8&#xff0c;编码时需要手动构建一个List 传统List构建方式 - 声明列表&#xff0c;依次添加元素&#xff1b; - 利用Arrays.asList方法声明并添加元素 //1 先声明创建 List<String> createList new ArrayList<>();//2 逐个添加元素 c…...

Java 单元测试框架比较:JUnit、TestNG 哪个更适合你?

Java 单元测试框架比较&#xff1a;JUnit、TestNG 哪个更适合你&#xff1f; 在 Java 开发领域&#xff0c;单元测试是保证代码质量的重要环节。而选择一个合适的单元测试框架&#xff0c;对于提升测试效率和代码可靠性至关重要。本文将深入比较 JUnit 和 TestNG 这两个主流的…...

变频器如何通过Profibus DP主站转Modbus RTU/TCP接入到上位机

变频器如何通过Profibus DP主站转Modbus RTU/TCP接入到上位机 在工业自动化领域&#xff0c;Profibus DP和Modbus RTU是两种常见的通信协议。Profibus DP广泛应用于工厂自动化&#xff0c;而Modbus RTU则常用于串行通信。当需要将支持Profibus DP的设备&#xff08;如变频器&a…...

ElasticSearch性能优化

ES基础概念介绍&#xff1a; 索引&#xff1a;类似于MySQL中的表&#xff0c;它是具有相同特征的一个数据集。文档&#xff1a;格式为JSON格式&#xff0c;类似于MySQL中的一条数据&#xff0c;它是数据存储的基本数据单元&#xff0c;每一条文档都有一个唯一的ID。查询&#x…...

pom.xml中的runtime

在 Maven 的 pom.xml 文件中&#xff0c;<scope> 元素可以指定依赖项的作用范围&#xff0c;而 runtime 是其中的一个作用范围值。以下是 runtime 作用范围的含义&#xff1a; 定义&#xff1a;runtime 作用范围表示该依赖项在编译时不需要&#xff0c;但在运行时需要。…...

第一章走进java世界

第一章 走进 java 世界 1.1 java 语言的特点(小面经) 使用简单、高效 去掉了 C 和 C++中的指针和多重继承技术通过垃圾自动回收机制简化了程序内存管理对于数据类型在不同字长的计算机上实现统一字节数占用完全面向对象 抽象、封装、继承、多态自动内存管理(Auto Garbage C…...

tigase源码学习笔记-事件总线EventBus

前言 最近看到了一些tigase关于异步事件解耦的地方的设计&#xff0c;做一个笔记记录一下。什么是事件总线&#xff0c;我自己对于事件总线的理解是&#xff0c;事件总线是一个对事件&#xff08;Event&#xff09;进行集中处理的基于发布订阅模式的一种处理机制。个人拙见&am…...

GO语言学习(六)

GO语言学习(六) 各位支持我的友友们我们现在不知不觉的已经进入第六期了&#xff0c;在上一期我们学习了一些简单的HTTP相关知识&#xff0c;还带大家基本的了解了web的深层原理及实现过程&#xff0c;相信大家应该都有自己的收获了&#xff0c;现在我就带领大家学习一下本期的…...

25.5.20学习总结

做题思路 数列分段 Section IIhttps://www.luogu.com.cn/problem/P1182正如题目所说&#xff0c;我们需要得到一个最小的最大段的值&#xff0c;可能有人将注意力放在分段上&#xff0c;事实上&#xff0c;我们更多的应该关注结果。这是一道二分答案的题&#xff0c;你可以先确…...

山东大学软件学院项目实训-基于大模型的模拟面试系统-Vditor编辑器上传图片

Vditor编辑器图片上传功能 使用Vditor进行图片上传时&#xff0c;会返回图片在后端的相对路径&#xff0c;这在前端是无法进行显示的&#xff08;如下图&#xff09; 于是为了将图片正常显示&#xff0c;我采取了和头像上传一样的解决方案&#xff0c;使用阿里云图床进行存储…...