flutter 微信通讯录

news2025/1/18 7:17:16

Flutter  仿制微信通讯录效果,致效果如下:

有几个技术细节

  1. 总体可滑动,少于屏幕长度也可滑动
  2. 对于数据的处理。昵称 拼音首字母排序,
  3. 右侧字母导航,点击/滑动;移动到指定位置
  4. 当点击/滑动 右侧移动到最底部的时候; 防止 回弹现象(值滑动到最下面);

    

 ➥总体可滑动,少于屏幕长度也可滑动

主要还是配置  ListView 的 physics 字段

ListView(
  physics: const BouncingScrollPhysics(
    parent: AlwaysScrollableScrollPhysics(),
  ),
  controller: _scrollController,
  children: [],
)

➥对于数据的处理。昵称 拼音首字母排序

这个是我通讯录的一个表字段:

 主要一个排序处理实在 initState 中的:右侧字母导航栏也是根据 我通讯录数据生成的:

@override
void initState() {
  // TODO: implement initState
  super.initState();

  Map<String, List<Map>> cache = {};
  String firstLetter = '';

  for (int i = 0; i < widget.mapContact.length; i++) {
    firstLetter = widget.mapContact[i]['first_letter'];
    if (cache.containsKey(firstLetter)) {
      cache[firstLetter]?.add(widget.mapContact[i]);
    } else {
      cache[firstLetter] = [widget.mapContact[i]];
    }
  }
  List<String> keys = cache.keys.toList();
  keys.sort((a, b) => (a.compareTo(b)));

  for (var element in keys) {
    if (cache[element] != null) {
      _listContact[element] = cache[element]!;
    }
  }
}

➥右侧字母导航,点击/滑动;移动到指定位置

右侧的 字母导航栏是根据,通讯录信息生成的:布局采用 Stack -> Positioned 布局,

主要的包裹了一层 GestureDetector,用来监听手势:

Widget _contactIndex() {
    return Align(
      alignment: Alignment.centerRight,
      child: GestureDetector(
        onVerticalDragDown: _onDragDown,
        onVerticalDragUpdate: _onDragUpdate,
        onVerticalDragEnd: _onDragEnd,
        behavior: HitTestBehavior.opaque,
        child: Column(
          mainAxisSize: MainAxisSize.min,
          mainAxisAlignment: MainAxisAlignment.center,
          children: _listContact.keys
              .toList()
              .asMap()
              .map(
                (key, value) => MapEntry(
                  key,
                  Container(
                    height: 45.cale,
                    width: 80.cale,
                    // color: Colors.red,
                    // padding: EdgeInsets.symmetric(
                    //   vertical: 5.cale,
                    //   horizontal: 20.cale,
                    // ),
                    child: Stack(
                      children: [
                        if (_currentIndex == key)
                          SizedBox(
                            width: 60.cale,
                            height: 60.cale,
                            child: CustomPaint(
                              painter: PainterLetter(
                                draggingLetter: _draggingLetter,
                              ),
                            ),
                          ),
                        Center(
                          child: Text(
                            value.toUpperCase(),
                            style: AppTextStyle.textStyle_22_000000,
                          ),
                        )
                      ],
                    ),
                  ),
                ),
              )
              .values
              .toList(),
        ),
      ),
    );
  }

三个 手势回调函数:

onVerticalDragDown,
onVerticalDragUpdate,
onVerticalDragEnd,
  _onDragDown(DragDownDetails details) {
    int i = details.localPosition.dy ~/ 45.cale;
    if (i >= 0 && i < _listContact.keys.length) {
      setState(() {
        _currentIndex = i;
        _draggingOffset = details.globalPosition;
        _draggingLetter = _listContact.keys.toList()[i];
        _scrollTo(_listContact.keys.toList()[i]);
      });
    }
  }

  _onDragUpdate(DragUpdateDetails details) {
    int i = details.localPosition.dy ~/ 45.cale;
    if (i >= 0 && i < _listContact.keys.length) {
      if (i != _currentIndex) {
        setState(() {
          _currentIndex = i;
          _draggingOffset = details.globalPosition;
          _draggingLetter = _listContact.keys.toList()[i];
          _scrollTo(_listContact.keys.toList()[i]);
        });
      }
    }
  }

  _onDragEnd(DragEndDetails details) {
    setState(() {
      _currentIndex = -1;
    });
  }

移动到指定位置:

这里有几个点需要注意下:( cale 是屏幕适配用的)

  • 480   是通讯录上面四个 常驻 按钮的高度 红色框
  • 60  是通讯录首字母高度   紫色框
  • 120 是 通讯录具体人的高度  橙色框

 

 滚动到具体的位置:

  为了防止 滚动到最底部出现 反弹的效果;这边加了判断:如果计算高度超过 listView 总高度(_scrollController.position.maxScrollExtent),则移动带 listview 高度度! 

  _scrollTo(String chat) {
    double offSet = 480.cale;
    int index = _listContact.keys.toList().indexOf(chat);
    if (index > -1) {
      for (int i = 0; i < index; i++) {
        offSet += 60.cale;
        List<Map>? list = _listContact[_listContact.keys.toList()[i]];
        if (list != null) {
          offSet += 120.cale * list.length;
        }
      }
      if (offSet > _scrollController.position.maxScrollExtent) {
        _scrollController.jumpTo(_scrollController.position.maxScrollExtent);
      } else {
        _scrollController.jumpTo(offSet);
      }
    }
  }

➥ 完成代码

对于通讯录模块主要的就是三个文件:

  • contact_element.dart: 通讯录中 按字母分割现实的 具体好友
  • painter_letter.dart: 拖动时候现实的 椭圆,也可以用图片代替
  • tabbar1_contacts.dart 通讯录主要显示

 完整代码如下:

contact_element.dart

import 'package:flutter/material.dart';
import 'package:imflutter/wrap/extension/extension.dart';

import '../../const/app_colors.dart';
import '../../const/app_textStyle.dart';
import '../../wrap/widget/app_widget.dart';

class ContactElement extends StatefulWidget {
  final List<Map> datum;
  final String keyName;
  const ContactElement({Key? key, required this.keyName, required this.datum})
      : super(key: key);

  @override
  State<ContactElement> createState() => _ContactElementState();
}

class _ContactElementState extends State<ContactElement>
    with AutomaticKeepAliveClientMixin {
  @override
  Widget build(BuildContext context) {
    return Container(
      width: 200.cale,
      // height: 200.cale,
      color: Colors.white,
      child: Column(
        crossAxisAlignment: CrossAxisAlignment.start,
        children: [
          Container(
            padding: EdgeInsets.only(left: 30.cale),
            height: 60.cale,
            width: double.infinity,
            color: AppColor.colorEDEDED,
            child: Align(
              alignment: Alignment.centerLeft,
              child: Text(
                widget.keyName,
                style: AppTextStyle.textStyle_24_505050,
              ),
            ),
          ),
          ...widget.datum
              .asMap()
              .map(
                (key, value) => MapEntry(
                  key,
                  Container(
                    padding: EdgeInsets.only(left: 28.cale),
                    height: 120.cale,
                    color: Colors.white,
                    child: Row(
                      children: [
                        ClipRRect(
                          borderRadius: BorderRadius.circular(7.cale),
                          child: AppWidget.cachedImage(
                            value['icon'],
                            width: 88.cale,
                            height: 88.cale,
                          ),
                        ),
                        Expanded(
                          child: Container(
                            height: 120.cale,
                            margin: EdgeInsets.only(
                              left: 30.cale,
                            ),
                            padding: EdgeInsets.only(
                              right: 80.cale,
                            ),
                            decoration: BoxDecoration(
                              border: Border(
                                bottom: key == widget.datum.length - 1
                                    ? BorderSide.none
                                    : BorderSide(
                                        width: 1, color: AppColor.colorEFEFEF),
                              ),
                            ),
                            child: Container(
                              alignment: Alignment.centerLeft,
                              child: Text(
                                value['nick_name'],
                                overflow: TextOverflow.ellipsis,
                                style: AppTextStyle.textStyle_30_000000,
                              ),
                            ),
                          ),
                        ),
                      ],
                    ),
                  ),
                ),
              )
              .values
              .toList()
        ],
      ),
    );
  }

  @override
  // TODO: implement wantKeepAlive
  bool get wantKeepAlive => true;
}

 painter_letter.dart

import 'dart:math';
import 'package:flutter/material.dart';
import 'package:imflutter/wrap/extension/extension.dart';

class PainterLetter extends CustomPainter {
  final String draggingLetter;
  const PainterLetter({Key? key, required this.draggingLetter});

