Java 结合实例学会使用 静态代理、JDK动态代理、CGLIB动态代理

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: Java 结合实例学会使用 静态代理、JDK动态代理、CGLIB动态代理

前言



代理 代理 代理 代理 代理 代理 代理 代理 代理 代理

代理 代理 代理 代理 代理 代理 代理 代理 代理 代理


很多人至今都是看到 代理就懵, 静态代理、动态代理、JDK动态代理、CGLIB动态代理......


知道AOP,知道增强,但是还是对代理模式支支吾吾? 这是因为你没有用心去了解过它......


您这种症状持续多久了?

现在看这篇文章,还来得及。

该篇文章咱们将会一起通过手敲实例代码,去了解和使用代理模式,实现静态代理、JDK动态代理、CGLIB动态代理。

实现完再回头看看,应该就不会支支吾吾了。


进入正文



一、代理模式


代理模式分为:


静态代理  


动态代理


先不管什么静还是动的,先知道这个代理模式的使用,是干嘛的。


文字描述:


为其他对象提供一个代理以控制对某个对象的访问。

代理类主要负责为委托了(真实对象)预处理消息、过滤消息、传递消息给委托类,代理类不现实具体服务,而是利用委托类来完成服务,并将执行结果封装处理。


听我说:

其实就是 在你调用 某个业务方法时, 在调用前 和调用后,加点东西。


加的这些东西 随时改动,也不会影响到业务方法代码的改动。 这样就很 解耦 ,就很喜欢。


就像, 调用某个业务方法前, 我加个 日志记录(记录一下调用的方法、参数、IP来源等等);


又比如说,调用某个业务方法前,我加个 权限拦截 (看看能不能给调用,判断一下token、身份、角色、ip等等)


从代理模式的实现效果来说,就是做到了2点:


1、可以对 业务代码(需要被代理的类)  进行  增强(附属一些逻辑代码)


2、实现代理之后 ,主业务方法不用动, 需要新增的,修改调整的一些业务,可以都丢到代理方法里面去做,这样一定程度实现了代码防入侵。


image.png

二、实例讲解


该篇文章使用到的实例是一个 我自己临时模拟的场景:


业务-  点餐下单


然后 运用上代理模式,实现添加一些 点餐下单 前前后后的  代码,记录日志、权限判断等等。


1.静态代理


1.1 就像往常一样,我们先写个接口 OrderService:


import java.util.Map;
/**
 * @Author : JCccc
 * @CreateTime : 2020/4/16
 * @Description :
 **/
public interface OrderService {
    //下单点餐
    Map executeFoodOrder(String userName,Map<String, Integer> foodMap);
}


1.2 就像往常一样,接口 OrderService的实现类 OrderServiceImpl:


import org.springframework.stereotype.Service;
import java.util.HashMap;
import java.util.Map;
/**
 * @Author : JCccc
 * @CreateTime : 2020/4/16
 * @Description :
 **/
@Service
public class OrderServiceImpl implements OrderService {
    @Override
    public  Map executeFoodOrder(String userName,Map<String, Integer> foodMap) {
        System.out.println("*********【下单主业务】*********");
        Map resultMap=new HashMap();
        for (Map.Entry<String, Integer> m : foodMap.entrySet()) {
            String foodName=m.getKey();
            Integer foodCount= m.getValue();
            //模拟一些杂七杂八的业务
            System.out.println("【下单主业务】"+foodName+"---模拟这个菜的一些杂七杂八的业务");
            resultMap.put(foodName,"ok");
        }
        return resultMap;
    }
}


1.3 现在我们有个业务需求,就是记录客户点餐信息,根据传入的用户名,去数据库查出来,然后跟点的菜啥的简单做个日志记录。


那么静态代理的方式是这样实现(这里介绍的是通过实现业务接口方式,而其实还可以简单通过继承来实现):


静态代理类   OrderServiceLogProxy.java :


import com.ilas.testboot.proxy.OrderService;
import java.util.Map;
/**
 * @Author : JCccc
 * @CreateTime : 2020/4/21
 * @Description :
 **/
