好得很程序员自学网

<tfoot draggable='sEl'></tfoot>

SpringBoot项目中接口防刷的完整代码

一、自定义注解

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

import java.lang.annotation.Retention;

import java.lang.annotation.Target;

 

import static java.lang.annotation.ElementType.METHOD;

import static java.lang.annotation.RetentionPolicy.RUNTIME;

 

/**

  * @author Yang

  * @version 1.0

  * @date 2021/2/22 10:28

  */

@Retention (RUNTIME)

@Target (METHOD)

public @interface AccessLimit {

 

     int seconds();

 

     int maxCount();

 

     boolean needLogin() default true ;

 

}

二、定义拦截器

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

import com.alibaba.fastjson.JSON;

import com.mengxiangnongfu.payment.annotation.AccessLimit;

import com.mengxiangnongfu.payment测试数据mons.RedisUtil;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.context.annotation.Configuration;

import org.springframework.stereotype.Component;

import org.springframework.web.method.HandlerMethod;

import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;

 

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import java.io.OutputStream;

 

/**

  * @author Yang

  * @version 1.0

  * @date 2021/2/22 10:29

  */

@Component

public class FangshuaInterceptor extends HandlerInterceptorAdapter {

 

     @Autowired

     private RedisUtil redisUtil;

 

     @Override

     public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

         //判断请求是否属于方法的请求

         if (handler instanceof HandlerMethod) {

             HandlerMethod hm = (HandlerMethod) handler;

             //获取方法中的注解,看是否有该注解

             AccessLimit accessLimit = hm.getMethodAnnotation(AccessLimit. class );

             if (accessLimit == null ) {

                 return true ;

             }

             int seconds = accessLimit.seconds();

             int maxCount = accessLimit.maxCount();

             boolean login = accessLimit.needLogin();

             String key = "1" ;

             //如果需要登录

             if (login) {

                 //获取登录的session进行判断

                 //.....

                 key += "" + "1" ;  //这里假设用户是1,项目中是动态获取的userId

             }

 

             //从redis中获取用户访问的次数

             Integer count = (Integer) redisUtil.get(key);

             if (count == null ) {

                 //第一次访问

                 redisUtil.set(key, 1 , seconds);

             } else if (count < maxCount) {

                 //加1

                 redisUtil.incr(key, 1 );

             } else {

                 //超出访问次数

                 render(response, "请求过于频繁~请稍后再试~" ); //这里的CodeMsg是一个返回参数

                 return false ;

             }

         }

 

         return true ;

 

     }

 

     private void render(HttpServletResponse response, String cm) throws Exception {

         response.setContentType( "application/json;charset=UTF-8" );

         OutputStream out = response.getOutputStream();

         String str = JSON.toJSONString(cm);

         out.write(str.getBytes( "UTF-8" ));

         out.flush();

         out.close();

     }

 

}

三、Redis工具类

?

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

84

85

86

87

88

89

90

91

92

93

94

95

96

97

98

99

100

101

102

103

104

105

106

107

108

109

110

111

112

113

114

115

116

117

118

119

120

121

122

123

124

125

126

127

128

129

130

131

132

133

134

135

136

137

138

139

140

141

142

143

144

145

146

147

148

149

150

151

152

153

154

155

156

157

158

159

160

161

162

163

164

165

166

167

168

169

170

171

172

173

174

175

176

177

178

179

180

181

182

183

184

185

186

187

188

189

190

191

192

193

194

195

196

197

198

199

200

201

202

203

204

205

206

207

208

209

210

211

212

213

214

215

216

217

218

219

220

221

222

223

224

225

226

227

228

229

230

231

232

233

234

235

236

237

238

239

240

241

242

243

244

245

246

247

248

249

250

251

252

253

254

255

256

257

258

259

260

261

262

263

264

265

266

267

268

269

270

271

272

273

274

275

276

277

278

279

280

281

282

283

284

285

286

287

288

289

290

291

292

293

294

295

296

297

298

299

300

301

302

303

304

305

306

307

308

309

310

311

312