  @override
  void paint(Canvas canvas, Size size) {
    // TODO: implement paint
    // 创建画笔

    // final Paint paint = Paint(); // 创建画笔
    // paint
    //   ..color = Colors.blue //颜色
    //   ..strokeWidth = 4 //线宽
    //   ..style = PaintingStyle.stroke; //模式--线型
    // canvas.drawLine(Offset(0, 0), Offset(100, 100), paint); //绘制线
    // return;

    //原点移到左下角
    canvas.translate(0, 25.cale);
    Paint paint = Paint()
      ..color = Colors.grey
      ..strokeWidth = 2
      ..style = PaintingStyle.fill;

    Path path = Path();
    // 绘制文字
    path.lineTo(0, -size.width.cale);
    // path.conicTo(33, -28, 20, 0, 1);

    path.arcToPoint(Offset(size.width.cale * 1.2, 0),
        radius: Radius.circular(size.width.cale * 1.2),
        largeArc: true,
        clockwise: true);
    path.close();
    // var bounds = path.getBounds();
    canvas.save();
    // canvas.translate(-bounds.width / 2.cale, bounds.height / 2.cale);
    canvas.rotate(pi * 1.2);
    canvas.drawPath(path, paint);
    canvas.restore();
    // 绘制文字
    var textPainter = TextPainter(
        text: TextSpan(
            text: draggingLetter.toUpperCase(),
            style: TextStyle(
              fontSize: 38.cale,
              foreground: Paint()
                ..style = PaintingStyle.fill
                ..color = Colors.white,
            )),
        textAlign: TextAlign.center,
        textDirection: TextDirection.ltr);
    textPainter.layout();
    canvas.translate(-35.cale, -53.cale);
    // canvas.translate(
    //     -size.width.cale * 2 - 10.cale, (-size.height / 2).cale - 10.cale);
    textPainter.paint(canvas, Offset(-25.cale, 37.cale));
  }

  @override
  bool shouldRepaint(covariant CustomPainter oldDelegate) {
    // TODO: implement shouldRepaint
    return true;
  }
}

tabbar1_contacts.dart

import 'package:flutter/material.dart';
import 'package:imflutter/pages/tabbar1/contact_element.dart';
import 'package:imflutter/pages/tabbar1/painter_letter.dart';
import 'package:imflutter/wrap/extension/extension.dart';
import '../../const/app_colors.dart';
import '../../const/app_textStyle.dart';

class TabBar1Contacts extends StatefulWidget {
  final List<Map> mapContact;
  const TabBar1Contacts({Key? key, required this.mapContact}) : super(key: key);

  @override
  State<TabBar1Contacts> createState() => _TabBar1ContactsState();
}

class _TabBar1ContactsState extends State<TabBar1Contacts> {
  final List<Map> _topOptions = [
    {'icon': 'assets/common/ic_new_friend.webp', 'title': '新朋友'},
    {'icon': 'assets/common/ic_group.webp', 'title': '群聊'},
    {'icon': 'assets/common/ic_tag.webp', 'title': '标签'},
    {'icon': 'assets/common/ic_offical.webp', 'title': '公众号'},
  ];
  final Map<String, List<Map>> _listContact = {};
  int _currentIndex = -1;
  Offset _draggingOffset = const Offset(0, 0);
  String _draggingLetter = 'd';
  final ScrollController _scrollController = ScrollController();

  @override
  void initState() {
    // TODO: implement initState
    super.initState();

    Map<String, List<Map>> cache = {};
    String firstLetter = '';

    for (int i = 0; i < widget.mapContact.length; i++) {
      firstLetter = widget.mapContact[i]['first_letter'];
      if (cache.containsKey(firstLetter)) {
        cache[firstLetter]?.add(widget.mapContact[i]);
      } else {
        cache[firstLetter] = [widget.mapContact[i]];
      }
    }
    List<String> keys = cache.keys.toList();
    keys.sort((a, b) => (a.compareTo(b)));

    for (var element in keys) {
      if (cache[element] != null) {
        _listContact[element] = cache[element]!;
      }
    }
  }

  @override
  Widget build(BuildContext context) {
    return Stack(
      children: [
        ListView(
          physics: const BouncingScrollPhysics(
            parent: AlwaysScrollableScrollPhysics(),
          ),
          controller: _scrollController,
          children: [
            ..._topEle(),
            ..._listContact.keys
                .toList()
                .asMap()
                .map(
                  (key, value) => MapEntry(
                    key,
                    ContactElement(
                      keyName: _listContact.keys.toList()[key].toUpperCase(),
                      datum: _listContact.values.toList()[key],
                    ),
                  ),
                )
                .values
                .toList()
          ],
        ),
        Positioned(
          child: _contactIndex(),
        ),
      ],
    );
  }