public class OrderServiceLogProxy implements OrderService {
    private OrderServiceImpl orderService;
    public OrderServiceLogProxy(OrderServiceImpl orderService) {
        this.orderService = orderService;
    }
    @Override
    public Map executeFoodOrder(String userName, Map<String, Integer> foodMap) {
        System.out.println("+静态代理LogProxy");
        System.out.println("+下单前我们做点什么.");
        System.out.println("+正在获取用户:"+userName+"用户信息......");
        System.out.println("+正在记录用户:"+userName+"选择的菜品:"+foodMap.toString());
        System.out.println("+准备执行主业务");
        Map map = orderService.executeFoodOrder(userName, foodMap);
        System.out.println("+下单后我们做点什么.");
        System.out.println("+记录用户"+userName+"下单后的详情信息:"+map.toString());
        return map;
    }
}


简单分析下,我们新创建了一个实现下单点餐接口的类,在这类里面,重写下单方法executeFoodOrder,在调用下单方法时加上一些日志记录等等的代码:


image.png


OK,有可能这样写,你还是不能看出所为的代理结构,那么如果说我 写成这样呢?


是不是这样看起来就有感觉了。(该篇文章后面的JDK动态代理、CGLIB动态代理,我就不一一抽出来写成 before和 after了)


image.png


调用方式:


public static void main(String[] args) {
        //静态代理调用方式
        OrderServiceImpl orderService=new OrderServiceImpl();
        OrderServiceLogProxy orderServiceLogProxy=new OrderServiceLogProxy(orderService);
        String userName="JCccc";
        Map<String,Integer> orderMap=new HashMap<>();
        orderMap.put("白米饭",2);
        orderMap.put("红烧肉",1);
        orderMap.put("水煮鱼",1);
        orderMap.put("番茄炒蛋",1);
        Map resultMap = orderServiceLogProxy.executeFoodOrder(userName, orderMap);
        System.out.println("静态代理方法执行完毕,结果:"+resultMap);
    }


可以看下控制台的输出情况:


蓝色:静态代理类做的事情 (可以看到在主业务执行的前后)


红色:原本主业务做的事情


image.png


静态代理使用感觉:


实现代理的代码都是提前设计好,而且还是一一对应起来,OrderService 的代理是 OrderServiceLogProxy;


而且实现代理的方法也是一一对应。


那么如果 需要记录日志 的不止是OrderService ,再来一个 UserService呢? 再来一个 GameService呢?


这样一来,需要我们写死的代码就非常非常多。


这就是所谓的 静态 的不好之处,于是乎有了 动态代理(能够根据我们传入的 需要被代理类,实现代理)。


2.动态代理

动态代理有两种:


JDK动态代理


CGLIB动态代理


2.1 JDK动态代理


基于前面创建好的 OrderService 和 OrderServiceImpl ,我们接下来采取 JDK动态代理的方式去实现我们的日志添加。


(已经有种解耦的意味了吧,我们根本不需要动原本的业务接口和业务实现类,代理模式的作用无形中已经慢慢崭露出来了)


创建咱们的日志动态代理类,DynamicsLogProxy.java :


import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;
/**
 * @Author : JCccc
 * @CreateTime : 2020/4/21
 * @Description :
 **/
public class DynamicsLogProxy implements InvocationHandler {
    Object obj;
    //绑定委托对象,并返回代理类
    public Object bind(Object obj)
    {
        this.obj = obj;
        //绑定该类实现的所有接口,取得代理类
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(),
                obj.getClass().getInterfaces(),
                this);
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("+静态代理LogProxy");
        System.out.println("+下单前我们做点什么.");
        System.out.println("+正在获取目前传入的参数:"+Arrays.toString(args));
        System.out.println("+正在记录.....");
        System.out.println("+准备执行主业务");
        Object res = method.invoke(obj, args);
        System.out.println("+下单后我们做点什么.");
        System.out.println("+可得到业务方法执行后结果"+res.toString());
        System.out.println("+记录.......");
        return res;
    }
}


JDK动态代理的一些代码剖析:


1. JDK动态代理类 必须 实现 InvocationHandler 接口:


image.png


2.被代理的类 必须 存在 类 和实现的接口 (OrderService  和  OrderServiceImpl)


3.需要把 被代理对象传过去,因为动态生成代理类时要使用到:


image.png


3.使用jdk的api 中的 java.lang.reflect.Proxy 帮我们即时创建 代理实例对象:


image.png


image.png


前面两个参数,可以理解为把需要被代理的类相关信息传过去,最后一个参数  InvocationHandler h,


这玩意讲究。


image.png