313

314

315

316

317

318

319

320

321

322

323

324

325

326

327

328

329

330

331

332

333

334

335

336

337

338

339

340

341

342

343

344

345

346

347

348

349

350

351

352

353

354

355

356

357

358

359

360

361

362

363

364

365

366

367

368

369

370

371

372

373

374

375

376

377

378

379

380

381

382

383

384

385

386

387

388

389

390

391

392

393

394

395

396

397

398

399

400

401

402

403

404

405

406

407

408

409

410

411

412

413

414

415

416

417

418

419

420

421

422

423

424

425

426

427

428

429

430

431

432

433

434

435

436

437

438

439

440

441

442

443

444

445

446

447

448

449

450

451

452

453

454

455

456

457

458

459

460

461

462

463

464

465

466

467

468

469

470

471

472

473

474

475

476

477

478

479

480

481

482

483

484

485

486

487

488

489

490

491

492

493

494

495

496

497

498

499

500

501

502

503

504

505

506

507

508

509

510

511

512

513

514

515

516

517

518

519

520

521

522

523

524

525

526

527

528

529

530

531

532

533

534

535

536

537

538

539

540

541

542

543

544

545

546

547

548

549

550

551

552

553

554

555

556

557

558

559

560

561

562

563

564

565

566

567

568

569

570

571

572

573

574

575

576

577

578

579

580

581

582

583

584

585

586

587

588

589

590

591

592

593

594

595

596

597

598

599

600

601

602

603

604

605

606

607

608

609

610

611

612

613

614

615

616

617

618

619

620

621

622

623

624

625

626

627

628

629

630

631

632

633

634

635

636

637

638

639

640

641

642

643

644

645

646

647

648

649

650

651

652

653

654

655

656

657

658

659

660

661

662

663

664

665

666

667

668

669

670

671

import lombok.extern.slf4j.Slf4j;

import org.springframework.beans.factory.annotation.Autowired;

import org.springframework.data.redis.core.RedisTemplate;

import org.springframework.stereotype.Component;

import org.springframework.util.CollectionUtils;

 

import java.util.List;

import java.util.Map;

import java.util.Set;

import java.util.concurrent.TimeUnit;

 

/**

  * @author Yang

  * @version 1.0

  * @date 2020/11/29 17:06

  */

@Component

@Slf4j

public class RedisUtil {

 

     @Autowired

     RedisTemplate redisTemplate;

 

     // =============================common============================

 

     /**

      * 指定缓存失效时间

      *

      * @param key  键

      * @param time 时间(秒)

      * @return

      */