  List<Widget> _topEle() {
    return _topOptions
        .asMap()
        .map(
          (key, value) => MapEntry(
            key,
            Container(
              padding: EdgeInsets.only(left: 28.cale),
              height: 120.cale,
              color: Colors.white,
              child: Row(
                children: [
                  ClipRRect(
                    borderRadius: BorderRadius.circular(7.cale),
                    child: Image.asset(
                      value['icon'],
                      width: 88.cale,
                      height: 88.cale,
                    ),
                  ),
                  Expanded(
                    child: Container(
                      height: 120.cale,
                      margin: EdgeInsets.only(left: 30.cale),
                      decoration: BoxDecoration(
                        border: Border(
                          bottom: key == _topOptions.length - 1
                              ? BorderSide.none
                              : BorderSide(
                                  width: 1, color: AppColor.colorEFEFEF),
                        ),
                      ),
                      child: Align(
                        alignment: Alignment.centerLeft,
                        child: Text(
                          value['title'],
                          style: AppTextStyle.textStyle_30_000000,
                        ),
                      ),
                    ),
                  ),
                ],
              ),
            ),
          ),
        )
        .values
        .toList();
  }

  Widget _contactIndex() {
    return Align(
      alignment: Alignment.centerRight,
      child: GestureDetector(
        onVerticalDragDown: _onDragDown,
        onVerticalDragUpdate: _onDragUpdate,
        onVerticalDragEnd: _onDragEnd,
        behavior: HitTestBehavior.opaque,
        child: Column(
          mainAxisSize: MainAxisSize.min,
          mainAxisAlignment: MainAxisAlignment.center,
          children: _listContact.keys
              .toList()
              .asMap()
              .map(
                (key, value) => MapEntry(
                  key,
                  Container(
                    height: 45.cale,
                    width: 80.cale,
                    // color: Colors.red,
                    // padding: EdgeInsets.symmetric(
                    //   vertical: 5.cale,
                    //   horizontal: 20.cale,
                    // ),
                    child: Stack(
                      children: [
                        if (_currentIndex == key)
                          SizedBox(
                            width: 60.cale,
                            height: 60.cale,
                            child: CustomPaint(
                              painter: PainterLetter(
                                draggingLetter: _draggingLetter,
                              ),
                            ),
                          ),
                        Center(
                          child: Text(
                            value.toUpperCase(),
                            style: AppTextStyle.textStyle_22_000000,
                          ),
                        )
                      ],
                    ),
                  ),
                ),
              )
              .values
              .toList(),
        ),
      ),
    );
  }

  _onDragDown(DragDownDetails details) {
    int i = details.localPosition.dy ~/ 45.cale;
    // print("-----------------details.localPosition:${details.localPosition}");
    // print("-----------------_onDragDown:$i");
    // print(
    //     "----------------- _listContact.keys.length:${_listContact.keys.length}");
    // print(
    //     "----------------- _listContact.keys.length:${_listContact.keys.toList()[i]}");
    // if(i<  _listContact.keys.length){
    //
    // }
    if (i >= 0 && i < _listContact.keys.length) {
      setState(() {
        _currentIndex = i;
        _draggingOffset = details.globalPosition;
        _draggingLetter = _listContact.keys.toList()[i];
        _scrollTo(_listContact.keys.toList()[i]);
      });
    }
  }

  _onDragUpdate(DragUpdateDetails details) {
    int i = details.localPosition.dy ~/ 45.cale;
    if (i >= 0 && i < _listContact.keys.length) {
      if (i != _currentIndex) {
        setState(() {
          _currentIndex = i;
          _draggingOffset = details.globalPosition;
          _draggingLetter = _listContact.keys.toList()[i];
          _scrollTo(_listContact.keys.toList()[i]);
        });
      }
    }
  }

  _onDragEnd(DragEndDetails details) {
    setState(() {
      _currentIndex = -1;
    });
  }