咱们的DynamicsLogProxy 实现了这玩意,那么Proxy帮我们创建出来的一个代理实例对象,自然是不清楚咱们要代理的具体方法。


而invoke就是帮我们根据传入的参数:


proxy表示动态代理类实例,method表示调用的方法,args表示调用方法的参数


实现 调用 目标对象(被代理的类)的 目标方法。


简单点理解:


Proxy会帮我们 动态创建一个代理类,  这个代理类 代理的谁, 是我们传入的。


而InvocationHandler 的invoke方法,随时待命从代理类中去调用 被代理类的对应方法。


调用方式:


public static void main(String[] args) {
        //动态代理调用方式
        OrderService dynamicsOrderProxy = (OrderService)new DynamicsLogProxy().bind(new OrderServiceImpl());
        String userName="JCccc";
        Map<String,Integer> orderMap=new HashMap<>();
        orderMap.put("白米饭",2);
        orderMap.put("红烧肉",1);
        orderMap.put("水煮鱼",1);
        orderMap.put("番茄炒蛋",1);
        //通过动态代理类去调用不同方法
        Map resultMap = dynamicsOrderProxy.executeFoodOrder(userName, orderMap);
        System.out.println("动态代理方法执行完毕,结果:"+resultMap);
    }


可以看下控制台的输出情况:


image.png


JDK的动态代理使用感觉:


挺好,我们只需要编写一个日志代理类之后;


想这个日志代理类与哪个 需要被代理的类绑定, 我们就动态传入;


想要在被代理的类哪个方法前后 去嵌入 一些东西, 我们就动态调用方法;


很灵活,不像静态代理那样死死的。


但是,显然我们在使用JDK动态代理的时候,我们发现了,Proxy在帮我们动态即时创建 代理类的时候,要求我们传入


Class<?>[] interfaces  


那就是意味着,我们的需要被代理的类,必须实现接口:


image.png


因为JDK动态代理其实也是悄咪咪帮我们创建一份代理类代码,也帮我们实现同样的接口。


也就是说, 如果你的 被代理类, 没有实现接口,那么就没办法使用 JDK动态代理。


那怎么办?  CGLIB动态代理出来了,它说 我不关心你是否实现接口,只需要告诉我 哪个类需要被代理。


2.2 CGLIB动态代理


上面咱们已经说了,CGLIB动态代理只关心 哪个类需要被代理。


因为这个家伙其实是悄咪咪地帮我们 针对需要被代理的类 创建一个 代理子类。根据调用的方法,拦截调用到父类的方法。


看CGLIB动态代理实现,CglibLogProxy.java:


import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
/**
 * @Author : JCccc
 * @CreateTime : 2020/4/21
 * @Description :
 **/
public class CglibLogProxy  implements MethodInterceptor {
    /**
     * 自定义方法:利用Enhancer类生成代理类
     *
     * @param clazz
     * @param <T>
     * @return
     */
    public <T> T getObjByEnhancer(Class<T> clazz) {
        Enhancer enhancer = new Enhancer();
        enhancer.setSuperclass(clazz);
        enhancer.setCallback(this);
        T res = (T) enhancer.create();
        return res;
    }
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println(" +CGLIB日志代理CglibLogProxy");
        System.out.println("+CGLIB日志代理获取到执行的方法名" + method.getName());
        System.out.println("+做一些日志记录......");
        System.out.println("+做一些日志记录...");
        Object res = methodProxy.invokeSuper(o, objects);
        System.out.println("+主业务方法执行后我们做点什么.");
        System.out.println("+可得到业务方法执行后结果"+res.toString());
        System.out.println("+记录.......");
        return res;
    }
}


实现代码简单剖析:


1. implements MethodInterceptor    cglib动态代理类实现了这个方法拦截


2. intercept 方法,这个也就是 提供地方给咱们 添加 日志记录、权限判断等等的地方


3. Enhancer


Enhancer  这个玩意是CGLIB动态代理类的核心。


它的职责是,创建出一个子类(代理类), 然后将我们传入的 被代理类  设置成 SuperClass 父类 。


这个子类(代理类) ’继承‘ 了父类( 被代理类)的 所有 可以公开的方法 ,然后设置拦截回调的类(父类),子类中拦截父类方法并织入方法增强逻辑。


ps:cglib动态代理不能代理声明为final类型的类和方法  


