主頁 > 後端開發 > Spring原始碼核心剖析

Spring原始碼核心剖析

2023-06-22 07:38:39 後端開發

前言

SpringAOP作為Spring最核心的能力之一,其重要性不言而喻,然后需要知道的是AOP并不只是Spring特有的功能,而是一種思想,一種通用的功能,而SpringAOP只是在AOP的基礎上將能力集成到SpringIOC中,使其作為bean的一種,從而我們能夠很方便的進行使用,

一、SpringAOP的使用方式

1.1 使用場景

當我們在日常業務開發中,例如有些功能模塊是通用的(日志、權限等),或者我們需要在某些功能前后去做一些增強,例如在某些方法執行后發送一條mq訊息等,

如果我們將這些通用模塊代碼與業務代碼放在一塊,那么每個業務代碼都要寫這些通用模塊,維護成本與耦合情況都十分嚴重,

因此,我們可以將此模塊抽象出來,就有了”切面“的概念,

1.2 常用方式

AOP的使用方式相對比較簡單,首先我們需要完成業務代碼

@Service
public class AopDemo implements AopInterface{

    public Student start(String name) {
        System.out.println("執行業務邏輯代碼.....");
        return new Student(name);
    }

}

業務邏輯比較簡單,接收一個name引數,

接下來我們需要創建其對應的切面

//將該切面加入spring容器
@Service
//宣告該類為一個切面
@Aspect
class AopAspect {

    //宣告要進行代理的方法
    @Pointcut("execution(* com.example.demo.aop.AopInterface.start(..))")
    public void startAspect() {
    }

    //在方法執行之前的邏輯
    @Before(value = "https://www.cnblogs.com/jingdongkeji/archive/2023/06/21/startAspect()")
    public void beforeAspect() {
        System.out.println("業務邏輯前代碼.....");
    }

    //在方法執行之后的邏輯
    @After(value = "https://www.cnblogs.com/jingdongkeji/archive/2023/06/21/startAspect()")
    public void afterAspect() {
        System.out.println("業務邏輯后代碼.....");
    }

    //圍繞方法前后的邏輯
    @Around("startAspect()")
    public Object aroundAspect(ProceedingJoinPoint point) throws Throwable {
        Object[] requestParams = point.getArgs();
        String name = requestParams[0].toString();
        System.out.println("傳入引數:" + name);
        requestParams[0] = "bob";
        return point.proceed(requestParams);
    }

}

可以看到,首先需要我們指明要代理的物件及方法,然后根據需要選擇不同的注解即可實作代理物件,

傳入引數:tom
業務邏輯前代碼.....
執行業務邏輯代碼.....
業務邏輯后代碼.....

二、SpringAOP原始碼決議

2.1 被代理物件的開始initializeBean

根據上面的使用情況,我們知道只需要宣告對應的注解即可,不需要其他額外的配置,然后我們獲得的bean物件就已經是被代理的了,那么我們可以推斷代理物件的程序一定是發生在bean創建的程序的,

我們回顧一下創建bean的流程

  1. 實體化bean
  2. 裝配屬性
  3. 初始化bean

只有第三步初始化bean的時候才會有機會進行代理,

找到對應的代碼位置:

protected Object initializeBean(String beanName, Object bean, @Nullable RootBeanDefinition mbd) {
   Object wrappedBean = bean;
   if (mbd == null || !mbd.isSynthetic()) {
      //前置處理器
      wrappedBean = applyBeanPostProcessorsBeforeInitialization(wrappedBean, beanName);
   }
	 //...
   try {
      //物件的初始化方法
      invokeInitMethods(beanName, wrappedBean, mbd);
   }
   if (mbd == null || !mbd.isSynthetic()) {
      //后置處理器,AOP開始的地方
      wrappedBean = applyBeanPostProcessorsAfterInitialization(wrappedBean, beanName);
   }

   return wrappedBean;
}

2.2 后置處理器applyBeanPostProcessorsAfterInitialization

后置處理器會執行那些實作了后置處理器介面的代碼:

public Object applyBeanPostProcessorsAfterInitialization(Object existingBean, String beanName)
      throws BeansException {

   Object result = existingBean;
   //獲取所有的后置處理器
   for (BeanPostProcessor processor : getBeanPostProcessors()) {
      //實作其要執行的方法
      Object current = processor.postProcessAfterInitialization(result, beanName);
      if (current == null) {
         return result;
      }
      result = current;
   }
   return result;
}

而AOP的后置處理器就是其中的一個: AbstractAutoProxyCreator

其對應的方法為(以下代碼不為同一個類,而是對應的執行順序):

public Object postProcessAfterInitialization(@Nullable Object bean, String beanName) {
   if (bean != null) {
      Object cacheKey = getCacheKey(bean.getClass(), beanName);
      if (this.earlyProxyReferences.remove(cacheKey) != bean) {
         //執行到下面方法
         return wrapIfNecessary(bean, beanName, cacheKey);
      }
   }
   return bean;
}

protected Object wrapIfNecessary(Object bean, String beanName, Object cacheKey) {
		// Create proxy if we have advice.
		Object[] specificInterceptors = getAdvicesAndAdvisorsForBean(bean.getClass(), beanName, null);
		if (specificInterceptors != DO_NOT_PROXY) {
			this.advisedBeans.put(cacheKey, Boolean.TRUE);
      //創建代理物件
			Object proxy = createProxy(
					bean.getClass(), beanName, specificInterceptors, new SingletonTargetSource(bean));
			this.proxyTypes.put(cacheKey, proxy.getClass());
			return proxy;
		}

		this.advisedBeans.put(cacheKey, Boolean.FALSE);
		return bean;
}

protected Object createProxy(Class beanClass, @Nullable String beanName,
			@Nullable Object[] specificInterceptors, TargetSource targetSource) {

		//獲取advisors
		Advisor[] advisors = buildAdvisors(beanName, specificInterceptors);
		proxyFactory.addAdvisors(advisors);
		proxyFactory.setTargetSource(targetSource);
		customizeProxyFactory(proxyFactory);

		proxyFactory.setFrozen(this.freezeProxy);
		if (advisorsPreFiltered()) {
			proxyFactory.setPreFiltered(true);
		}

		// Use original ClassLoader if bean class not locally loaded in overriding class loader
		ClassLoader classLoader = getProxyClassLoader();
		if (classLoader instanceof SmartClassLoader && classLoader != beanClass.getClassLoader()) {
			classLoader = ((SmartClassLoader) classLoader).getOriginalClassLoader();
		}
    //通過代理工廠創建代理物件
		return proxyFactory.getProxy(classLoader);
}

public Object getProxy(@Nullable ClassLoader classLoader) {
    //首先獲取對應的代理
		return createAopProxy().getProxy(classLoader);
}

//該方法根據要被代理的類選擇使用jdk代理還是cglib代理
public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
		if (!NativeDetector.inNativeImage() &&
				(config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config))) {
			Class targetClass = config.getTargetClass();
      //如果被代理的類是一個介面則使用jdk代理
			if (targetClass.isInterface() || Proxy.isProxyClass(targetClass) || ClassUtils.isLambdaClass(targetClass)) 			{
				return new JdkDynamicAopProxy(config);
			}
      //否則使用cglib代理
			return new ObjenesisCglibAopProxy(config);
		}
		else {
      //根據配置選擇強制使用jdk代理
			return new JdkDynamicAopProxy(config);
		}
}

我們知道,代理方式有jdk動態代理與cglib動態代理兩種方式,而我們一個bean使用那種代理方式則由上述的方法決定,

至此,我們已經確定了使用那種代理方式獲取代理物件,

2.3 獲取代理物件

從上文中,我們已經確定了選用何種方式構建代理物件,接下來就是通過不同的方式是如何獲取代理物件的,

看懂本章需要實作了解jdk動態代理或者cglib動態代理的方式,

2.3.1 JDK代理

首先在獲取代理物件時選擇 JdkDynamicAopProxy

