이 기사에서는 WeChat 결제 기능을 통합한 WeChat 오픈 플랫폼 모바일 애플리케이션의 예를 설명합니다. 참고할 수 있도록 모든 사람과 공유하세요. 구체적인 분석은 다음과 같습니다.
WechatAppPay 파일 코드는 다음과 같습니다.
네임스페이스 commonservicesWechatPay;
WechatAppPay 클래스는 WechatPayBase를 확장합니다
{
//패키지 매개변수
공개 $패키지 = [];
//비동기 알림 매개변수
공개 $notify = [];
//선불 주문 매개변수 푸시
보호된 $config = [];
//액세스 토큰과 획득 시간을 저장하는 파일
보호된 $file;
//액세스 토큰
보호된 $accessToken;
//액세스 토큰 URL 가져오기
const ACCESS_TOKEN_URL = 'https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s';
//선불주문 제출주소 생성
const POST_ORDER_URL = 'https://api.weixin.qq.com/pay/genprepay?access_token=%s';
공개 함수 __construct()
{
$this->file = __DIR__ '/payAccessToken.txt';
}
/**
* *APP 결제 최종 반환 매개변수 생성
* @예외 발생
* @return 다중 유형:문자열 NULL
*/
공개 함수 createAppPayData()
{
$this->generateConfig();
$prepayid = $this->getPrepayid();
시도해 보세요{
$배열 = [
'appid' => $this->appid,
'appkey' => $this->paySignkey,
'noncestr' =>$this->getRandomStr(),
'패키지' => '서명=WXPay',
'partnerid' =>$this->partnerId,
'선불' => $선불,
'타임스탬프' => (문자열)시간(),
];
$array['sign'] = $this->sha1Sign($array);
unset($array['appkey']);
} catch(예외 $e) {
새로운 예외 발생($e->getMessage());
}
$배열 반환;
}
/**
* * 결제 성공 확인 후 알림 매개변수
* *
* @예외 발생
* @return 부울
*/
공개 함수 verifyNotify()
{
시도해 보세요{
$staySignStr = $this->알림;
unset($staySignStr['sign']);
$sign = $this->signData($staySignStr);
return $this->notify['sign'] === $sign;
} catch(예외 $e) {
새로운 예외 발생($e->getMessage());
}
}
/**
* 마법의 방법, 결제 매개변수 추가
* *
* @param string $name 매개변수 이름
* @param string $value 매개변수 값
*/
공개 함수 __set($name, $value)
{
$this->$name = $value;
}
/**
* 设置액세스 토큰
* @param 문자열 $token
* @예외 발생
* @return 부울
*/
공개 함수 setAccessToken()
{
시도해 보세요{
if(!file_exists($this->file) || !is_file($this->file)) {
$f = fopen($this->file, 'a');
fclose($f);
}
$content = file_get_contents($this->file);
if(!empty($content)) {
$info = json_decode($content, true);
if( time() - $info['getTime'] < 7150 ) {
$this->accessToken = $info['accessToken'];
true를 반환합니다.
}
}
//文件内容为空或액세스 토큰已失效,중신설
$this->outputAccessTokenToFile();
} catch(예외 $e) {
새로운 예외 발생($e->getMessage());
}
true를 반환합니다.
}
/**
* 파일에 액세스 토큰 쓰기
* @예외 발생
* @return 부울
*/
보호된 함수 outputAccessTokenToFile()
{
시도해 보세요{
$f = fopen($this->file, 'wb');
$토큰 = [
'액세스 토큰' => $this->getAccessToken(),
'getTime' => 시간(),
];
무리($f, LOCK_EX);
fwrite($f, json_encode($token));
무리($f, LOCK_UN);
fclose($f);
$this->accessToken = $token['accessToken'];
} catch(예외 $e) {
새로운 예외 발생($e->getMessage());
}
true를 반환합니다.
}
/**
* 取액세스 토큰
*
* @예외 발생
* @return 문자열
*/
보호된 함수 getAccessToken()
{
$url = sprintf(self::ACCESS_TOKEN_URL, $this->appid, $this->appSecret);
$result = json_decode( $this->getUrl($url), true );
if(isset($result['errcode'])) {
새로운 예외 발생("액세스 토큰 가져오기 실패:{$result['errmsg']}");
}
$result['access_token'];
반환
}
/**
* 선불 세션 ID 받기
* *
* @예외 발생
* @return 문자열
*/
보호된 함수 getPrepayid()
{
$data = json_encode($this->config);
$url = sprintf(self::POST_ORDER_URL, $this->accessToken);
$result = json_decode( $this->postUrl($url, $data), true );
if( isset($result['errcode']) && $result['errcode'] != 0 ) {
새로운 예외 발생($result['errmsg']);
}
if( !isset($result['prepayid']) ) {
새로운 예외 발생('선불 ID 가져오기 실패, URL 요청 오류.');
}
$result['prepayid'];
반환
}
/**
* 선불 매개변수 조합
* *
* @예외 발생
*/
보호된 함수 generateConfig()
{
시도해 보세요{
$this->config = [
'앱' => $this->appid,
'추적' => $this->traceid
'noncestr' => $this->getRandomStr(),
'타임스탬프' => 시간(),
'패키지' => $this->generatePackage(),
'sign_method' => $this->sign_method,
];
$this->config['app_signature'] = $this->generateSign();
} catch(예외 $e) {
새로운 예외 발생($e->getMessage());
}
}
/**
* 패키지 필드 생성
* *
* 규칙 생성:
* 1. 부호 signValue
의 값을 생성합니다.
* 2. 패키지 매개변수를 쿼리 문자열에 다시 연결하고 값은 urlencode여야 합니다
* 3. 2에서 생성된 문자열에 sign=signValue를 붙여 최종 패키지 문자열
을 얻습니다.
* *
* * 2단계에서 urlencode 공백은
대신에 인코딩되어야 합니다.
* *
* RFC 1738은 공백을
으로 인코딩합니다.
* RFC 3986은 공백을
으로 인코딩합니다.
* *
* @return 문자열
*/
보호된 함수 generatePackage()
{
$this->package['sign'] = $this->signData($this->패키지);
return http_build_query($this->package, '', '&', PHP_QUERY_RFC3986);
}
/**
* 서명 생성
* *
* @return 문자열
*/
보호된 함수 generateSign()
{
$signArray = [
'앱' => $this->appid,
'앱키' => $this->paySignkey,
'noncestr' => $this->config['noncestr'],
'패키지' => $this->config['패키지'],
'타임스탬프' => $this->config['timestamp'],
'추적' => $this->traceid
];
return $this->sha1Sign($signArray);
}
/**
*시그니처 데이터
* *
* 규칙 생성:
* 1. 사전 정렬, 쿼리 문자열 형식으로 연결, urlencode 필요 없음
* 2. 이전 단계에서 얻은 문자열은 최종적으로 key=paternerKey
로 이어집니다.
* 3. MD5는 문자열을 해시하고 대문자로 변환하여 부호 signValue
의 값을 얻습니다.
* *
* @param array $data 서명할 데이터
* @return 문자열 최종 서명 결과
*/
보호된 함수 signData($data)
{
ksort($data);
$str = $this->arrayToString($data);
$str .= "&key={$this->partnerKey}";
return strtoupper( $this->signMd5($str) );
}
/**
* sha1 서명
*서명규칙
* 1. 사전정렬
* 2. 쿼리 문자열 연결
* 3. sha1 작업
* *
* @param 배열 $arr
* @return 문자열
*/
보호된 함수 sha1Sign($arr)
{
ksort($arr);
return sha1( $this->arrayToString($arr) );
}
}
希望本文所述对大家程序设计有所帮助。