事不宜迟,我们贴出CGLIB动态代理调用方式:


public static void main(String[] args) {
        CglibLogProxy  cglibLogProxy=new CglibLogProxy();
        OrderServiceImpl   orderProxy=   cglibLogProxy.getObjByEnhancer(OrderServiceImpl.class);
        String userName="JCccc";
        Map<String,Integer> orderMap=new HashMap();
        orderMap.put("白米饭",2);
        orderMap.put("红烧肉",1);
        orderMap.put("水煮鱼",1);
        orderMap.put("番茄炒蛋",1);
        Map map = orderProxy.executeFoodOrder(userName, orderMap);
        System.out.println(map.toString());
    }


注意看, 调用方法的时候,用的是什么?


orderProxy.executeFoodOrder(userName, orderMap);


没错,跟JDK动态代理看似一样直接调用 我们执行的业务方法,


但是接下来就不一样了,再看看cglib动态代理类的实现代码:


image.png


没错,orderProxy是我们通过Enhancer去生成的,特意设置了回调拦截方法。从拦截的这个意思去看,就更有AOP的意味,更有代理的感觉了。


调用父类方法(被代理类)前,拦住, 先调用咱们自己写的 拦截方法, 啥时候完事了,再通过 methodProxy.invokeSuper 切回到主业务方法去。


可以看到控制台输出:


image.png


简单的介绍就到这里吧。


总结



静态代理 和 动态代理  对比 :


静态代理是 一开始咱们编码的时候,已经写死了,哪个代理类对应哪个被代理类。


而动态代理都是 运行时动态即时创建的。


疑问:


那么是不是静态代理就肯定不如动态代理?


那肯定不能这么说,人家静态代理一开始就码好了,运行调用的时候,肯定就快啊。


而动态代理时运行时才去开始,所以自然也会慢。


JDK动态代理 和 CGLIB动态代理:


JDK动态代理,代理的类 必须实现接口,没有实现接口的类 无法使用JDK动态代理。


CGLIB动态代理,代理的类必须能被子类继承使用(方法也得能被重写),所以不能代理声明为final类型的类和方法。


多级代理实现方式:


补充:


如果我们想实现, 在调用下单点餐这个方法时,  不单是 进行日志记录 ,我们还想进行 一个营业判断或者说权限判断。


(JDK动态代理-多层级代理)

这时就会引出一个, 多级代理的概念。


也就是说,


image.png


对应的动态代理切点不止一个:


image.png


我们需要的流程是, 当通过代理方式去调用主业务方法 executeFoodOrder ,先给我进行 日志记录, 再给我进行 营业时间判断(可以类比成一些权限判断、角色判断、身份判断等等),时间判断符合在营业时间时,最后再给我去调用主业务方法 executeFoodOrder 。


实现代码,两个动态代理类


DynamicsLogProxy.java:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;
/**
 * @Author : JCccc
 * @CreateTime : 2020/4/21
 * @Description :
 **/
public class DynamicsLogProxy implements InvocationHandler {
    Object obj;
    //绑定委托对象,并返回代理类
    public Object bind(Object obj)
    {
        this.obj = obj;
        //绑定该类实现的所有接口,取得代理类
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(), obj.getClass().getInterfaces(), this);
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("+JDK动态代理LogProxy");
        System.out.println("+下单前我们做点什么.");
        System.out.println("+正在获取目前传入的参数:"+Arrays.toString(args));
        System.out.println("+正在记录.....");
        System.out.println("+准备执行主业务");
        Object res = method.invoke(obj, args);
        System.out.println("+JDK动态代理LogProxy下单后我们做点什么.");
        System.out.println("+可得到业务方法执行后结果"+res.toString());
        System.out.println("+记录.......");
        return res;
    }
}
DynamicsOperateProxy.java:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
 * @Author : JCccc
 * @CreateTime : 2020/4/21
 * @Description :
 **/