public Object getProxy(@Nullable ClassLoader classLoader) {
   if (logger.isTraceEnabled()) {
      logger.trace("Creating JDK dynamic proxy: " + this.advised.getTargetSource());
   }
   //這里通過反射創建代理物件
   return Proxy.newProxyInstance(classLoader, this.proxiedInterfaces, this);
}

當被代理物件執行被代理的方法時,會進入到此方法,(jdk動態代理的概念)

JDK通過反射創建物件,效率上來說相對低一些,

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

		try {
			// 獲取被代理物件的所有切入點
			List chain = this.advised.getInterceptorsAndDynamicInterceptionAdvice(method, targetClass);

			// 如果呼叫鏈路為空說明沒有需要執行的切入點,直接執行對應的方法即可
			if (chain.isEmpty()) {
				// We can skip creating a MethodInvocation: just invoke the target directly
				// Note that the final invoker must be an InvokerInterceptor so we know it does
				// nothing but a reflective operation on the target, and no hot swapping or fancy proxying.
				Object[] argsToUse = AopProxyUtils.adaptArgumentsIfNecessary(method, args);
				retVal = AopUtils.invokeJoinpointUsingReflection(target, method, argsToUse);
			}
			else {
				// 如果有切入點的話則按照切入點順序開始執行
				MethodInvocation invocation =
						new ReflectiveMethodInvocation(proxy, target, method, args, targetClass, chain);
				// Proceed to the joinpoint through the interceptor chain.
				retVal = invocation.proceed();
			}
			
			return retVal;
		}
}

invocation.proceed();這個方法就是通過遞回的方式執行所有的呼叫鏈路,

public Object proceed() throws Throwable {
   // We start with an index of -1 and increment early.
   if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {
      return invokeJoinpoint();
   }

   Object interceptorOrInterceptionAdvice =
         this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex);
   if (interceptorOrInterceptionAdvice instanceof InterceptorAndDynamicMethodMatcher) {
      InterceptorAndDynamicMethodMatcher dm =
            (InterceptorAndDynamicMethodMatcher) interceptorOrInterceptionAdvice;
      Class targetClass = (this.targetClass != null ? this.targetClass : this.method.getDeclaringClass());
      if (dm.methodMatcher.matches(this.method, targetClass, this.arguments)) {
         return dm.interceptor.invoke(this);
      }
      else {
         // 繼續執行
         return proceed();
      }
   }
   else {
      // 如果呼叫鏈路還持續的話,下一個方法仍會呼叫proceed()
      return ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this);
   }
}

2.3.2 cglib代理

public Object getProxy(@Nullable ClassLoader classLoader) {

   try {
      //配置CGLIB Enhancer...
      Enhancer enhancer = createEnhancer();
      if (classLoader != null) {
         enhancer.setClassLoader(classLoader);
         if (classLoader instanceof SmartClassLoader &&
               ((SmartClassLoader) classLoader).isClassReloadable(proxySuperClass)) {
            enhancer.setUseCache(false);
         }
      }
      enhancer.setSuperclass(proxySuperClass);
      enhancer.setInterfaces(AopProxyUtils.completeProxiedInterfaces(this.advised));
      enhancer.setNamingPolicy(SpringNamingPolicy.INSTANCE);
      enhancer.setStrategy(new ClassLoaderAwareGeneratorStrategy(classLoader));

      //1.獲取回呼函式,對于代理類上所有方法的呼叫,都會呼叫CallBack,而Callback則需要實作intercept()方法
      Callback[] callbacks = getCallbacks(rootClass);
      Class[] types = new Class[callbacks.length];
      for (int x = 0; x < types.length; x++) {
         types[x] = callbacks[x].getClass();
      }
      // fixedInterceptorMap only populated at this point, after getCallbacks call above
      enhancer.setCallbackFilter(new ProxyCallbackFilter(
            this.advised.getConfigurationOnlyCopy(), this.fixedInterceptorMap, this.fixedInterceptorOffset));
      enhancer.setCallbackTypes(types);

      //2.創建代理物件
      return createProxyClassAndInstance(enhancer, callbacks);
   }
   catch (CodeGenerationException | IllegalArgumentException ex) {
      throw new AopConfigException("Could not generate CGLIB subclass of " + this.advised.getTargetClass() +
            ": Common causes of this problem include using a final class or a non-visible class",
            ex);
   }
   catch (Throwable ex) {
      // TargetSource.getTarget() failed
      throw new AopConfigException("Unexpected AOP exception", ex);
   }
}

