목차
回复内容:
백엔드 개발 PHP 튜토리얼 mysql - PHP高并发下单用事务可以解决吗?

mysql - PHP高并发下单用事务可以解决吗?

Jul 06, 2016 pm 01:53 PM
mysql php thinkphp

一个下单的小示例(上代码,没加事务的时候):

<code>class IndexController extends Controller {

    public function index(){
        $stock = M('stock');
        $log = M('log');
        $condition['id'] = 1;
        if($stock->where($condition)->getField('stock_left') > 0) {
            $stock->where($condition)->setDec("stock_left");
            $data['op'] = 1;
            $log->add($data);
        } else {
            echo "已经没剩余了";
        }
    }
}</code>
로그인 후 복사
로그인 후 복사

库存默认有100个:
mysql - PHP高并发下单用事务可以解决吗?

日志表:
mysql - PHP高并发下单用事务可以解决吗?

Apache ab工具并发一下:
ab -n 1200 -c 1200 -w http://localhost/queue/index.php >> D:/1.html

结果出并发问题(很自然的):
mysql - PHP高并发下单用事务可以解决吗?

然后加了事务控制之后:

<code>class IndexController extends Controller {

    public function index(){
        $stock = M('stock');
        $log = M('log');
        $condition['id'] = 1;
        M()->startTrans();
        if($stock->where($condition)->getField('stock_left') > 0) {
            $res1 = $stock->where($condition)->setDec("stock_left");
            $data['op'] = 1;
            $res2 = $log->add($data);
            if($res1 !== false && $res2) {
                M()->commit();
            } else {
                M()->rollback();
            }
        } else {
            echo "已经没剩余了";
        }
    }
}</code>
로그인 후 복사
로그인 후 복사

再测试并发一下:
ab -n 1200 -c 1200 -w http://localhost/queue/index.php >> D:/1.html
结果呢,输出的结果没问题(但是真的解决了并发问题吗?):
mysql - PHP高并发下单用事务可以解决吗?

很多人说了用Redis队列来做,具体实施我还是有点不太清楚,请大家帮忙

回复内容:

一个下单的小示例(上代码,没加事务的时候):

<code>class IndexController extends Controller {

    public function index(){
        $stock = M('stock');
        $log = M('log');
        $condition['id'] = 1;
        if($stock->where($condition)->getField('stock_left') > 0) {
            $stock->where($condition)->setDec("stock_left");
            $data['op'] = 1;
            $log->add($data);
        } else {
            echo "已经没剩余了";
        }
    }
}</code>
로그인 후 복사
로그인 후 복사

库存默认有100个:
mysql - PHP高并发下单用事务可以解决吗?

日志表:
mysql - PHP高并发下单用事务可以解决吗?

Apache ab工具并发一下:
ab -n 1200 -c 1200 -w http://localhost/queue/index.php >> D:/1.html

结果出并发问题(很自然的):
mysql - PHP高并发下单用事务可以解决吗?

然后加了事务控制之后:

<code>class IndexController extends Controller {

    public function index(){
        $stock = M('stock');
        $log = M('log');
        $condition['id'] = 1;
        M()->startTrans();
        if($stock->where($condition)->getField('stock_left') > 0) {
            $res1 = $stock->where($condition)->setDec("stock_left");
            $data['op'] = 1;
            $res2 = $log->add($data);
            if($res1 !== false && $res2) {
                M()->commit();
            } else {
                M()->rollback();
            }
        } else {
            echo "已经没剩余了";
        }
    }
}</code>
로그인 후 복사
로그인 후 복사

再测试并发一下:
ab -n 1200 -c 1200 -w http://localhost/queue/index.php >> D:/1.html
结果呢,输出的结果没问题(但是真的解决了并发问题吗?):
mysql - PHP高并发下单用事务可以解决吗?

很多人说了用Redis队列来做,具体实施我还是有点不太清楚,请大家帮忙