public class DynamicsOperateProxy implements InvocationHandler {
    Object obj;
    //绑定委托对象,并返回代理类
    public Object bind(Object obj)
    {
        this.obj = obj;
        //绑定该类实现的所有接口,取得代理类
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(),
                obj.getClass().getInterfaces(),
                this);
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("-----JDK动态代理 OperateProxy 执行主方法前-----");
        Object res =null;
        if (method.getName().equals("executeFoodOrder")){
            System.out.println("-----检测当前时需要下单点餐......");
            System.out.println("-----执行相关判断......");
            System.out.println("-----查看该店是否营业......");
            System.out.println("-----检测XXXXX......");
            System.out.println("-----可以调用主业务");
            // 如果不符合要求, 不执行 method.invoke(obj, args);
            //做拦截处理即可,抛出异常或者重定向等等
            res=   method.invoke(obj, args);
            System.out.println("-----JDK动态代理OperateProxy 执行主方法后----");
        }
        return res;
    }
}


注意此时 ,多级代理的 调用方式:


public static void main(String[] args) {
        OrderService operateProxy= (OrderService) new DynamicsOperateProxy().bind(new OrderServiceImpl());
        OrderService operateAndLogProxy = (OrderService)new DynamicsLogProxy().bind(operateProxy);
        Map<String,Integer> orderMap=new HashMap();
        orderMap.put("白米饭",2);
        orderMap.put("红烧肉",1);
        orderMap.put("水煮鱼",1);
        orderMap.put("番茄炒蛋",1);
        Map result = operateAndLogProxy.executeFoodOrder("JCccc",orderMap);
        System.out.println(result.toString());
    }


细看:


image.png


我们先是让 需要被代理的类 绑定到  DynamicsOperateProxy 上,


然后绑定完创建出来的类,我们继续 绑定到 DynamicsLogProxy 上,然后创建我们最后的多级动态代理类。


看控制台输出:


注意到此时的调用执行顺序,


第一步,进入到了log动态代理,执行 调用主业务方法 前 的 代码;


第二步,即将切回到主业务方法时,发现还有一级代理,operate动态代理,那么进入 operate动态代理 ,执行相关的 执行 调用主业务方法 前 的 代码;


第三步,切入主业务方法,拿到主业务方法的返回结果,回到 operate动态代理,执行 主业务方法调用后的 代码;


第四步,把 ‘拿到主业务方法的返回结果’ 往外面一层传递,也就是传递给 log动态代理,  log动态代理执行 调用主业务方法 后 的 代码。


第五步,最后把最终结果返回到最外面去。


image.png


所以多层级代理使用的时候,需要自己多注意动态代理实现的顺序,记得多测试看看流程。


那么可以看到这种多层级代理的实现方式是使用的JDK动态代理,那么如果我们想用CGLIB动态代理方式去实现这种多层级代理呢?


对于这个我也思索了很久,还未能找到一些实现起来很舒服的方式。


如果是基于Spirng框架或者是Springboot,那么实现多层级代理,直接去使用  注解AOP方式去,多重拦截判断算了。


因为SpringBoot默认的AOP实现就是使用的CGLIB动态代理代理,当然你想切换成JDK动态方式也是可以的,改下配置项就像,但是如果你真的改成JDK动态代理方式,记得,你所编写的需要被代理的类,必须实现接口。