     public boolean expire(String key, long time) {

         try {

             if (time > 0 ) {

                 redisTemplate.expire(key, time, TimeUnit.SECONDS);

             }

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 根据key 获取过期时间

      *

      * @param key 键 不能为null

      * @return 时间(秒) 返回0代表为永久有效

      */

     public long getExpire(String key) {

         return redisTemplate.getExpire(key, TimeUnit.SECONDS);

     }

 

     /**

      * 判断key是否存在

      *

      * @param key 键

      * @return true 存在 false不存在

      */

     public boolean hasKey(String key) {

         try {

             return redisTemplate.hasKey(key);

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 删除缓存

      *

      * @param key 可以传一个值 或多个

      */

     @SuppressWarnings ( "unchecked" )

     public void del(String... key) {

         if (key != null && key.length > 0 ) {

             if (key.length == 1 ) {

                 redisTemplate.delete(key[ 0 ]);

             } else {

                 redisTemplate.delete(CollectionUtils.arrayToList(key));

             }

         }

     }

 

     // ============================String=============================

 

     /**

      * 普通缓存获取

      *

      * @param key 键

      * @return 值

      */

     public Object get(String key) {

         return key == null ? null : redisTemplate.opsForValue().get(key);

     }

 

     /**

      * 普通缓存放入

      *

      * @param key   键

      * @param value 值

      * @return true成功 false失败

      */

     public boolean set(String key, Object value) {

         try {

             redisTemplate.opsForValue().set(key, value);

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

 

     }

 

     /**

      * 普通缓存放入并设置时间

      *

      * @param key   键

      * @param value 值

      * @param time  时间(秒) time要大于0 如果time小于等于0 将设置无限期

      * @return true成功 false 失败

      */

     public boolean set(String key, Object value, long time) {

         try {

             if (time > 0 ) {

                 redisTemplate.opsForValue().set(key, value, time, TimeUnit.SECONDS);

             } else {

                 set(key, value);

             }

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 递增 适用场景: https://blog.csdn.net/y_y_y_k_k_k_k/article/details/79218254 高并发生成订单号,秒杀类的业务逻辑等。。

      *

      * @param key   键

      * @param delta 要增加几(大于0)

      * @return

      */

     public long incr(String key, long delta) {

         if (delta < 0 ) {

             throw new RuntimeException( "递增因子必须大于0" );

         }

         return redisTemplate.opsForValue().increment(key, delta);

     }

 

     /**

      * 递减

      *

      * @param key   键

      * @param delta 要减少几(小于0)

      * @return

      */

     public long decr(String key, long delta) {

         if (delta < 0 ) {

             throw new RuntimeException( "递减因子必须大于0" );

         }

         return redisTemplate.opsForValue().increment(key, -delta);

     }

 

     // ================================Map=================================

 

     /**

      * HashGet

      *

      * @param key  键 不能为null

      * @param item 项 不能为null

      * @return 值

      */

     public Object hget(String key, String item) {

         return redisTemplate.opsForHash().get(key, item);

     }

 

     /**

      * 获取hashKey对应的所有键值

      *

      * @param key 键

      * @return 对应的多个键值

      */

     public Map<Object, Object> hmget(String key) {

         return redisTemplate.opsForHash().entries(key);

     }

 

     /**

      * HashSet

      *

      * @param key 键

      * @param map 对应多个键值

      * @return true 成功 false 失败

      */

     public boolean hmset(String key, Map<String, Object> map) {

         try {

             redisTemplate.opsForHash().putAll(key, map);

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * HashSet 并设置时间

      *

      * @param key  键

      * @param map  对应多个键值

      * @param time 时间(秒)

      * @return true成功 false失败

      */

     public boolean hmset(String key, Map<String, Object> map, long time) {

         try {

             redisTemplate.opsForHash().putAll(key, map);

             if (time > 0 ) {

                 expire(key, time);

             }

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 向一张hash表中放入数据,如果不存在将创建

      *

      * @param key   键

      * @param item  项

      * @param value 值

      * @return true 成功 false失败

      */

     public boolean hset(String key, String item, Object value) {

         try {

             redisTemplate.opsForHash().put(key, item, value);

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 向一张hash表中放入数据,如果不存在将创建

      *

      * @param key   键

      * @param item  项

      * @param value 值

      * @param time  时间(秒) 注意:如果已存在的hash表有时间,这里将会替换原有的时间

      * @return true 成功 false失败

      */

     public boolean hset(String key, String item, Object value, long time) {

         try {

             redisTemplate.opsForHash().put(key, item, value);

             if (time > 0 ) {

                 expire(key, time);

             }

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 删除hash表中的值

      *

      * @param key  键 不能为null

      * @param item 项 可以使多个 不能为null

      */

     public void hdel(String key, Object... item) {

         redisTemplate.opsForHash().delete(key, item);

     }

 

     /**

      * 判断hash表中是否有该项的值

      *

      * @param key  键 不能为null

      * @param item 项 不能为null

      * @return true 存在 false不存在

      */

     public boolean hHasKey(String key, String item) {

         return redisTemplate.opsForHash().hasKey(key, item);

     }

 

     /**

      * hash递增 如果不存在,就会创建一个 并把新增后的值返回

      *

      * @param key  键

      * @param item 项

      * @param by   要增加几(大于0)

      * @return

      */

     public double hincr(String key, String item, double by) {

         return redisTemplate.opsForHash().increment(key, item, by);

     }

 

     /**

      * hash递减

      *

      * @param key  键

      * @param item 项

      * @param by   要减少记(小于0)

      * @return

      */

     public double hdecr(String key, String item, double by) {

         return redisTemplate.opsForHash().increment(key, item, -by);

     }

 

     // ============================set=============================

 

     /**

      * 根据key获取Set中的所有值

      *

      * @param key 键

      * @return

      */

     public Set<Object> sGet(String key) {

         try {

             return redisTemplate.opsForSet().members(key);

         } catch (Exception e) {

             log.error(key, e);

             return null ;

         }

     }

 

     /**

      * 根据value从一个set中查询,是否存在

      *

      * @param key   键

      * @param value 值

      * @return true 存在 false不存在

      */

     public boolean sHasKey(String key, Object value) {

         try {

             return redisTemplate.opsForSet().isMember(key, value);

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 将数据放入set缓存

      *

      * @param key    键

      * @param values 值 可以是多个

      * @return 成功个数

      */

     public long sSet(String key, Object... values) {

         try {

             return redisTemplate.opsForSet().add(key, values);

         } catch (Exception e) {

             log.error(key, e);

             return 0 ;

         }

     }

 

     /**

      * 将set数据放入缓存

      *

      * @param key    键

      * @param time   时间(秒)

      * @param values 值 可以是多个

      * @return 成功个数

      */

     public long sSetAndTime(String key, long time, Object... values) {

         try {

             Long count = redisTemplate.opsForSet().add(key, values);

             if (time > 0 ) {

                 expire(key, time);

             }

             return count;

         } catch (Exception e) {

             log.error(key, e);

             return 0 ;

         }

     }

 

     /**

      * 获取set缓存的长度

      *

      * @param key 键

      * @return

      */

     public long sGetSetSize(String key) {

         try {

             return redisTemplate.opsForSet().size(key);

         } catch (Exception e) {

             log.error(key, e);

             return 0 ;

         }

     }

 

     /**

      * 移除值为value的

      *

      * @param key    键

      * @param values 值 可以是多个

      * @return 移除的个数

      */

     public long setRemove(String key, Object... values) {

         try {

             Long count = redisTemplate.opsForSet().remove(key, values);

             return count;

         } catch (Exception e) {

             log.error(key, e);

             return 0 ;

         }

     }

 

     // ============================zset=============================

 

     /**

      * 根据key获取Set中的所有值

      *

      * @param key 键

      * @return

      */

     public Set<Object> zSGet(String key) {

         try {

             return redisTemplate.opsForSet().members(key);

         } catch (Exception e) {

             log.error(key, e);

             return null ;

         }

     }

 

     /**

      * 根据value从一个set中查询,是否存在

      *

      * @param key   键

      * @param value 值

      * @return true 存在 false不存在

      */

     public boolean zSHasKey(String key, Object value) {

         try {

             return redisTemplate.opsForSet().isMember(key, value);

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     public Boolean zSSet(String key, Object value, double score) {

         try {

             return redisTemplate.opsForZSet().add(key, value, 2 );

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 将set数据放入缓存

      *

      * @param key    键

      * @param time   时间(秒)

      * @param values 值 可以是多个

      * @return 成功个数

      */

     public long zSSetAndTime(String key, long time, Object... values) {

         try {

             Long count = redisTemplate.opsForSet().add(key, values);

             if (time > 0 ) {

                 expire(key, time);

             }

             return count;

         } catch (Exception e) {

             log.error(key, e);

             return 0 ;

         }

     }

 

     /**

      * 获取set缓存的长度

      *

      * @param key 键

      * @return

      */

     public long zSGetSetSize(String key) {

         try {

             return redisTemplate.opsForSet().size(key);

         } catch (Exception e) {

             log.error(key, e);

             return 0 ;

         }

     }

 

     /**

      * 移除值为value的

      *

      * @param key    键

      * @param values 值 可以是多个

      * @return 移除的个数

      */

     public long zSetRemove(String key, Object... values) {

         try {

             Long count = redisTemplate.opsForSet().remove(key, values);

             return count;

         } catch (Exception e) {

             log.error(key, e);

             return 0 ;

         }

     }

     // ===============================list=================================

 

     /**

      * 获取list缓存的内容

      *

      * @param key   键

      * @param start 开始 0 是第一个元素

      * @param end   结束 -1代表所有值

      * @return

      * @取出来的元素 总数 end-start+1

      */

     public List<Object> lGet(String key, long start, long end) {

         try {

             return redisTemplate.opsForList().range(key, start, end);

         } catch (Exception e) {

             log.error(key, e);

             return null ;

         }

     }

 

     /**

      * 获取list缓存的长度

      *

      * @param key 键

      * @return

      */

     public long lGetListSize(String key) {

         try {

             return redisTemplate.opsForList().size(key);

         } catch (Exception e) {

             log.error(key, e);

             return 0 ;

         }

     }

 

     /**

      * 通过索引 获取list中的值

      *

      * @param key   键

      * @param index 索引 index>=0时, 0 表头,1 第二个元素,依次类推;index<0时,-1,表尾,-2倒数第二个元素,依次类推

      * @return

      */

     public Object lGetIndex(String key, long index) {

         try {

             return redisTemplate.opsForList().index(key, index);

         } catch (Exception e) {

             log.error(key, e);

             return null ;

         }

     }

 

     /**

      * 将list放入缓存

      *

      * @param key   键

      * @param value 值

      * @return

      */

     public boolean lSet(String key, Object value) {

         try {

             redisTemplate.opsForList().rightPush(key, value);

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 将list放入缓存

      *

      * @param key   键

      * @param value 值

      * @param time  时间(秒)

      * @return

      */

     public boolean lSet(String key, Object value, long time) {

         try {

             redisTemplate.opsForList().rightPush(key, value);

             if (time > 0 ) {

                 expire(key, time);

             }

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 将list放入缓存

      *

      * @param key   键

      * @param value 值

      * @return

      */

     public boolean lSet(String key, List<Object> value) {

         try {

             redisTemplate.opsForList().rightPushAll(key, value);

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 将list放入缓存

      *

      * @param key   键

      * @param value 值

      * @param time  时间(秒)

      * @return

      */

     public boolean lSet(String key, List<Object> value, long time) {

         try {

             redisTemplate.opsForList().rightPushAll(key, value);

             if (time > 0 ) {

                 expire(key, time);

             }

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 根据索引修改list中的某条数据

      *

      * @param key   键

      * @param index 索引

      * @param value 值

      * @return

      */

     public boolean lUpdateIndex(String key, long index, Object value) {

         try {

             redisTemplate.opsForList().set(key, index, value);

             return true ;

         } catch (Exception e) {

             log.error(key, e);

             return false ;

         }

     }

 

     /**

      * 移除N个值为value

      *

      * @param key   键

      * @param count 移除多少个

      * @param value 值

      * @return 移除的个数

      */

     public long lRemove(String key, long count, Object value) {

         try {

             Long remove = redisTemplate.opsForList().remove(key, count, value);

             return remove;

         } catch (Exception e) {

             log.error(key, e);

             return 0 ;

         }

     }

 

}

四、POM文件

?

1

2

3

4

5

6

7

<groupId>org.springframework.boot</groupId>

             <artifactId>spring-boot-starter-aop</artifactId>

         </dependency>

         <dependency>

             <groupId>org.springframework.boot</groupId>

             <artifactId>spring-boot-starter-data-redis</artifactId>

         </dependency>

记得配置redis的连接

在需要拦截的地方加入注解即可@AccessLimit(seconds = 5, maxCount = 1, needLogin = false) seconds重置访问频率时间 maxCount 最多请求次数

到此这篇关于SpringBoot项目中接口防刷的完整代码的文章就介绍到这了,更多相关SpringBoot接口防刷内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!

原文链接:https://HdhCmsTestcnblogs测试数据/sansui/p/14469721.html

查看更多关于SpringBoot项目中接口防刷的完整代码的详细内容...

  阅读:11次