1、问题

问题:想要添加日志记录、性能监控、安全监测

2、最初解决方案

2.1、最初解决方案

缺点:太多重复代码,且紧耦合

2.2、抽象类进行共性设计,子类进行个性设计,此处不讲解,缺点一荣俱荣,一损俱损

2.3、使用装饰器模式/代理模式改进的解决方案

装饰器模式:动态地给一个对象添加一些额外的职责。就增加功能来说, 装饰器模式相比生成子类更为灵活。
代理模式:为其他对象提供一种代理以控制对这个对象的访问。




 
缺点:紧耦合,每个业务逻辑需要一个装饰器实现或代理

2.4、JDK动态代理解决方案(比较通用的解决方案) 
  1. public class MyInvocationHandler implements InvocationHandler {
  2. private Object target;
  3. public MyInvocationHandler(Object target) {
  4. this.target = target;
  5. }
  6. @Override
  7. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  8. //1.记录日志    2.时间统计开始      3.安全检查
  9. Object retVal = method.invoke(target, args);
  10. //4.时间统计结束
  11. return retVal;
  12. }
  13. public static Object proxy(Object target) {
  14. return Proxy.newProxyInstance(target.getClass().getClassLoader(),
  15. target.getClass().getInterfaces(), new MyInvocationHandler(target));
  16. }
  17. }

编程模型

  1. //proxy     在其上调用方法的代理实例
  2. //method 拦截的方法
  3. //args       拦截的参数
  4. Override
  5. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  6. Object retVal=null;
  7. //预处理
  8. //前置条件判断
  9. boolean ok = true;
  10. if(!ok) {//不满足条件
  11. throw new RuntimeException("你没有权限");
  12. }
  13. else {//反射调用目标对象的某个方法
  14. retVal = method.invoke(target, args);
  15. }
  16. //后处理
  17. return retVal;
  18. }

缺点:使用麻烦,不能代理类,只能代理接口

CGLIB动态代理解决方案(比较通用的解决方案)
  1. public class MyInterceptor implements MethodInterceptor  {
  2. private Object target;
  3. public MyInterceptor(Object target) {
  4. this.target = target;
  5. }
  6. @Override
  7. public Object intercept(Object proxy, Method method, Object[] args,
  8. MethodProxy invocation) throws Throwable {
  9. //1.记录日志 2.时间统计开始   3.安全检查
  10. Object retVal = invocation.invoke(target, args);
  11. //4.时间统计结束
  12. return retVal;
  13. }
  14. public static Object proxy(Object target) {
  15. return Enhancer.create(target.getClass(), new MyInterceptor(target));
  16. }
  17. }

编程模型

  1. //proxy 在其上调用方法的代理实例    method拦截的方法    args  拦截的参数
  2. //invocation 用来去调用被代理对象方法的
  3. @Override
  4. public Object intercept(Object proxy, Method method, Object[] args,
  5. MethodProxy invocation) throws Throwable {
  6. //预处理
  7. //前置条件判断
  8. boolean ok = true;
  9. if(!ok) {//不满足条件
  10. throw new RuntimeException("出错了");
  11. }
  12. else {//调用目标对象的某个方法
  13. Object retVal = invocation.invoke(target, args);
  14. }
  15. //后处理
  16. return retVal;
  17. }
优点:能代理接口和类
缺点:使用麻烦,不能代理final类

动态代理本质

本质:对目标对象增强
           最终表现为类(动态创建子类),看手工生成(子类)还是自动生成(子类)
代理限制:
           只能在父类方法被调用之前或之后进行增强(功能的修改),不能在中间进行修改,要想在方法调用中增强,需要ASM(java 字节码生成库)
其他动态代理框架
jboss:javassist (hibernate 3.3中默认为javassist)
                           (hibernate 3.3之前中默认为cglib)


 




 
2.5、AOP解决方案(通用且简单的解决方案)
  1. @Aspect
  2. public class PayEbiAspect {
  3. @Pointcut(value="execution(* pay(..))")
  4. public void pointcut() {}
  5. @Around(value="pointcut()")
  6. public Object around(ProceedingJoinPoint pjp) throws Throwable {
  7. //1.记录日志
  8. //2.时间统计开始
  9. //3.安全检查
  10. Object retVal = pjp.proceed();//调用目标对象的真正方法
  11. //4.时间统计结束
  12. return retVal;
  13. }
  14. }
