C#设计模式12——代理模式的写法

1. 什么是代理模式?

代理模式是一种结构型设计模式,它允许通过代理对象来控制对真实对象的访问,以提供额外的功能或控制访问权限。

2. 代理模式的作用是什么?

代理模式可以为对象提供保护代理、远程代理、虚拟代理和智能引用四种不同的实现方式,以适应不同的应用场景。代理模式还可以用于延迟对象的创建和初始化,提高系统的性能和响应速度。

3. 代理模式的优点是什么?

代理模式可以提高代码的模块化和可维护性,将客户端代码和被代理对象解耦,降低系统的耦合度。代理模式还可以对对象进行透明处理,让使用者感觉不到代理的存在。

4. 代理模式的缺点是什么?

代理模式的缺点是可能会降低系统的性能和增加内存消耗,因为需要通过代理对象来访问真实对象。另外,在一些情况下需要特别注意保护代理的安全性,避免被恶意攻击。

5. 代理模式的适用场景是什么?

代理模式可以应用于网络通信、缓存处理、权限控制等多种场景,例如:

- 网络通信中的远程代理。
- 对象缓存中的虚拟代理。
- 访问控制中的保护代理。
- 对象生成和初始化时的延迟代理。
- 对象的复杂操作和事务管理中的智能引用代理。

6. 代理模式的实现方式有哪些?

代理模式的实现方式有静态代理和动态代理两种。静态代理需要手动编写代理类,而动态代理可以在运行时动态生成代理类,更加灵活和可扩展。C#中,静态代理可以使用实现接口或继承基类的方式实现,而动态代理则可以使用库函数或Emit技术进行实现。


以下是一个简单的C#代码示例,演示了如何使用静态代理来实现对一个Math类的加减乘除操作进行日志记录和权限验证:

// 接口定义Math操作
public interface IMath
{
    int Add(int x, int y);
    int Subtract(int x, int y);
    int Multiply(int x, int y);
    int Divide(int x, int y);
}

// 实现Math操作类
public class Math : IMath
{
    public int Add(int x, int y)
    {
        return x + y;
    }

    public int Subtract(int x, int y)
    {
        return x - y;
    }

    public int Multiply(int x, int y)
    {
        return x * y;
    }

    public int Divide(int x, int y)
    {
        return x / y;
    }
}

// 实现Math操作的代理类
public class MathProxy : IMath
{
    private Math math = new Math();

    // 记录日志的方法
    private void Log(string message)
    {
        Console.WriteLine("[Log] " + message);
    }

    // 验证权限的方法
    private bool CheckPermission()
    {
        Console.WriteLine("Checking permission...");
        return true;
    }

    public int Add(int x, int y)
    {
        Log("Add " + x + " and " + y + ".");
        if (CheckPermission())
        {
            return math.Add(x, y);
        }
        else
        {
            Console.WriteLine("Permission denied.");
            return 0;
        }
    }

    public int Subtract(int x, int y)
    {
        Log("Subtract " + y + " from " + x + ".");
        if (CheckPermission())
        {
            return math.Subtract(x, y);
        }
        else
        {
            Console.WriteLine("Permission denied.");
            return 0;
        }
    }

    public int Multiply(int x, int y)
    {
        Log("Multiply " + x + " and " + y + ".");
        if (CheckPermission())
        {
            return math.Multiply(x, y);
        }
        else
        {
            Console.WriteLine("Permission denied.");
            return 0;
        }
    }

    public int Divide(int x, int y)
    {
        Log("Divide " + x + " by " + y + ".");
        if (CheckPermission())
        {
            return math.Divide(x, y);
        }
        else
        {
            Console.WriteLine("Permission denied.");
            return 0;
        }
    }
}

// 客户端代码
class Client
{
    static void Main(string[] args)
    {
        IMath math = new MathProxy();

        Console.WriteLine("10 + 5 = " + math.Add(10, 5));
        Console.WriteLine("10 - 5 = " + math.Subtract(10, 5));
        Console.WriteLine("10 * 5 = " + math.Multiply(10, 5));
        Console.WriteLine("10 / 5 = " + math.Divide(10, 5));
    }
}

在这个示例中,我们创建了一个Math类作为被代理对象,并根据需要添加了日志记录和权限验证的功能。然后我们创建了一个MathProxy类作为静态代理对象,通过实现IMath接口并保存一个Math对象的引用来实现对Math类的操作。MathProxy类实现了IMath接口,所有的操作都由MathProxy类进行实现,可以在其中添加日志和权限验证等功能,并在需要时调用Math类的对应方法进行实际操作。

输出结果如下:

```
[Log] Add 10 and 5.
Checking permission...
10 + 5 = 15
[Log] Subtract 5 from 10.
Checking permission...
10 - 5 = 5
[Log] Multiply 10 and 5.
Checking permission...
10 * 5 = 50
[Log] Divide 10 by 5.
Checking permission...
10 / 5 = 2
``` 

可以看到,在客户端调用MathProxy类的操作时,实际上是由MathProxy类实现的,并添加了日志和权限验证的功能。由于MathProxy类实现了IMath接口,并保存了一个Math对象的引用,因此可以在需要时调用Math类的操作进行实际操作。这样可以将客户端代码和Math类解耦,并添加额外的功能。

posted @ 2023-05-09 21:27  明志德道  阅读(268)  评论(0编辑  收藏  举报