事务跟并发没有任何关系。你使用事务只能保证这一段逻辑成功或者失败,而不能保证并发时能控制住你的程序逻辑。
对并发进行控制还是需要锁 来解决,比如楼上有提到的mysql 实现乐观锁。
如 upadte table set a = a - 1 where a = 5; 只有在a=5的情况下这个update才会真正修改数据,使用这种方法是可以保证如果要修改的数据版本跟你预想中的不同,就不进行操作,通过影响行数来判断是否有修改,然后继续下面的操作或者退出。

还有就是使用排他锁,如果单机可以直接使用flock 来达到阻塞锁的目的。
或者redis和memcache来实现锁。
请求比较多的的情况下推荐使用redis,memcache来进行锁操作,或者考虑用消息队列来处理并发的情况。

用4000并发测试了下,还是会出现问题的:

mysql - PHP高并发下单用事务可以解决吗?

mysql - PHP高并发下单用事务可以解决吗?

用事务吧,PS用事务的时候,请务必这样操作

SELECT * FROM TABLE WHERE A=X LIMIT 1 FOR UPDATE;

务必使用FOR UPDATE;

不用 for update 一定会出现负数的情况。

你也可以给数据表加锁也行(InnoDB引擎就用行锁;MyISAM引擎就用表锁)

最核心还是在数据库,你需要做一个悲观锁或者乐观锁。
表面看你是解决了问题,其实特殊情况下还是会出现超卖的。

事务可以解决,统一提交即可,但是一般解决这个问题还得靠锁。
如果不用锁,就得用队列,就是排队插入,把异步改成同步,这样是最保险的。

可以参考我这个答案。

https://segmentfault.com/q/1010000005105041/a-1020000005106490

队列的方法可以是,一个商品库存(也可以所有商品一起,跑一个下单队列)在后台有一个脚本在跑,然后把请求变成串行。这个方案会被推崇是因为可控性,我们可以根据系统需要控制处理的频率。

缓存的做法是,定时将商品库存更新到缓存里面去,利用缓存的原子读写,对缓存里的库存进行自减操作,如果自减后大于零,就可以走后面的下单流程(下单流程仍然需要完整的事务加锁来保证一致性),缓存的目的在于,避免流量冲击,只有有效流量进入db。

把第一个例子中的$condition['id'] = 1;换成"id=1 and stock_left > 0"的等效条件就解决问题了,不需要事务,事务在这个时候起不到什么作用。后面的逻辑当然也要相应调整,因为setDec肯定成功,但是不一定真有记录被修改了,所以伪码示例:

<code>$sql = "update table set num = num - 1 where num > 0";
$updatedRows = get_updated_rows($db->exec($sql));
if ($updatedRows > 0) {
    //成功
} else {
    //失败
}</code>
로그인 후 복사
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

PHP와 Python : 다른 패러다임이 설명되었습니다 PHP와 Python : 다른 패러다임이 설명되었습니다 Apr 18, 2025 am 12:26 AM

PHP는 주로 절차 적 프로그래밍이지만 객체 지향 프로그래밍 (OOP)도 지원합니다. Python은 OOP, 기능 및 절차 프로그래밍을 포함한 다양한 패러다임을 지원합니다. PHP는 웹 개발에 적합하며 Python은 데이터 분석 및 기계 학습과 같은 다양한 응용 프로그램에 적합합니다.

PHP와 Python 중에서 선택 : 가이드 PHP와 Python 중에서 선택 : 가이드 Apr 18, 2025 am 12:24 AM

PHP는 웹 개발 및 빠른 프로토 타이핑에 적합하며 Python은 데이터 과학 및 기계 학습에 적합합니다. 1.PHP는 간단한 구문과 함께 동적 웹 개발에 사용되며 빠른 개발에 적합합니다. 2. Python은 간결한 구문을 가지고 있으며 여러 분야에 적합하며 강력한 라이브러리 생태계가 있습니다.