编程模型
  1. //2 切入点
  2. @Pointcut(value="execution(* *(..))")
  3. public void pointcut() {}
  4. //3 拦截器的interceptor
  5. @Around(value="pointcut()")
  6. public Object around(ProceedingJoinPoint pjp) throws Throwable {
  7. Object retVal=null;
  8. //预处理
  9. //前置条件判断
  10. boolean ok = true;
  11. if(!ok) {//不满足条件
  12. throw new RuntimeException("你没有权限");
  13. }
  14. else {//调用目标对象的某个方法
  15. retVal = pjp.proceed();
  16. }
  17. //后处理
  18. return retVal;
  19. }
缺点:依赖AOP框架 

AOP入门
概念:
n关注点:可以认为是所关注的任何东西,比如上边的支付组件;
n关注点分离:将问题细化为单独部分,即可以理解为不可再分割的组件,如上边的日志组件和支付组件;
n横切关注点:会在多个模块中出现,使用现有的编程方法,横切关注点会横越多个模块,结果是使系统难以设计、理解、实现和演进,如日志组件横切于支付组件。
织入:横切关注点分离后,需要通过某种技术将横切关注点融合到系统中从而完成需要的功能,因此需要织入,织入可能在编译期、加载期、运行期等进行。

nAOP是什么(Aspect   Oriented   Programming)
 AOP是一种编程范式,提供从另一个角度来考虑程序结构以完善面向对象编程(OOP)。
 AOP为开发者提供了一种描述横切关注点的机制,并能够自动将横切关注点织入到面向对象的软件系统中,从而实现了横切关注点的模块化。
 AOP能够将那些与业务无关,却为业务模块所共同调用的逻辑或责任,例如事务处理、日志管理、权限控制等,封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。
nAOP能干什么,也是AOP带来的好处
1:降低模块的耦合度
2:使系统容易扩展
3:设计决定的迟绑定:使用AOP,设计师可以推迟为将来的需求作决定,因为它
可以把这种需求作为独立的方面很容易的实现。
4:更好的代码复用性


 
AOP基本概念
 连接点(Joinpoint):
    表示需要在程序中插入横切关注点的扩展点,连接点可能是类初始化、方法执行、方法调用、字段调用或处理异常等等,Spring只支持方法执行连接点,在AOP中表示为“在哪里做”
切入点(Pointcut):
    选择一组相关连接点的模式,即可以认为连接点的集合,Spring支持perl5正则表达式和AspectJ切入点模式,Spring默认使用AspectJ语法,在AOP中表示为“在哪里做的集合”
增强(Advice):或称为增强
    在连接点上执行的行为,增强提供了在AOP中需要在切入点所选择的连接点处进行扩展现有行为的手段;包括前置增强(before advice)、后置增强 (after advice)、环绕增强 (around advice),在Spring中通过代理模式实现AOP,并通过拦截器模式以环绕连接点的拦截器链织入增强 ;在AOP中表示为“做什么”;
方面/切面(Aspect):
      横切关注点的模块化,比如上边提到的日志组件。可以认为是增强、引入和切入点的组合;在Spring中可以使用Schema和@AspectJ方式进行组织实现;在AOP中表示为“在哪里做和做什么集合”;
目标对象(Target Object):
    需要被织入横切关注点的对象,即该对象是切入点选择的对象,需要被增强的对象,从而也可称为“被增强对象”;由于Spring AOP 通过代理模式实现,从而这个对象永远是被代理对象,在AOP中表示为“对谁做”
AOP代理(AOP Proxy):
    AOP框架使用代理模式创建的对象,从而实现在连接点处插入增强(即应用切面),就是通过代理来对目标对象应用切面。在Spring中,AOP代理可以用JDK动态代理或CGLIB代理实现,而通过拦截器模型应用切面。
织入(Weaving):
    织入是一个过程,是将切面应用到目标对象从而创建出AOP代理对象的过程,织入可以在编译期、类装载期、运行期进行。
引入(inter-type declaration):
    也称为内部类型声明,为已有的类添加额外新的字段或方法,Spring允许引入新的接口(必须对应一个实现)到所有被代理对象(目标对象), 在AOP中表示为“做什么(新增什么)”

