网站首页 > 基础教程 正文
前两篇中简单介绍并使用了IDL,那么IDL支持哪些数据类型,语法又是什么?本篇文章将与大家一起学习。
Java中数据类型可以分为基本数据类型和对象类型。IDL中也类似,包含基本数据类型、容器、结构体等。更准确的说Thrift 类型系统由预定义的基本类型、用户定义的结构、容器类型、异常和服务定义组成。下面会分别进行介绍。
1.基本数据类型
基本数据类型有如下几种:
- bool: 布尔型,1个字节,取值为true或false。
- byte: 有符号单字节(8位)整型。
- i16: 有符号16位整型。
- i32: 有符号32位整型。
- i64: 有符号64位整型。
- double: 64位浮点数。
- binary: 字节数组。
- string: 字符串
对应Java的数据类型按照顺序分别是
boolean、byte、short、int、long、double、ByteBuffer、String
2.容器
Thrift 容器是强类型容器,映射到常用编程语言中最常用的容器。它们使用 Java 泛型样式进行注释。有三种容器类型可供选择: list<t1>:t1 类型元素的有序列表。可能包含重复项。 set<t1>:类型为 t1 的无序唯一元素集。 map<t1,t2>:t1 - t2 类型键值对。
容器中使用的类型可以是任何有效的 Thrift 类型(包括结构和异常),但不包括service。
3.结构体
结构体可对应Java的类,但是不能继承,可以进行结构体嵌套。下面结合demo进行说明。
UserResponse.thrift
namespace java ltd.klein.thrift.server.api.resp
include '../param/UserType.thrift'
struct UserResponse {
/*
* 请求成功/失败
*/
1: bool success;
/*
* 请求结果信息
*/
2: string msg;
3: list<User> users;
}
struct User {
1: required i32 id;
2: required string nickName;
3: required i64 registerTime;
4: UserType.UserType userType = UserType.UserType.CONSUMER;
5: binary avatar;
6: optional i16 age;
7: optional string desc;
}
- namespace:
- 编译为Java文件后,代表的是Java的package
- include
- 引入其他thrift文件定义的struct等类型
- struct
- 定义一个结构体,编译后对应的是Java的类
- 成员变量需要写明顺序
- 同一个thrfit文件中,可以定义多个struct,不需要与thrift文件名一致
- required 标注的参数,无论是否赋值,在序列化与反序列化时都会被处理
- optional 标注的参数,如果没有赋值,在序列化与反序列化时会被忽略
由thrift文件生成的Java文件如下
/**
* Autogenerated by Thrift Compiler (0.21.0)
*
* DO NOT EDIT UNLESS YOU ARE SURE THAT YOU KNOW WHAT YOU ARE DOING
* @generated
*/
package ltd.klein.thrift.server.api.resp;
@SuppressWarnings({"cast", "rawtypes", "serial", "unchecked", "unused"})
@javax.annotation.Generated(value = "Autogenerated by Thrift Compiler (0.21.0)", date = "2024-11-22")
public class UserResponse implements org.apache.thrift.TBase<UserResponse, UserResponse._Fields>, java.io.Serializable, Cloneable, Comparable<UserResponse> {
private static final org.apache.thrift.protocol.TStruct STRUCT_DESC = new org.apache.thrift.protocol.TStruct("UserResponse");
private static final org.apache.thrift.protocol.TField SUCCESS_FIELD_DESC = new org.apache.thrift.protocol.TField("success", org.apache.thrift.protocol.TType.BOOL, (short)1);
private static final org.apache.thrift.protocol.TField MSG_FIELD_DESC = new org.apache.thrift.protocol.TField("msg", org.apache.thrift.protocol.TType.STRING, (short)2);
private static final org.apache.thrift.protocol.TField USERS_FIELD_DESC = new org.apache.thrift.protocol.TField("users", org.apache.thrift.protocol.TType.LIST, (short)3);
private static final org.apache.thrift.scheme.SchemeFactory STANDARD_SCHEME_FACTORY = new UserResponseStandardSchemeFactory();
private static final org.apache.thrift.scheme.SchemeFactory TUPLE_SCHEME_FACTORY = new UserResponseTupleSchemeFactory();
public boolean success; // required
public @org.apache.thrift.annotation.Nullable java.lang.String msg; // required
public @org.apache.thrift.annotation.Nullable java.util.List<User> users; // required
/** The set of fields this struct contains, along with convenience methods for finding and manipulating them. */
public enum _Fields implements org.apache.thrift.TFieldIdEnum {
SUCCESS((short)1, "success"),
MSG((short)2, "msg"),
USERS((short)3, "users");
private static final java.util.Map<java.lang.String, _Fields> byName = new java.util.HashMap<java.lang.String, _Fields>();
static {
for (_Fields field : java.util.EnumSet.allOf(_Fields.class)) {
byName.put(field.getFieldName(), field);
}
}
/**
* Find the _Fields constant that matches fieldId, or null if its not found.
*/
@org.apache.thrift.annotation.Nullable
public static _Fields findByThriftId(int fieldId) {
switch(fieldId) {
case 1: // SUCCESS
return SUCCESS;
case 2: // MSG
return MSG;
case 3: // USERS
return USERS;
default:
return null;
}
}
/**
* Find the _Fields constant that matches fieldId, throwing an exception
* if it is not found.
*/
public static _Fields findByThriftIdOrThrow(int fieldId) {
_Fields fields = findByThriftId(fieldId);
if (fields == null) throw new java.lang.IllegalArgumentException("Field " + fieldId + " doesn't exist!");
return fields;
}
/**
* Find the _Fields constant that matches name, or null if its not found.
*/
@org.apache.thrift.annotation.Nullable
public static _Fields findByName(java.lang.String name) {
return byName.get(name);
}
private final short _thriftId;
private final java.lang.String _fieldName;
_Fields(short thriftId, java.lang.String fieldName) {
_thriftId = thriftId;
_fieldName = fieldName;
}
@Override
public short getThriftFieldId() {
return _thriftId;
}
@Override
public java.lang.String getFieldName() {
return _fieldName;
}
}
// isset id assignments
private static final int __SUCCESS_ISSET_ID = 0;
private byte __isset_bitfield = 0;
public static final java.util.Map<_Fields, org.apache.thrift.meta_data.FieldMetaData> metaDataMap;
static {
java.util.Map<_Fields, org.apache.thrift.meta_data.FieldMetaData> tmpMap = new java.util.EnumMap<_Fields, org.apache.thrift.meta_data.FieldMetaData>(_Fields.class);
tmpMap.put(_Fields.SUCCESS, new org.apache.thrift.meta_data.FieldMetaData("success", org.apache.thrift.TFieldRequirementType.DEFAULT,
new org.apache.thrift.meta_data.FieldValueMetaData(org.apache.thrift.protocol.TType.BOOL)));
tmpMap.put(_Fields.MSG, new org.apache.thrift.meta_data.FieldMetaData("msg", org.apache.thrift.TFieldRequirementType.DEFAULT,
new org.apache.thrift.meta_data.FieldValueMetaData(org.apache.thrift.protocol.TType.STRING)));
tmpMap.put(_Fields.USERS, new org.apache.thrift.meta_data.FieldMetaData("users", org.apache.thrift.TFieldRequirementType.DEFAULT,
new org.apache.thrift.meta_data.ListMetaData(org.apache.thrift.protocol.TType.LIST,
new org.apache.thrift.meta_data.StructMetaData(org.apache.thrift.protocol.TType.STRUCT, User.class))));
metaDataMap = java.util.Collections.unmodifiableMap(tmpMap);
org.apache.thrift.meta_data.FieldMetaData.addStructMetaDataMap(UserResponse.class, metaDataMap);
}
public UserResponse() {
}
public UserResponse(
boolean success,
java.lang.String msg,
java.util.List<User> users)
{
this();
this.success = success;
setSuccessIsSet(true);
this.msg = msg;
this.users = users;
}
/**
* Performs a deep copy on <i>other</i>.
*/
public UserResponse(UserResponse other) {
__isset_bitfield = other.__isset_bitfield;
this.success = other.success;
if (other.isSetMsg()) {
this.msg = other.msg;
}
if (other.isSetUsers()) {
java.util.List<User> __this__users = new java.util.ArrayList<User>(other.users.size());
for (User other_element : other.users) {
__this__users.add(new User(other_element));
}
this.users = __this__users;
}
}
@Override
public UserResponse deepCopy() {
return new UserResponse(this);
}
@Override
public void clear() {
setSuccessIsSet(false);
this.success = false;
this.msg = null;
this.users = null;
}
public boolean isSuccess() {
return this.success;
}
public UserResponse setSuccess(boolean success) {
this.success = success;
setSuccessIsSet(true);
return this;
}
public void unsetSuccess() {
__isset_bitfield = org.apache.thrift.EncodingUtils.clearBit(__isset_bitfield, __SUCCESS_ISSET_ID);
}
/** Returns true if field success is set (has been assigned a value) and false otherwise */
public boolean isSetSuccess() {
return org.apache.thrift.EncodingUtils.testBit(__isset_bitfield, __SUCCESS_ISSET_ID);
}
public void setSuccessIsSet(boolean value) {
__isset_bitfield = org.apache.thrift.EncodingUtils.setBit(__isset_bitfield, __SUCCESS_ISSET_ID, value);
}
@org.apache.thrift.annotation.Nullable
public java.lang.String getMsg() {
return this.msg;
}
public UserResponse setMsg(@org.apache.thrift.annotation.Nullable java.lang.String msg) {
this.msg = msg;
return this;
}
public void unsetMsg() {
this.msg = null;
}
/** Returns true if field msg is set (has been assigned a value) and false otherwise */
public boolean isSetMsg() {
return this.msg != null;
}
public void setMsgIsSet(boolean value) {
if (!value) {
this.msg = null;
}
}
public int getUsersSize() {
return (this.users == null) ? 0 : this.users.size();
}
@org.apache.thrift.annotation.Nullable
public java.util.Iterator<User> getUsersIterator() {
return (this.users == null) ? null : this.users.iterator();
}
public void addToUsers(User elem) {
if (this.users == null) {
this.users = new java.util.ArrayList<User>();
}
this.users.add(elem);
}
@org.apache.thrift.annotation.Nullable
public java.util.List<User> getUsers() {
return this.users;
}
public UserResponse setUsers(@org.apache.thrift.annotation.Nullable java.util.List<User> users) {
this.users = users;
return this;
}
public void unsetUsers() {
this.users = null;
}
/** Returns true if field users is set (has been assigned a value) and false otherwise */
public boolean isSetUsers() {
return this.users != null;
}
public void setUsersIsSet(boolean value) {
if (!value) {
this.users = null;
}
}
@Override
public void setFieldValue(_Fields field, @org.apache.thrift.annotation.Nullable java.lang.Object value) {
switch (field) {
case SUCCESS:
if (value == null) {
unsetSuccess();
} else {
setSuccess((java.lang.Boolean)value);
}
break;
case MSG:
if (value == null) {
unsetMsg();
} else {
setMsg((java.lang.String)value);
}
break;
case USERS:
if (value == null) {
unsetUsers();
} else {
setUsers((java.util.List<User>)value);
}
break;
}
}
@org.apache.thrift.annotation.Nullable
@Override
public java.lang.Object getFieldValue(_Fields field) {
switch (field) {
case SUCCESS:
return isSuccess();
case MSG:
return getMsg();
case USERS:
return getUsers();
}
throw new java.lang.IllegalStateException();
}
/** Returns true if field corresponding to fieldID is set (has been assigned a value) and false otherwise */
@Override
public boolean isSet(_Fields field) {
if (field == null) {
throw new java.lang.IllegalArgumentException();
}
switch (field) {
case SUCCESS:
return isSetSuccess();
case MSG:
return isSetMsg();
case USERS:
return isSetUsers();
}
throw new java.lang.IllegalStateException();
}
@Override
public boolean equals(java.lang.Object that) {
if (that instanceof UserResponse)
return this.equals((UserResponse)that);
return false;
}
public boolean equals(UserResponse that) {
if (that == null)
return false;
if (this == that)
return true;
boolean this_present_success = true;
boolean that_present_success = true;
if (this_present_success || that_present_success) {
if (!(this_present_success && that_present_success))
return false;
if (this.success != that.success)
return false;
}
boolean this_present_msg = true && this.isSetMsg();
boolean that_present_msg = true && that.isSetMsg();
if (this_present_msg || that_present_msg) {
if (!(this_present_msg && that_present_msg))
return false;
if (!this.msg.equals(that.msg))
return false;
}
boolean this_present_users = true && this.isSetUsers();
boolean that_present_users = true && that.isSetUsers();
if (this_present_users || that_present_users) {
if (!(this_present_users && that_present_users))
return false;
if (!this.users.equals(that.users))
return false;
}
return true;
}
@Override
public int hashCode() {
int hashCode = 1;
hashCode = hashCode * 8191 + ((success) ? 131071 : 524287);
hashCode = hashCode * 8191 + ((isSetMsg()) ? 131071 : 524287);
if (isSetMsg())
hashCode = hashCode * 8191 + msg.hashCode();
hashCode = hashCode * 8191 + ((isSetUsers()) ? 131071 : 524287);
if (isSetUsers())
hashCode = hashCode * 8191 + users.hashCode();
return hashCode;
}
@Override
public int compareTo(UserResponse other) {
if (!getClass().equals(other.getClass())) {
return getClass().getName().compareTo(other.getClass().getName());
}
int lastComparison = 0;
lastComparison = java.lang.Boolean.compare(isSetSuccess(), other.isSetSuccess());
if (lastComparison != 0) {
return lastComparison;
}
if (isSetSuccess()) {
lastComparison = org.apache.thrift.TBaseHelper.compareTo(this.success, other.success);
if (lastComparison != 0) {
return lastComparison;
}
}
lastComparison = java.lang.Boolean.compare(isSetMsg(), other.isSetMsg());
if (lastComparison != 0) {
return lastComparison;
}
if (isSetMsg()) {
lastComparison = org.apache.thrift.TBaseHelper.compareTo(this.msg, other.msg);
if (lastComparison != 0) {
return lastComparison;
}
}
lastComparison = java.lang.Boolean.compare(isSetUsers(), other.isSetUsers());
if (lastComparison != 0) {
return lastComparison;
}
if (isSetUsers()) {
lastComparison = org.apache.thrift.TBaseHelper.compareTo(this.users, other.users);
if (lastComparison != 0) {
return lastComparison;
}
}
return 0;
}
@org.apache.thrift.annotation.Nullable
@Override
public _Fields fieldForId(int fieldId) {
return _Fields.findByThriftId(fieldId);
}
@Override
public void read(org.apache.thrift.protocol.TProtocol iprot) throws org.apache.thrift.TException {
scheme(iprot).read(iprot, this);
}
@Override
public void write(org.apache.thrift.protocol.TProtocol oprot) throws org.apache.thrift.TException {
scheme(oprot).write(oprot, this);
}
@Override
public java.lang.String toString() {
java.lang.StringBuilder sb = new java.lang.StringBuilder("UserResponse(");
boolean first = true;
sb.append("success:");
sb.append(this.success);
first = false;
if (!first) sb.append(", ");
sb.append("msg:");
if (this.msg == null) {
sb.append("null");
} else {
sb.append(this.msg);
}
first = false;
if (!first) sb.append(", ");
sb.append("users:");
if (this.users == null) {
sb.append("null");
} else {
sb.append(this.users);
}
first = false;
sb.append(")");
return sb.toString();
}
public void validate() throws org.apache.thrift.TException {
// check for required fields
// check for sub-struct validity
}
private void writeObject(java.io.ObjectOutputStream out) throws java.io.IOException {
try {
write(new org.apache.thrift.protocol.TCompactProtocol(new org.apache.thrift.transport.TIOStreamTransport(out)));
} catch (org.apache.thrift.TException te) {
throw new java.io.IOException(te);
}
}
private void readObject(java.io.ObjectInputStream in) throws java.io.IOException, java.lang.ClassNotFoundException {
try {
// it doesn't seem like you should have to do this, but java serialization is wacky, and doesn't call the default constructor.
__isset_bitfield = 0;
read(new org.apache.thrift.protocol.TCompactProtocol(new org.apache.thrift.transport.TIOStreamTransport(in)));
} catch (org.apache.thrift.TException te) {
throw new java.io.IOException(te);
}
}
private static class UserResponseStandardSchemeFactory implements org.apache.thrift.scheme.SchemeFactory {
@Override
public UserResponseStandardScheme getScheme() {
return new UserResponseStandardScheme();
}
}
private static class UserResponseStandardScheme extends org.apache.thrift.scheme.StandardScheme<UserResponse> {
@Override
public void read(org.apache.thrift.protocol.TProtocol iprot, UserResponse struct) throws org.apache.thrift.TException {
org.apache.thrift.protocol.TField schemeField;
iprot.readStructBegin();
while (true)
{
schemeField = iprot.readFieldBegin();
if (schemeField.type == org.apache.thrift.protocol.TType.STOP) {
break;
}
switch (schemeField.id) {
case 1: // SUCCESS
if (schemeField.type == org.apache.thrift.protocol.TType.BOOL) {
struct.success = iprot.readBool();
struct.setSuccessIsSet(true);
} else {
org.apache.thrift.protocol.TProtocolUtil.skip(iprot, schemeField.type);
}
break;
case 2: // MSG
if (schemeField.type == org.apache.thrift.protocol.TType.STRING) {
struct.msg = iprot.readString();
struct.setMsgIsSet(true);
} else {
org.apache.thrift.protocol.TProtocolUtil.skip(iprot, schemeField.type);
}
break;
case 3: // USERS
if (schemeField.type == org.apache.thrift.protocol.TType.LIST) {
{
org.apache.thrift.protocol.TList _list0 = iprot.readListBegin();
struct.users = new java.util.ArrayList<User>(_list0.size);
@org.apache.thrift.annotation.Nullable User _elem1;
for (int _i2 = 0; _i2 < _list0.size; ++_i2)
{
_elem1 = new User();
_elem1.read(iprot);
struct.users.add(_elem1);
}
iprot.readListEnd();
}
struct.setUsersIsSet(true);
} else {
org.apache.thrift.protocol.TProtocolUtil.skip(iprot, schemeField.type);
}
break;
default:
org.apache.thrift.protocol.TProtocolUtil.skip(iprot, schemeField.type);
}
iprot.readFieldEnd();
}
iprot.readStructEnd();
// check for required fields of primitive type, which can't be checked in the validate method
struct.validate();
}
@Override
public void write(org.apache.thrift.protocol.TProtocol oprot, UserResponse struct) throws org.apache.thrift.TException {
struct.validate();
oprot.writeStructBegin(STRUCT_DESC);
oprot.writeFieldBegin(SUCCESS_FIELD_DESC);
oprot.writeBool(struct.success);
oprot.writeFieldEnd();
if (struct.msg != null) {
oprot.writeFieldBegin(MSG_FIELD_DESC);
oprot.writeString(struct.msg);
oprot.writeFieldEnd();
}
if (struct.users != null) {
oprot.writeFieldBegin(USERS_FIELD_DESC);
{
oprot.writeListBegin(new org.apache.thrift.protocol.TList(org.apache.thrift.protocol.TType.STRUCT, struct.users.size()));
for (User _iter3 : struct.users)
{
_iter3.write(oprot);
}
oprot.writeListEnd();
}
oprot.writeFieldEnd();
}
oprot.writeFieldStop();
oprot.writeStructEnd();
}
}
private static class UserResponseTupleSchemeFactory implements org.apache.thrift.scheme.SchemeFactory {
@Override
public UserResponseTupleScheme getScheme() {
return new UserResponseTupleScheme();
}
}
private static class UserResponseTupleScheme extends org.apache.thrift.scheme.TupleScheme<UserResponse> {
@Override
public void write(org.apache.thrift.protocol.TProtocol prot, UserResponse struct) throws org.apache.thrift.TException {
org.apache.thrift.protocol.TTupleProtocol oprot = (org.apache.thrift.protocol.TTupleProtocol) prot;
java.util.BitSet optionals = new java.util.BitSet();
if (struct.isSetSuccess()) {
optionals.set(0);
}
if (struct.isSetMsg()) {
optionals.set(1);
}
if (struct.isSetUsers()) {
optionals.set(2);
}
oprot.writeBitSet(optionals, 3);
if (struct.isSetSuccess()) {
oprot.writeBool(struct.success);
}
if (struct.isSetMsg()) {
oprot.writeString(struct.msg);
}
if (struct.isSetUsers()) {
{
oprot.writeI32(struct.users.size());
for (User _iter4 : struct.users)
{
_iter4.write(oprot);
}
}
}
}
@Override
public void read(org.apache.thrift.protocol.TProtocol prot, UserResponse struct) throws org.apache.thrift.TException {
org.apache.thrift.protocol.TTupleProtocol iprot = (org.apache.thrift.protocol.TTupleProtocol) prot;
java.util.BitSet incoming = iprot.readBitSet(3);
if (incoming.get(0)) {
struct.success = iprot.readBool();
struct.setSuccessIsSet(true);
}
if (incoming.get(1)) {
struct.msg = iprot.readString();
struct.setMsgIsSet(true);
}
if (incoming.get(2)) {
{
org.apache.thrift.protocol.TList _list5 = iprot.readListBegin(org.apache.thrift.protocol.TType.STRUCT);
struct.users = new java.util.ArrayList<User>(_list5.size);
@org.apache.thrift.annotation.Nullable User _elem6;
for (int _i7 = 0; _i7 < _list5.size; ++_i7)
{
_elem6 = new User();
_elem6.read(iprot);
struct.users.add(_elem6);
}
}
struct.setUsersIsSet(true);
}
}
}
private static <S extends org.apache.thrift.scheme.IScheme> S scheme(org.apache.thrift.protocol.TProtocol proto) {
return (org.apache.thrift.scheme.StandardScheme.class.equals(proto.getScheme()) ? STANDARD_SCHEME_FACTORY : TUPLE_SCHEME_FACTORY).getScheme();
}
}
thrfit的IDL也可以定义枚举,可以指定枚举对应的值,如下:
UserType.thrift
namespace java ltd.klein.thrift.server.api.param
enum UserType {
CONSUMER,
MERCHANT = 1
}
生成的Java的文件如下:
/**
* Autogenerated by Thrift Compiler (0.21.0)
*
* DO NOT EDIT UNLESS YOU ARE SURE THAT YOU KNOW WHAT YOU ARE DOING
* @generated
*/
package ltd.klein.thrift.server.api.param;
@javax.annotation.Generated(value = "Autogenerated by Thrift Compiler (0.21.0)", date = "2024-11-22")
public enum UserType implements org.apache.thrift.TEnum {
CONSUMER(0),
MERCHANT(1);
private final int value;
private UserType(int value) {
this.value = value;
}
/**
* Get the integer value of this enum value, as defined in the Thrift IDL.
*/
@Override
public int getValue() {
return value;
}
/**
* Find a the enum type by its integer value, as defined in the Thrift IDL.
* @return null if the value is not found.
*/
@org.apache.thrift.annotation.Nullable
public static UserType findByValue(int value) {
switch (value) {
case 0:
return CONSUMER;
case 1:
return MERCHANT;
default:
return null;
}
}
}
4.Service
service对应的就是Java的接口,参数可以是基本类型也可以是结构体,同样需要标注顺序。
返回值引用的UserResponse.thrift的UserResponse struct,书写的格式如下:thrfit文件名.struct名
namespace java ltd.klein.thrift.server.api.iface
include '../resp/UserResponse.thrift'
service UserThriftServiceV2 {
UserResponse.UserResponse findUser(1:i32 id);
}
实现类的需要实现UserThriftServiceV2.iface
package ltd.klein.thrift.server.service;
import ltd.klein.thrift.server.api.iface.UserThriftServiceV2;
import ltd.klein.thrift.server.api.resp.User;
import ltd.klein.thrift.server.api.resp.UserResponse;
import ltd.klein.thrift.server.builder.UserResponseBuilder;
import org.apache.thrift.TException;
import org.springframework.stereotype.Service;
import java.util.ArrayList;
import java.util.List;
/**
* @author Klein
*/
@Service
public class UserThriftServiceV2Impl implements UserThriftServiceV2.Iface {
@Override
public UserResponse findUser(int id) throws TException {
if (id == 0) {
User user = new User();
user.setId(id);
user.setNickName("zhangsan");
List<User> users = List.of(user);
return UserResponseBuilder.buildSuccess(users);
}
return UserResponseBuilder.buildSuccess(new ArrayList<>(0));
}
}
到这里对于IDL文件的介绍就结束了,主要介绍了IDL的基本数据类型、容器、结构体与service的编写与用法。如有问题欢迎留言,共同进步。
猜你喜欢
- 2024-12-18 吊打 ThreadLocal,谈谈FastThreadLocal为啥能这么快?
- 2024-12-18 分布式锁中的王者方案 - Redisson
- 2024-12-18 你管这玩意儿叫高并发? 什么叫高并发
- 2024-12-18 系统数据实时同步方案一落地 系统间数据同步解决方案
- 2024-12-18 分布式锁工具:Redisson 分布式锁 redis zookeeper
- 2024-12-18 Spring Cloud Circuit Breaker快速入门Demo
- 2024-12-18 BitMap是啥?脑袋一下空白? bitmap文件头
- 2024-12-18 一亿个8位数字,用什么排序方法 一亿个8位数字,用什么排序方法最好
- 2024-12-18 Java基础-数据类型和数据结构,初阶小白看过来~
- 2024-12-18 10张图带你搞定高并发之网络IO模型
- 最近发表
- 标签列表
-
- gitpush (61)
- pythonif (68)
- location.href (57)
- tail-f (57)
- pythonifelse (59)
- deletesql (62)
- c++模板 (62)
- css3动画 (57)
- c#event (59)
- linuxgzip (68)
- 字符串连接 (73)
- nginx配置文件详解 (61)
- html标签 (69)
- c++初始化列表 (64)
- exec命令 (59)
- canvasfilltext (58)
- mysqlinnodbmyisam区别 (63)
- arraylistadd (66)
- node教程 (59)
- console.table (62)
- c++time_t (58)
- phpcookie (58)
- mysqldatesub函数 (63)
- window10java环境变量设置 (66)
- c++虚函数和纯虚函数的区别 (66)