可以看到我們在創建代理物件前會先獲取代理物件的所有回呼函式:

首先可以看到我們一共有7個回呼方法,其中第一個為AOP相關的方法,其他的為spring相關,

在第一個對調物件中持有的 advised 物件中有 advisors 屬性,就是對應我們的代理類中四個切片,@Before等等,

然后我們看一下 createProxyClassAndInstance()都做了什么,

//CglibAopProxy類的創建代理物件方法
protected Object createProxyClassAndInstance(Enhancer enhancer, Callback[] callbacks) {
   enhancer.setInterceptDuringConstruction(false);
   enhancer.setCallbacks(callbacks);
   return (this.constructorArgs != null && this.constructorArgTypes != null ?
         enhancer.create(this.constructorArgTypes, this.constructorArgs) :
         enhancer.create());
}

//ObjenesisCglibAopProxy繼承了CglibAopProxy類,并覆寫了其方法
protected Object createProxyClassAndInstance(Enhancer enhancer, Callback[] callbacks) {
		Class proxyClass = enhancer.createClass();
		Object proxyInstance = null;

    //1.嘗試使用objenesis創建物件
		if (objenesis.isWorthTrying()) {
			try {
				proxyInstance = objenesis.newInstance(proxyClass, enhancer.getUseCache());
			}
			catch (Throwable ex) {
				logger.debug("Unable to instantiate proxy using Objenesis, " +
						"falling back to regular proxy construction", ex);
			}
		}

    //2.根據commit的提交記錄發現,objenesis有可能創建物件失敗,如果失敗的話則選用放射的方式創建物件
		if (proxyInstance == null) {
			// Regular instantiation via default constructor...
			try {
				Constructor ctor = (this.constructorArgs != null ?
						proxyClass.getDeclaredConstructor(this.constructorArgTypes) :
						proxyClass.getDeclaredConstructor());
				ReflectionUtils.makeAccessible(ctor);
				proxyInstance = (this.constructorArgs != null ?
						ctor.newInstance(this.constructorArgs) : ctor.newInstance());
			}
			catch (Throwable ex) {
				throw new AopConfigException("Unable to instantiate proxy using Objenesis, " +
						"and regular proxy instantiation via default constructor fails as well", ex);
			}
		}

    //
		((Factory) proxyInstance).setCallbacks(callbacks);
		return proxyInstance;
	}

2.3.3 cglib

此處有個遇到的問題,當我在debug的時候,發現怎么都進不去 createProxyClassAndInstance(),百思不得其解,然后看到IDEA旁邊有一個向下的箭頭,代表該方法可能其子類被覆寫了,然后在其子類處打斷點果然發現是其子類的實作,

此處在2.2中也可看到:

可以看到回傳的是其子類的物件,而不是CglibAopProxy本身的物件,

作者:京東科技 韓國凱

來源:京東云開發者社區

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/555742.html

標籤:其他

上一篇:【python基礎】類-類屬性

下一篇:返回列表