AOP的Advice类型
前置增强(Before advice):
    在某连接点之前执行的增强,但这个增强不能阻止连接点前的执行(除非它抛出一个异常)。
后置返回增强(After returning advice):
    在某连接点正常完成后执行的增强:例如,一个方法没有抛出任何异常,正常返回。
后置异常增强(After throwing advice):
    在方法抛出异常退出时执行的增强。
后置最终增强(After (finally) advice):
    当某连接点退出的时候执行的增强(不论是正常返回还是异常退出)。
环绕增强(Around Advice):
    包围一个连接点的增强,如方法调用。这是最强大的一种增强类型。 环绕增强可以在方法调用前后完成自定义的行为。它也会选择是否继续执行连接点或直接返回它们自己的返回值或抛出异常来结束执行。


AOP开发步骤
  类似于IoC/DI容器开发步骤,需要描述哪个连接点需要哪个通用功能(增强

横切关注点的表现有:  

  ·代码纠结/混乱——
当一个模块或代码段同时管理多个关注点时发生这种情况。 

  ·代码分散——当一个关注点分布在许多模块中并且未能很好地局部化和模块化时发生这种情况 。
AOP包括三个清晰的开发步骤:
1:功能横切:找出横切关注点。
2:实现分离:各自独立的实现这些横切关注点所需要完成的功能。
3:功能回贴:在这一步里,方面集成器通过创建一个模块单元—— 方面来指定重组的规则。重组过程——也叫织入或结合—— 则使用这些信息来构建最终系统。

推荐阅读书籍:
AspectJ in Action

AOSD中文版--基于用例的面向方面软件开发


1、问题

问题:想要添加日志记录、性能监控、安全监测

2、最初解决方案

2.1、最初解决方案

缺点:太多重复代码,且紧耦合

2.2、抽象类进行共性设计,子类进行个性设计,此处不讲解,缺点一荣俱荣,一损俱损

2.3、使用装饰器模式/代理模式改进的解决方案

装饰器模式:动态地给一个对象添加一些额外的职责。就增加功能来说, 装饰器模式相比生成子类更为灵活。
代理模式:为其他对象提供一种代理以控制对这个对象的访问。




 
缺点:紧耦合,每个业务逻辑需要一个装饰器实现或代理

2.4、JDK动态代理解决方案(比较通用的解决方案) 
  1. public class MyInvocationHandler implements InvocationHandler {
  2. private Object target;
  3. public MyInvocationHandler(Object target) {
  4. this.target = target;
  5. }
  6. @Override
  7. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  8. //1.记录日志    2.时间统计开始      3.安全检查
  9. Object retVal = method.invoke(target, args);
  10. //4.时间统计结束
  11. return retVal;
  12. }
  13. public static Object proxy(Object target) {
  14. return Proxy.newProxyInstance(target.getClass().getClassLoader(),
  15. target.getClass().getInterfaces(), new MyInvocationHandler(target));
  16. }
  17. }

编程模型

  1. //proxy     在其上调用方法的代理实例
  2. //method 拦截的方法
  3. //args       拦截的参数
  4. Override
  5. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  6. Object retVal=null;
  7. //预处理
  8. //前置条件判断
  9. boolean ok = true;
  10. if(!ok) {//不满足条件
  11. throw new RuntimeException("你没有权限");
  12. }
  13. else {//反射调用目标对象的某个方法
  14. retVal = method.invoke(target, args);
  15. }
  16. //后处理
  17. return retVal;
  18. }

缺点:使用麻烦,不能代理类,只能代理接口

CGLIB动态代理解决方案(比较通用的解决方案)
  1. public class MyInterceptor implements MethodInterceptor  {
  2. private Object target;
  3. public MyInterceptor(Object target) {
  4. this.target = target;
  5. }
  6. @Override
  7. public Object intercept(Object proxy, Method method, Object[] args,
  8. MethodProxy invocation) throws Throwable {
  9. //1.记录日志 2.时间统计开始   3.安全检查
  10. Object retVal = invocation.invoke(target, args);
  11. //4.时间统计结束
  12. return retVal;
  13. }
  14. public static Object proxy(Object target) {
  15. return Enhancer.create(target.getClass(), new MyInterceptor(target));
  16. }
  17. }