LARAVEL 소개 예 LARAVEL 소개 예 Apr 18, 2025 pm 12:45 PM

Laravel은 웹 응용 프로그램을 쉽게 구축하기위한 PHP 프레임 워크입니다. 설치 : Composer를 사용하여 전 세계적으로 Laravel CLI를 설치하고 프로젝트 디렉토리에서 응용 프로그램을 작성하는 등 다양한 기능을 제공합니다. 라우팅 : Routes/Web.php에서 URL과 핸들러 간의 관계를 정의하십시오. 보기 : 리소스/뷰에서보기를 작성하여 응용 프로그램의 인터페이스를 렌더링합니다. 데이터베이스 통합 : MySQL과 같은 데이터베이스와 상자 외 통합을 제공하고 마이그레이션을 사용하여 테이블을 작성하고 수정합니다. 모델 및 컨트롤러 : 모델은 데이터베이스 엔티티를 나타내고 컨트롤러는 HTTP 요청을 처리합니다.

PHP와 Python : 그들의 역사에 깊은 다이빙 PHP와 Python : 그들의 역사에 깊은 다이빙 Apr 18, 2025 am 12:25 AM

PHP는 1994 년에 시작되었으며 Rasmuslerdorf에 의해 개발되었습니다. 원래 웹 사이트 방문자를 추적하는 데 사용되었으며 점차 서버 측 스크립팅 언어로 진화했으며 웹 개발에 널리 사용되었습니다. Python은 1980 년대 후반 Guidovan Rossum에 의해 개발되었으며 1991 년에 처음 출시되었습니다. 코드 가독성과 단순성을 강조하며 과학 컴퓨팅, 데이터 분석 및 기타 분야에 적합합니다.

PHP의 영향 : 웹 개발 및 그 이상 PHP의 영향 : 웹 개발 및 그 이상 Apr 18, 2025 am 12:10 AM

phphassignificallyimpactedwebdevelopmentandextendsbeyondit

데이터베이스 연결 문제 해결 : Minii/DB 라이브러리 사용 실질적인 사례 데이터베이스 연결 문제 해결 : Minii/DB 라이브러리 사용 실질적인 사례 Apr 18, 2025 am 07:09 AM

작은 응용 프로그램을 개발할 때 까다로운 문제가 발생했습니다. 가벼운 데이터베이스 운영 라이브러리를 신속하게 통합해야합니다. 여러 라이브러리를 시도한 후에는 기능이 너무 많거나 호환되지 않는다는 것을 알았습니다. 결국, 나는 내 문제를 완벽하게 해결하는 YII2를 기반으로 단순화 된 버전 인 Minii/DB를 발견했습니다.

Laravel 프레임 워크 설치 방법 Laravel 프레임 워크 설치 방법 Apr 18, 2025 pm 12:54 PM

기사 요약 :이 기사는 Laravel 프레임 워크를 쉽게 설치하는 방법에 대한 독자들을 안내하기위한 자세한 단계별 지침을 제공합니다. Laravel은 웹 애플리케이션의 개발 프로세스를 가속화하는 강력한 PHP 프레임 워크입니다. 이 자습서는 시스템 요구 사항에서 데이터베이스 구성 및 라우팅 설정에 이르기까지 설치 프로세스를 다룹니다. 이러한 단계를 수행함으로써 독자들은 라벨 프로젝트를위한 탄탄한 토대를 빠르고 효율적으로 놓을 수 있습니다.

Laravel과 ThinkPhp의 차이점 Laravel과 ThinkPhp의 차이점 Apr 18, 2025 pm 01:09 PM

Laravel과 ThinkPHP는 모두 인기있는 PHP 프레임 워크이며 개발에 고유 한 장점과 단점이 있습니다. 이 기사는 두 가지 깊이를 비교하여 건축, 기능 및 성능 차이를 강조하여 개발자가 특정 프로젝트 요구에 따라 정보에 입각 한 선택을 할 수 있도록 도와줍니다.

See all articles