相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
5天前
|
Java
Java——接口的使用实例
Comparable接口用于自定义类的对象比较。通过实现此接口并重写`compareTo`方法,可以定义自定义类型的比较规则。 接下来介绍了Comparator接口,它提供了一种更灵活的比较方式。通过实现Comparator接口并重写`compare`方法,可以根据不同属性定义不同的比较规则。例如,定义一个`BrandComparator`类来比较汽车的品牌。 最后,介绍了Cloneable接口,用于实现对象的克隆。实现该接口并重写`clone`方法后,可以创建对象的浅拷贝或深拷贝。浅拷贝仅复制对象本身,深拷贝则会递归复制所有成员变量。
12 4
Java——接口的使用实例
|
6天前
|
存储 Java
Java内置数据类型和实例的详解
Java内置数据类型分为基本和引用两类。基本数据类型包括整型(`byte`、`short`、`int`、`long`)、浮点型(`float`、`double`)、字符型(`char`)和布尔型(`boolean`),用于存储简单的数值;引用数据类型则用于存储对象的引用,包括类(如`String`)、接口和数组。掌握这两类数据类型是Java编程的基础。以下示例展示了各种数据类型的使用方法。
|
7天前
|
Java
Java实例详解
Java实例是通过类创建的对象,其核心在于将抽象的类定义转化为具体的实体。类作为对象的模板定义了属性和行为,而实例则是这些定义的具体实现。通过`new`关键字可以创建实例,并利用点运算符访问其属性和方法。实例拥有自己的生命周期,从创建到使用直至被垃圾回收机制自动清理。此外,实例变量和静态变量的区别在于前者属于单个实例,后者则为所有实例共享。理解Java实例的概念及其管理对编程至关重要。
|
13天前
|
安全 Java API
【性能与安全的双重飞跃】JDK 22外部函数与内存API:JNI的继任者,引领Java新潮流!
【9月更文挑战第7天】JDK 22外部函数与内存API的发布,标志着Java在性能与安全性方面实现了双重飞跃。作为JNI的继任者,这一新特性不仅简化了Java与本地代码的交互过程,还提升了程序的性能和安全性。我们有理由相信,在外部函数与内存API的引领下,Java将开启一个全新的编程时代,为开发者们带来更加高效、更加安全的编程体验。让我们共同期待Java在未来的辉煌成就!
41 11
|
11天前
|
监控 Java 大数据
【Java内存管理新突破】JDK 22:细粒度内存管理API,精准控制每一块内存!
【9月更文挑战第9天】虽然目前JDK 22的确切内容尚未公布,但我们可以根据Java语言的发展趋势和社区的需求,预测细粒度内存管理API可能成为未来Java内存管理领域的新突破。这套API将为开发者提供前所未有的内存控制能力,助力Java应用在更多领域发挥更大作用。我们期待JDK 22的发布,期待Java语言在内存管理领域的持续创新和发展。
|
14天前
|
Oracle Java 关系型数据库
【颠覆性升级】JDK 22:超级构造器与区域锁,重塑Java编程的两大基石!
【9月更文挑战第6天】JDK 22的发布标志着Java编程语言在性能和灵活性方面迈出了重要的一步。超级构造器和区域锁这两大基石的引入,不仅简化了代码设计,提高了开发效率,还优化了垃圾收集器的性能,降低了应用延迟。这些改进不仅展示了Oracle在Java生态系统中的持续改进和创新精神,也为广大Java开发者提供了更多的可能性和便利。我们有理由相信,在未来的Java编程中,这些新特性将发挥越来越重要的作用,推动Java技术不断向前发展。
|
24天前
|
Java 应用服务中间件 HSF
Java应用结构规范问题之AllLoggers接口获取异常日志的Logger实例的问题如何解决
Java应用结构规范问题之AllLoggers接口获取异常日志的Logger实例的问题如何解决
|
19天前
|
C# Windows 开发者
当WPF遇见OpenGL:一场关于如何在Windows Presentation Foundation中融入高性能跨平台图形处理技术的精彩碰撞——详解集成步骤与实战代码示例
【8月更文挑战第31天】本文详细介绍了如何在Windows Presentation Foundation (WPF) 中集成OpenGL,以实现高性能的跨平台图形处理。通过具体示例代码,展示了使用SharpGL库在WPF应用中创建并渲染OpenGL图形的过程,包括开发环境搭建、OpenGL渲染窗口创建及控件集成等关键步骤,帮助开发者更好地理解和应用OpenGL技术。
63 0
|
19天前
|
开发者 C# 容器
【独家揭秘】当WPF邂逅DirectX:看这两个技术如何联手打造令人惊艳的高性能图形渲染体验,从环境搭建到代码实践,一步步教你成为图形编程高手
【8月更文挑战第31天】本文通过代码示例详细介绍了如何在WPF应用中集成DirectX以实现高性能图形渲染。首先创建WPF项目并使用SharpDX作为桥梁,然后在XAML中定义承载DirectX内容的容器。接着,通过C#代码初始化DirectX环境,设置渲染逻辑,并在WPF窗口中绘制图形。此方法适用于从简单2D到复杂3D场景的各种图形处理需求,为WPF开发者提供了高性能图形渲染的技术支持和实践指导。
49 0
|
20天前
|
算法 Java 开发者
你的Java代码还可以这样写:JavaIO的简单代码实例和展示
在编程世界中,Java无疑是璀璨的明星。本文通过两个示例介绍抽象类与接口的魅力:首先,通过抽象类`Shape`及其子类`Circle`和`Rectangle`展示多态性;接着,通过接口`PerimeterCalculator`为形状类添加周长计算功能,展现了接口的灵活性。掌握这两者将助你在Java编程中更进一步,应对复杂项目游刃有余。
33 0