编程模型

  1. //proxy 在其上调用方法的代理实例    method拦截的方法    args  拦截的参数
  2. //invocation 用来去调用被代理对象方法的
  3. @Override
  4. public Object intercept(Object proxy, Method method, Object[] args,
  5. MethodProxy invocation) throws Throwable {
  6. //预处理
  7. //前置条件判断
  8. boolean ok = true;
  9. if(!ok) {//不满足条件
  10. throw new RuntimeException("出错了");
  11. }
  12. else {//调用目标对象的某个方法
  13. Object retVal = invocation.invoke(target, args);
  14. }
  15. //后处理
  16. return retVal;
  17. }
优点:能代理接口和类
缺点:使用麻烦,不能代理final类

动态代理本质

本质:对目标对象增强
           最终表现为类(动态创建子类),看手工生成(子类)还是自动生成(子类)
代理限制:
           只能在父类方法被调用之前或之后进行增强(功能的修改),不能在中间进行修改,要想在方法调用中增强,需要ASM(java 字节码生成库)
其他动态代理框架
jboss:javassist (hibernate 3.3中默认为javassist)
                           (hibernate 3.3之前中默认为cglib)


 




 
2.5、AOP解决方案(通用且简单的解决方案)
  1. @Aspect
  2. public class PayEbiAspect {
  3. @Pointcut(value="execution(* pay(..))")
  4. public void pointcut() {}
  5. @Around(value="pointcut()")
  6. public Object around(ProceedingJoinPoint pjp) throws Throwable {
  7. //1.记录日志
  8. //2.时间统计开始
  9. //3.安全检查
  10. Object retVal = pjp.proceed();//调用目标对象的真正方法
  11. //4.时间统计结束
  12. return retVal;
  13. }
  14. }
编程模型
  1. //2 切入点
  2. @Pointcut(value="execution(* *(..))")
  3. public void pointcut() {}
  4. //3 拦截器的interceptor
  5. @Around(value="pointcut()")
  6. public Object around(ProceedingJoinPoint pjp) throws Throwable {
  7. Object retVal=null;
  8. //预处理
  9. //前置条件判断
  10. boolean ok = true;
  11. if(!ok) {//不满足条件
  12. throw new RuntimeException("你没有权限");
  13. }
  14. else {//调用目标对象的某个方法
  15. retVal = pjp.proceed();
  16. }
  17. //后处理
  18. return retVal;
  19. }
缺点:依赖AOP框架 

AOP入门
概念:
n关注点:可以认为是所关注的任何东西,比如上边的支付组件;
n关注点分离:将问题细化为单独部分,即可以理解为不可再分割的组件,如上边的日志组件和支付组件;
n横切关注点:会在多个模块中出现,使用现有的编程方法,横切关注点会横越多个模块,结果是使系统难以设计、理解、实现和演进,如日志组件横切于支付组件。
织入:横切关注点分离后,需要通过某种技术将横切关注点融合到系统中从而完成需要的功能,因此需要织入,织入可能在编译期、加载期、运行期等进行。

nAOP是什么(Aspect   Oriented   Programming)
 AOP是一种编程范式,提供从另一个角度来考虑程序结构以完善面向对象编程(OOP)。
 AOP为开发者提供了一种描述横切关注点的机制,并能够自动将横切关注点织入到面向对象的软件系统中,从而实现了横切关注点的模块化。
 AOP能够将那些与业务无关,却为业务模块所共同调用的逻辑或责任,例如事务处理、日志管理、权限控制等,封装起来,便于减少系统的重复代码,降低模块间的耦合度,并有利于未来的可操作性和可维护性。
nAOP能干什么,也是AOP带来的好处
1:降低模块的耦合度
2:使系统容易扩展
3:设计决定的迟绑定:使用AOP,设计师可以推迟为将来的需求作决定,因为它
可以把这种需求作为独立的方面很容易的实现。
4:更好的代码复用性


 
AOP基本概念
 连接点(Joinpoint):
    表示需要在程序中插入横切关注点的扩展点,连接点可能是类初始化、方法执行、方法调用、字段调用或处理异常等等,Spring只支持方法执行连接点,在AOP中表示为“在哪里做”