標籤雲
其他(161430) Python(38244) JavaScript(25512) Java(18251) C(15238) 區塊鏈(8271) C#(7972) AI(7469) 爪哇(7425) MySQL(7260) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5875) 数组(5741) R(5409) Linux(5347) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4606) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2436) ASP.NET(2404) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1984) HtmlCss(1970) 功能(1967) Web開發(1951) C++(1942) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1881) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Spring原始碼核心剖析

    SpringAOP作為Spring最核心的能力之一,其重要性不言而喻。然后需要知道的是AOP并不只是Spring特有的功能,而是一種思想,一種通用的功能。而SpringAOP只是在AOP的基礎上將能力集成到SpringIOC中,使其作為bean的一種,從而我們能夠很方便的進行使用。 ......

    uj5u.com 2023-06-22 07:38:39 more
  • 【python基礎】類-類屬性

    在初始類中,我們介紹了如何訪問類屬性,除了訪問類屬性外還有其他操作類屬性的情況,我們將在這里做詳細介紹: # 1.給類屬性指定默認值 類中的每個屬性都必須有初始值,哪怕這個值是0或者空字串。在有些情況下,如設定默認值時,在方法\_\_init\_\_方法內指定這種初始值是可行的,如果對某個屬性這樣 ......

    uj5u.com 2023-06-22 07:38:31 more
  • SLF4J門面日志框架原始碼探索

    我們通過代碼入手,層層加碼,直觀感受SLF4J列印日志,并跟蹤代碼追本溯源。主要了解,SLF4J是如何作為門面和其他日志框架進行解耦。 ......

    uj5u.com 2023-06-22 07:38:23 more
  • Rust語言 - 介面設計的建議之受約束(Constrained)

    # Rust語言 - 介面設計的建議之受約束(Constrained) - [Rust API 指南 GitHub](https://github.com/rust-lang/api-guidelines):https://github.com/rust-lang/api-guidelines - ......

    uj5u.com 2023-06-22 07:38:19 more
  • Scala泛型

    # 泛型的定義 ```Scala object _11_泛型 { def main(args: Array[String]): Unit = { //[A] 這個代表的就是泛型 ==》 在創建物件的時候,可以指定需要傳進去的型別 //作用就是在創建物件的時候,可以對傳進去的引數一個約束,當設定泛型位 ......

    uj5u.com 2023-06-22 07:37:55 more
  • celery筆記五之訊息佇列的介紹

    > 本文首發于公眾號:Hunter后端 > 原文鏈接:[celery筆記五之訊息佇列的介紹](https://mp.weixin.qq.com/s/fw7b1Gha0XpTYuCg3aZcWA) 前面我們介紹過 task 的處理方式,將 task 發送到佇列 queue,然后 worker 從 qu ......

    uj5u.com 2023-06-22 07:31:20 more
  • java~位元組碼操作Javassist

    Javassist是一個開源的Java位元組碼操作庫,它提供了一組簡單而強大的API,用于在運行時修改和生成Java位元組碼。Javassist的名稱是"Java Programming Assistant"的縮寫,它的目標是簡化對位元組碼的操作,使開發人員能夠更輕松地實作動態代碼生成和修改。 Javas ......

    uj5u.com 2023-06-21 09:03:36 more
  • java~位元組碼操作ASM

    ASM(全稱為"Objectweb ASM")是一個用于分析和轉換Java位元組碼的框架。它允許您以程式化的方式讀取、修改和生成Java類檔案,而無需直接操作Java源代碼。ASM提供了強大而靈活的工具,使您能夠對位元組碼進行細粒度的操作,包括修改現有類、生成新的類以及在類加載時對位元組碼進行增強。 AS ......

    uj5u.com 2023-06-21 09:03:30 more
  • java~位元組碼操作ASM

    ASM(全稱為"Objectweb ASM")是一個用于分析和轉換Java位元組碼的框架。它允許您以程式化的方式讀取、修改和生成Java類檔案,而無需直接操作Java源代碼。ASM提供了強大而靈活的工具,使您能夠對位元組碼進行細粒度的操作,包括修改現有類、生成新的類以及在類加載時對位元組碼進行增強。 AS ......

    uj5u.com 2023-06-21 08:57:18 more
  • 網站怎么接入微信掃碼支付?

    # 第01章-準備作業 ## 1、微信支付產品介紹 參考資料:[產品中心 - 微信支付商戶平臺 (qq.com)](https://pay.weixin.qq.com/static/product/product_index.shtml#payment_product) 付款碼支付、JSAPI支付、 ......

    uj5u.com 2023-06-21 07:55:09 more