zoukankan      html  css  js  c++  java
  • 增强请求响应——代理模式

    写博客的目的是记录我的学习心得,方便自己以后查看,可能写的不是很高深,不是很对,欢迎各位看客留言评论~

    这篇文章记录我对web请求的理解,以及增强请求和响应的方式之一。

    用户状态保持

    从登录开始,当用户填写完数据并发送请求之后,我们需要到数据库去查询用户的记录进行比对,虽然在一次会话中会有很多的请求和响应,但是请求和响应是没有状态的,所以当用户登录成功之后,我们需要将用户的登录状态进行保存,一个比较古老的方案是将登录状态存在服务端的Session对象中,一种是存在客户端的Cookie中,Session的实现是基于cookie的,当我们将登录状态存到session中之后,响应给浏览器一个sessionId,客户端利用cookie的方式,将这个sessionid保存下来,下次发送请求的时候,带着这个sessionId来服务端查找数据。恢复状态。

    拦截器对用户的请求做处理

    当用户的请求发送过来之后,并不是第一时间就到了我们要访问的资源那里去,会先经过一个拦截器,由拦截器对数据做处理并判断是放行还是转发,当拦截器做了一些响应的处理并放行之后,请求访问完资源,做出响应之后,响应也会再一次经过这个拦截器,拦截器再做一些相应的处理。响应回来之后会直接执行放行后面的代码~

    我的理解是,在后端,对于请求的增强和响应的增强也是通过这个拦截器来做的,但是代理模式并不是只适用于后端,它是一种思想,是一种解决方案,可以适用于很多地方,这里就记录在后端部分的实现~

    代理模式

    在软件开发过程中,有些对象有时候会因为各种阻碍,我们不能直接访问到,或者如果直接访问会给系统带来不必要的复杂性,这个时候,我们可以在客户端和目标对象之间加一层中间层,让代理对象替代这些对象,然后客户端只需要访问代理对象,由代理对象去帮我们去请求对象并返回结果给客户端。

    代理模式的分类

    远程代理(Remote):为一个位于不同地址空间的对象提供一个局部代理对象。这个不同的地址空间可以在本电脑中,也可以在另一台电脑中,最典型的例子就是--客户端调用web服务或者WCF服务

    虚拟(Virtual)代理:根据需要创建一个资源消耗较大的对象,使得对象只在需要时才会被真正创建

    Copy-on-Write代理虚拟代理的一种,把复制(或者叫克隆)拖延到只有在客户端需要时,才真正采取行动。

    保护(Protect or Access)代理控制一个对象的访问,可以给不同的用户提供不同级别的使用权限。

    防火墙(Firewall)代理:保护目标不让恶意用户接近

    智能引用(Smart Reference)代理:当一个对象被引用时,提供一些额外的操作,比如将对此对象调用的次数记录下来等。

    Cache代理:为某一个目标操作的结果提供临时的存储空间,以便多个客户端可以这些结果

    在这些代理模式中,虚拟代理,远程代理,只能引用代理和保护代理较为常见。

    代理模式定义:

    就是给某一个对象提供一个代理,并由代理对象控制对原对象的引用,一些情况下,一个客户不想或不能直接引用一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用,例如:电脑桌桌面的快捷方式就是一个代理对象,快捷方式是他所引用的程序的一个代理。

    .Net实现代理模式的例子:

    场景模拟

    在现实生活中,如果有同事出国或者朋友出国的情况下,我们经常会拖这位朋友帮忙带一些电子产品或化妆品等东西,这个场景中,出国的朋友就是一个代理,他(她)是他(她)朋友的一个代理,由于他朋友不能去国外买东西,他却可以,所以朋友们都托他帮忙带一些东西的

    // 客户端调用
        class Client
        {
            static void Main(string[] args)
            {
                // 创建一个代理对象并发出请求
                Person proxy = new Friend();
                proxy.BuyProduct();
                Console.Read();
            }
        }
    
        // 抽象主题角色
        public abstract class Person
        {
            public abstract void BuyProduct();
        }
    
        //真实主题角色
        public class RealBuyPerson : Person
        {
            public override void BuyProduct()
            {
                Console.WriteLine("帮我买一个IPhone和一台苹果电脑");
            }
        }
    
        // 代理角色
        public class Friend:Person
        {
            // 引用真实主题实例
            RealBuyPerson realSubject;
            //相当于拦截器中的实现方法(只是类似,表述可能不太准确)
            public override void BuyProduct()
            {
                Console.WriteLine("通过代理类访问真实实体对象的方法");
                if (realSubject == null)
                {
                    realSubject = new RealBuyPerson();
                }
                //调用真实角色前的一些增强,相当于对请求(request)的增强.(只是类似,表述可能不太准确)
                this.PreBuyProduct();
                // 调用真实主题方法
                realSubject.BuyProduct();
                //调用真实角色后的一些增强,相当于对响应(response)的增强,(只是类似,表述可能不太准确)
                this.PostBuyProduct();
            }
    
            // 代理角色执行的一些操作
            public void PreBuyProduct()
            {
                // 可能不知一个朋友叫这位朋友带东西,首先这位出国的朋友要对每一位朋友要带的东西列一个清单等
                Console.WriteLine("我怕弄糊涂了,需要列一张清单,张三:要带相机,李四:要带Iphone...........");
            }
            
            // 买完东西之后,代理角色需要针对每位朋友需要的对买来的东西进行分类
            public void PostBuyProduct()
            {
                Console.WriteLine("终于买完了,现在要对东西分一下,相机是张三的;Iphone是李四的..........");
            }
        }

    抽象主题角色(Person):声明了真实主题和代理主题的公共接口,这样一来在使用真实主题的任何地方都可以使用代理主题。

    代理主题角色(Friend):代理主题角色内部含有对真实主题的引用,从而可以操作真实主题对象;代理主题角色负责在需要的时候创建真实主题对象;代理角色通常在将客户端调用传递到真实主题之前或之后,都要执行一些其他的操作,而不是单纯地将调用传递给真实主题对象。例如这里的PreBuyProduct和PostBuyProduct方法就是代理主题角色所执行的其他操作。

    真实主题角色(RealBuyPerson):定义了代理角色所代表的真是对象。

    附:在实际开发过程中,我们在客户端添加服务引用的时候,在客户程序中会添加一些额外的类,在客户端生成的类扮演着代理主题角色,我们客户端也是直接调用这些代理角色来访问远程服务提供的操作。这个是远程代理的一个典型例子。

    Java语言使用代理模式对request和response增强

    在Java中可以在拦截器中通过Proxy.newProxyInstance()方法增强请求和响应

    public interface SaleComputer {
    
        public String sale(double money);
    
        public void show();
    }
    
    
    /**
     * 真实类
     */
    public class Lenovo implements SaleComputer {
        @Override
        public String sale(double money) {
    
            System.out.println("花了"+money+"元买了一台联想电脑...");
            return "联想电脑";
        }
    
        @Override
        public void show() {
            System.out.println("展示电脑....");
        }
    }
    
    
    import java.lang.reflect.InvocationHandler;
    import java.lang.reflect.Method;
    import java.lang.reflect.Proxy;
    
    public class ProxyTest {
    
        public static void main(String[] args) {
            //1.创建真实对象
            Lenovo lenovo = new Lenovo();
            
            //2.动态代理增强lenovo对象
            /*
                三个参数:
                    1. 类加载器:真实对象.getClass().getClassLoader()
                    2. 接口数组:真实对象.getClass().getInterfaces()
                    3. 处理器:new InvocationHandler()
             */
            SaleComputer proxy_lenovo = (SaleComputer) Proxy.newProxyInstance(lenovo.getClass().getClassLoader(), lenovo.getClass().getInterfaces(), new InvocationHandler() {
    
                /*
                    代理逻辑编写的方法:代理对象调用的所有方法都会触发该方法执行
                        参数:
                            1. proxy:代理对象
                            2. method:代理对象调用的方法,被封装为的对象
                            3. args:代理对象调用的方法时,传递的实际参数
                 */
                @Override
                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                   
                    //判断是否是sale方法
                    if(method.getName().equals("sale")){
                        //1.增强参数
                        double money = (double) args[0];
                        money = money * 0.85;
                        System.out.println("专车接你....");
                        //使用真实对象调用该方法
                        String obj = (String) method.invoke(lenovo, money);
                        System.out.println("免费送货...");
                        //2.增强返回值
                        return obj+"_鼠标垫";
                    }else{
                        Object obj = method.invoke(lenovo, args);
                        return obj;
                    }
    
    
    
                }
            });
    
            //3.调用方法
    
           /* String computer = proxy_lenovo.sale(8000);
            System.out.println(computer);*/
    
            proxy_lenovo.show();
        }
    }

    参考博客:

    https://www.cnblogs.com/zhili/p/ProxyPattern.html

  • 相关阅读:
    拓扑排序
    最少硬币问题(无穷硬币)
    第三届蓝桥杯预赛
    矩形嵌套
    《心术》影评
    区间选点问题
    [转贴] 游戏服务器架构
    D3DXPlaneFromPoints 函数
    内存池实现
    [转]MMORPG游戏服务器端的设计
  • 原文地址:https://www.cnblogs.com/Xiaomoml/p/12290258.html
Copyright © 2011-2022 走看看