切入点(Pointcut):
    选择一组相关连接点的模式,即可以认为连接点的集合,Spring支持perl5正则表达式和AspectJ切入点模式,Spring默认使用AspectJ语法,在AOP中表示为“在哪里做的集合”
增强(Advice):或称为增强
    在连接点上执行的行为,增强提供了在AOP中需要在切入点所选择的连接点处进行扩展现有行为的手段;包括前置增强(before advice)、后置增强 (after advice)、环绕增强 (around advice),在Spring中通过代理模式实现AOP,并通过拦截器模式以环绕连接点的拦截器链织入增强 ;在AOP中表示为“做什么”;
方面/切面(Aspect):
      横切关注点的模块化,比如上边提到的日志组件。可以认为是增强、引入和切入点的组合;在Spring中可以使用Schema和@AspectJ方式进行组织实现;在AOP中表示为“在哪里做和做什么集合”;
目标对象(Target Object):
    需要被织入横切关注点的对象,即该对象是切入点选择的对象,需要被增强的对象,从而也可称为“被增强对象”;由于Spring AOP 通过代理模式实现,从而这个对象永远是被代理对象,在AOP中表示为“对谁做”
AOP代理(AOP Proxy):
    AOP框架使用代理模式创建的对象,从而实现在连接点处插入增强(即应用切面),就是通过代理来对目标对象应用切面。在Spring中,AOP代理可以用JDK动态代理或CGLIB代理实现,而通过拦截器模型应用切面。
织入(Weaving):
    织入是一个过程,是将切面应用到目标对象从而创建出AOP代理对象的过程,织入可以在编译期、类装载期、运行期进行。
引入(inter-type declaration):
    也称为内部类型声明,为已有的类添加额外新的字段或方法,Spring允许引入新的接口(必须对应一个实现)到所有被代理对象(目标对象), 在AOP中表示为“做什么(新增什么)”

AOP的Advice类型
前置增强(Before advice):
    在某连接点之前执行的增强,但这个增强不能阻止连接点前的执行(除非它抛出一个异常)。
后置返回增强(After returning advice):
    在某连接点正常完成后执行的增强:例如,一个方法没有抛出任何异常,正常返回。
后置异常增强(After throwing advice):
    在方法抛出异常退出时执行的增强。
后置最终增强(After (finally) advice):
    当某连接点退出的时候执行的增强(不论是正常返回还是异常退出)。
环绕增强(Around Advice):
    包围一个连接点的增强,如方法调用。这是最强大的一种增强类型。 环绕增强可以在方法调用前后完成自定义的行为。它也会选择是否继续执行连接点或直接返回它们自己的返回值或抛出异常来结束执行。