  _scrollTo(String chat) {
    double offSet = 480.cale;
    int index = _listContact.keys.toList().indexOf(chat);
    if (index > -1) {
      for (int i = 0; i < index; i++) {
        offSet += 60.cale;
        List<Map>? list = _listContact[_listContact.keys.toList()[i]];
        if (list != null) {
          offSet += 120.cale * list.length;
        }
      }
      if (offSet > _scrollController.position.maxScrollExtent) {
        _scrollController.jumpTo(_scrollController.position.maxScrollExtent);
      } else {
        _scrollController.jumpTo(offSet);
      }
    }
  }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/364887.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

大数据实操项目分享:餐饮智能推荐服务在线实习项目

项目背景&#xff1a;在“互联网"背景下&#xff0c;餐饮企业的经营方式发生了很大的变革&#xff1a;团购和020拓宽了销售 渠道&#xff0c;电子点餐、店内WIFI等信息技术提升了服务水平&#xff0c;大数据、私人定制更好地满足了细分市场的需求等。但是与此同时&#xf…

天!转转MySQL机房迁移半小时结束战斗?

文章目录1 背景2 迁移方案选择2.1 方案一&#xff1a;扩容主从切换2.2 方案二&#xff1a;级联切换2.3 方案对比3 如何又快又稳完成MySQL机房迁移3.1 提前搭建级联3.2 停服3.3 批量操作自动化&#xff0c;关键步骤解耦3.4 集群分级3.5 切换前、后置检查3.6 灰度切换验证4 写在最…

rk3288-android8.1-以太网ethernet和蓝牙Bluetooth

遇到一个现象,以太网和蓝牙打不开 经过不断分析和查找发现问题在.config中 CONFIG_MOTORCOMM_PHYy 会导致以太网的eth0注册不成功(现在是双网口,还有个USB网卡) 改成# CONFIG_MOTORCOMM_PHY is not set 后以太网可以正常 # CONFIG_RTC_DRV_RK808 is not set 会导致蓝牙打不…

【分类评价指标】如何评估多(二)分类算法的性能:Acc、Precision、Recall、F1等

【分类评价指标】如何评估多&#xff08;二&#xff09;分类算法的性能&#xff1a;Acc、Precision、Recall、F1等 文章目录【分类评价指标】如何评估多&#xff08;二&#xff09;分类算法的性能&#xff1a;Acc、Precision、Recall、F1等1. 前言2. 二分类任务2.1 混淆矩阵2.2…

工控攻击,黑客组织GhostSec 称入侵以色列55 家Berghof PLC

“巴以冲突”在网络上依然硝烟弥漫。当地时间9月12日消息&#xff0c; 一个名为GhostSec的黑客组织声称入侵了以色列55台Berghof可编程逻辑控制器&#xff08;PLC&#xff09;。该网络攻击行为被视为“解放巴勒斯坦”运动的组成部分。 以色列工业网络安全公司OTORIO对此次事件…

JVM内存结构和GC调优

一 、JVM简介 1.1 JVM是什么&#xff1f; Java Virtual Machine(Java虚拟机) Write Once Run Anywhere 1.2 JDK JRE JVM Java官网 &#xff1a;https://docs.oracle.com/javase/8/ Reference -> Developer Guides -> 定位到: https://docs.oracle.com/javase/8/docs…

MySQL数据库——JDBC编程

文章目录一、什么是Java的JDBC二、JDBC编程三、代码整体展示一、什么是Java的JDBC JDBC&#xff0c;即Java Database Connectivity。意思是java数据库连接。是一种用来执行 SQL 语句的 JavaAPI&#xff0c;是Java中数据库的连接规范。这个 API 由 java.sql* 和 javax.sql* 包中…

Windows 环境下,使用 ESP32-S3 USB 接口进行 JTAG 调试的流程

前提 在 windows 上安装 esp-idf CMD 软件编译环境&#xff0c;可参考“Windows 上搭建 ESP-IDF SDK 编译环境 Visual Studio Code 软件编程环境”说明。硬件上使用 ESP32-S3 USB 接口与 PC 端建立连接&#xff0c;为方便测试&#xff0c;可使用官方发布的 ESP32-S3-DevKitC-…

“办”了三年数字化活动,这家公司成为行业独角兽

疫情三年&#xff0c;竟然有公司靠办活动成为行业独角兽&#xff1f;你没听错&#xff01;持续的疫情&#xff0c;让各行各业的数字化进程大大提速&#xff0c;活动这个历史悠久的领域也不例外。在疫情期间&#xff0c;数字化深刻影响着各行业招聘、展览、发布会等多种活动&…

2023年新能源汽车行业研究报告

第一章 行业概况 新能源汽车&#xff0c;是指采用新型动力系统&#xff0c;完全或者主要依靠新型能源驱动的汽车&#xff0c;包括纯电动汽车、插电式混合动力汽车、增程式混合动力汽车和燃料电池汽车等。国际上&#xff0c;混合动力汽车&#xff08;含中混、强混、插电式混动&…

浅谈常用的日志框架

文章目录1.为什么需要日志框架2.常见日志框架2.1.日志框架介绍2.2.市面上的日志框架3.Slf4j使用3.1.如何在系统中使用SLF4j3.2.可能存在的问题4.SpringBoot日志的默认配置5.SpringBoot指定日志文件6.切换日志框架1.为什么需要日志框架 通过日志的方式记录系统运行的过程或错误以…

vscode SSH 保存密码自动登录服务器

先在win local上拿到秘钥&#xff0c;然后再把这秘钥copy 进服务器 1. 创建 RSA 密钥对 第一步是在客户端机器&#xff08;通常是您的计算机 win 10&#xff09;上创建密钥对&#xff1a;打开powershell, 输入 ssh-keygen默认情况下ssh-keygen将创建一个 2048 位 RSA 密钥对…

套接字及分层模型(一)

套接字通信 1.内核开发的工程师将网络相关的头文件存储到一个专门目录include/net中&#xff0c;而不是存储到include/linux 2.ISO/OS和TCP/IP参考模型 应用层&#xff1a;网络服务与最终用户的接口&#xff1b; 表示层&#xff1a;数据的表示&#xff0c;安全及压缩&#xf…

SQL常用 增删改 语句

目录 1 插入语句 2 更新语句 2.1 更新单个值 2.2 更新表中的所有行 2.3 更新多个值 3 删除语句 1 插入语句 INSERT INTO 表名称 (字段名称) VALUES (值) 多个字段或值用逗号分隔字段与值的顺序要对应 我们添加后查询一下来验证结果 2 更新语句 2.1 更新单个值…

U盘安装CentOS 7 初体验

一、刻录镜像到U盘 把下载好的linux的 .iso 镜像文件 要通过刻录到U盘里才可以使用&#xff0c;不像虚拟机中一样可以直接导入使用。 使用软件 UltraISO 刻录到U盘上。 二、改启动方式 BIOS 改为U盘优先启动。 三、安装遇到的问题 1、找不到装有镜像的U盘盘符&#xff0c;…

企业级IM即时通讯私有云解决方案

信息无疑是当今社会最重要的资源之一&#xff0c;网络安全也成为了维护信息安全的一道防线。私有云办公通信平台成为不少企业的选择&#xff0c;不仅能实现自主部署&#xff0c;还能真正自主掌控云和数据&#xff0c;确保业务数据的安全、可控。然而企业如果自主研发私有云办公…

生产者消费者模型(多线程工作)

目录 1.模型前提 2.阻塞队列&#xff08;消费场所&#xff09; 3. 实验 4.有关效率 1.模型前提 以单生产者对单消费者为例子&#xff1a; 前提一&#xff1a;有一个缓冲区作为消费场所。 前提二&#xff1a;有两种功能不同的线程分别具有消费与生产的能力。 前提三&…

分布式事务框架Seata

分布式事务基础 <<分布式事务基础理论>> <<分布式事务解决方案>> Seata 一款开源的分布式事务解决方案&#xff0c;致力于提供高性能和简单易用的分布式事务服务。Seata 将为用户提供了 AT、TCC、SAGA 和 XA 事务模式&#xff0c;为用户打造一站式的…

Socket编程 | TCP服务器 之 并发阻塞模型(多进程实现)

TCP服务器IO模型 之 并发阻塞 1. 引言 在 Linux 环境下多进程的应用很多,其中最主要的就是网络/客户服务器。多进程服务器是当客户有请求时,服务器用一个子进程来处理客户请求。父进程继续等待其它客户的请求。这种方法的优点是当客户有请求时,服务器能及时处理客户,特别是…

JavaEE简单示例——动态SQL元素<where>

简单介绍&#xff1a; 在我们之前使用where关键字进行查询的时候&#xff0c;总是会在后面添加一个11恒等式&#xff0c;并且在每一个可能拼接的SQL语句前面都加上一个and关键字&#xff0c;防止当后续的所有条件都不满足的时候&#xff0c;where关键字在最后直接跟and的时候也…