AOP开发步骤
  类似于IoC/DI容器开发步骤,需要描述哪个连接点需要哪个通用功能(增强

横切关注点的表现有:  

  ·代码纠结/混乱——
当一个模块或代码段同时管理多个关注点时发生这种情况。 

  ·代码分散——当一个关注点分布在许多模块中并且未能很好地局部化和模块化时发生这种情况 。
AOP包括三个清晰的开发步骤:
1:功能横切:找出横切关注点。
2:实现分离:各自独立的实现这些横切关注点所需要完成的功能。
3:功能回贴:在这一步里,方面集成器通过创建一个模块单元—— 方面来指定重组的规则。重组过程——也叫织入或结合—— 则使用这些信息来构建最终系统。

推荐阅读书籍:
AspectJ in Action

AOSD中文版--基于用例的面向方面软件开发


我对AOP的理解的更多相关文章

  1. 对IOC和DI以及AOP的理解

    为了理解Spring的IoC与DI从网上查了很多资料,作为初学者,下面的描述应该是最详细,最易理解的方式了. 首先想说说IoC(Inversion of Control,控制倒转).这是spring的 ...

  2. AOP 怎么理解?

    什么是 AOP ? AOP 为 Aspect Oriented Programming 的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术.AOP 是 OOP ...

  3. Spring AOP深入理解之拦截器调用

    Spring AOP深入理解之拦截器调用 Spring AOP代理对象生成回想 上一篇博客中:深入理解Spring AOP之二代理对象生成介绍了Spring代理对象是怎样生成的,当中重点介绍了JDK动 ...

  4. Spring框架学习(9)AOP技术理解与使用

    内容源自:AOP技术理解与使用 一.什么是AOP? aop技术是面向切面编程思想,作为OOP(面向对象编程)的延续思想添加到企业开发中,用于弥补OOP开发过程中的缺陷而提出的编程思想. AOP底层也是 ...

  5. Spring 的 IOC 和 AOP 的理解

    Spring 的 IOC 和 AOP 的理解: https://www.jianshu.com/p/bf1adc3b75e6 对Spring的核心(AOP和IOC)的理解(大白话) https://w ...

  6. 谈谈对AOP的理解

    Aspect Oriented Programming  面向切面编程.解耦是程序员编码开发过程中一直追求的.AOP也是为了解耦所诞生. 具体思想是:定义一个切面,在切面的纵向定义处理方法,处理完成之 ...

  7. 关于Spring中AOP的理解

    AOP简介[理解][重点] 1.AOP(Aspect Oriented Programing)面向切面/方面编程 2.AOP隶属软件工程的范畴,指导开发人员如何制作开发软件,进行结构设计 3.AOP联 ...

  8. 对Spring IOC和AOP的理解

    控制反转(IOC)是什么?(理解好Ioc的关键是要明确"谁控制谁,控制什么,为何是反转(有反转就应该有正转了),哪些方面反转了") 1.Ioc-Inversion of Contr ...

  9. Spring AOP概念理解

    1.我所知道的aop 初看aop,上来就是一大堆术语,而且还有个拉风的名字,面向切面编程,都说是OOP的一种有益补充等等.一下子让你不知所措,心想着:怪不得很多人都和我说aop多难多难.当我看进去以后 ...

随机推荐

  1. 【Unity Shaders】Shader学习资源和Surface Shader概述

    写在前面 写这篇文章的时候,我断断续续学习Unity Shader半年了,其实还是个门外汉.我也能体会很多童鞋那种想要学好Shader却无从下手的感觉.在这个期间,我找到一些学习Shader的教程以及 ...

  2. 利用并查集+贪心解决 Hdu1232

    畅通工程 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  3. Android性能优化之加快应用启动速度

    应用的启动 启动方式 通常来说,在安卓中应用的启动方式分为两种:冷启动和热启动. 1.冷启动:当启动应用时,后台没有该应用的进程,这时系统会重新创建一个新的进程分配给该应用,这个启动方式就是冷启动. ...

  4. SQL 2012 Always On 为 MSCRMSqlClrLogin SQL 登录名创建非对称密钥时报语法错误

    根据实施手册中的下图的SQL在为MSCRMSqlClrLogin 创建非对称秘钥时报语法错误,具体的错误在就path那,调了多次还是报错,索性就把SQL拆开执行. 先执行这条 SELECT * FRO ...

  5. Mybatis执行ReuseExecutor(五)

    ReuseExecutor顾名思义就是重复使用执行,其定义了一个Map<String, Statement>,将执行的sql作为key,将执行的Statement作为value保存,这样执 ...

  6. 《java入门第一季》之泛型类引入

    首先用一个例子引入泛型类. 我定义一个Object类: package cn.it_03; public class ObjectTool { private Object obj; public O ...

  7. Android进阶(十一)Android系统架构讲解

    如果说一个成功男人的背后会有一个默默支持的女人的话,那么一个优越稳定的平台背后必有一个成熟的系统架构所支撑着.那么,Android的系统架构是怎么样的呢?从下图我们可以从整体上有个大致的了解(图片来源 ...

  8. NSData 与 struct 以及XML的转换。

    在做OC与C++ 混编的时候,我们可能会用到struct 与NSData的相互转换.在这里做一个记录 1.struct转换为NSData 例如如下的struct: struct tagPackageH ...

  9. 【Unity Shaders】Reflecting Your World —— 在Unity3D中创建Cubemaps

    本系列主要参考<Unity Shaders and Effects Cookbook>一书(感谢原书作者),同时会加上一点个人理解或拓展. 这里是本书所有的插图.这里是本书所需的代码和资源 ...

  10. SpriteBuilder中CCMotionStreak提示图片文件找不到

    今天写代码时遇到上述问题,代码如下: player.streak = [CCMotionStreak streakWithFade:3.f minSeg:1 width:30 color:[CCCol ...