Qt中QFile、QByteArray QDataStream和QTextStream区别及示例

news2024/11/16 2:49:59

在Qt中,QFileQByteArrayQDataStreamQTextStream是常用的文件和数据处理类。

主要功能和区别

  1. QFile

QFile是用于读写文本和二进制文件以及资源的I/O设备。可以单独使用QFile,或者更方便地与QTextStreamQDataStream一起使用。

通常在构造函数中传入文件名,但也可以随时使用setFileName()进行设置。QFile期望文件分隔符为’/‘,不论操作系统为何。不支持使用分隔符(例如’')。

可以使用exists()检查文件是否存在,并使用remove()删除文件。(更高级的文件系统相关操作由QFileInfoQDir提供。)

文件使用open()打开,使用close()关闭,并使用flush()刷新。通常使用QDataStreamQTextStream进行数据的读取和写入,但也可以使用QIODevice继承的函数read()readLine()readAll()write()进行操作。QFile还继承了getChar()、putChar()和ungetChar(),可以逐个字符地进行操作。

使用size()可以获取文件的大小。可以使用pos()获取当前文件位置,或者使用seek()移动到新的文件位置。如果已经到达文件末尾,atEnd()返回true。

  1. QByteArray

QByteArray类提供了一个字节数组。
QByteArray可用于存储原始字节(包括’\0’)和传统的8位’\0’终止的字符串。使用QByteArray比使用const char *更方便。在幕后,它始终确保数据后跟’\0’终止符,并使用隐式共享(写时复制)来减少内存使用和避免无谓的数据复制。
除了QByteArray,Qt还提供了QString类用于存储字符串数据。对于大多数情况,应使用QString类。它存储16位Unicode字符,使得在应用程序中存储非ASCII/非Latin-1字符变得容易。此外,在Qt API中广泛使用QString。QByteArray适用的两种主要情况是当需要存储原始二进制数据时,以及内存保护至关重要的情况(例如,在Qt for Embedded Linux中)。
初始化QByteArray的一种方法是将const char *传递给其构造函数。例如,以下代码创建一个包含数据"Hello"的大小为5的字节数组:

  QByteArray ba("Hello");

尽管size()为5,字节数组还在末尾保留了一个额外的’\0’字符,以便在使用需要指向底层数据的指针的函数(例如调用data())时,所指向的数据保证是以’\0’结尾的。
QByteArray会对const char *数据进行深拷贝,因此您可以稍后进行修改而不会出现副作用。(如果出于性能原因,您不想对字符数据进行深拷贝,请改用QByteArray::fromRawData()。)

另一种方法是使用resize()设置数组的大小,并逐个字节初始化数据。QByteArray使用基于0的索引,就像C++数组一样。要访问特定索引位置的字节,可以使用operator。对于非const字节数组,operator返回一个字节的引用,可以在赋值的左侧使用。例如:

  QByteArray ba;
  ba.resize(5);
  ba[0] = 0x3c;
  ba[1] = 0xb8;
  ba[2] = 0x64;
  ba[3] = 0x18;
  ba[4] = 0xca;

对于只读访问,使用at()的替代语法:

  for (int i = 0; i < ba.size(); ++i) {
      if (ba.at(i) >= 'a' && ba.at(i) <= 'f')
          cout << "Found character in range [a-f]" << endl;
  }

at()可能比operator更快,因为它永远不会导致深拷贝发生。
要一次提取多个字节,请使用left()right()mid()
QByteArray可以嵌入’\0’字节。size()函数始终返回整个数组的大小,包括嵌入的’\0’字节,但不包括QByteArray添加的终止’\0’。例如:

  QByteArray ba1("ca\0r\0t");
  ba1.size();                     // 返回2。
  ba1.constData();                // 返回带有终止\0的"ca"。

  QByteArray ba2("ca\0r\0t", 3);
  ba2.size();                     // 返回3。
  ba2.constData();                // 返回带有终止\0的"ca\0"。

  QByteArray ba3("ca\0r\0t", 4);
  ba3.size();                     // 返回4。
  ba3.constData();                // 返回带有终止\0的"ca\0r"。

  const char cart[] = {'c', 'a', '\0', 'r', '\0', 't'};
  QByteArray ba4(QByteArray::fromRawData(cart, 6));
  ba4.size();                     // 返回6。
  ba4.constData();                // 返回不带终止\0的"ca\0r\0t"。

如果要获取数据长度,直到但不包括第一个’\0’字符,可以在字节数组上调用qstrlen()。
在调用resize()后,新分配的字节的值是未定义的。要将所有字节设置为特定值,请调用fill()。
要获取指向实际字符数据的指针,请调用data()或constData()。这些函数返回指向数据开头的指针。该指针保证在QByteArray上调用非const函数之前保持有效。还保证除非QByteArray是从原始数据创建的,否则数据以’\0’字节结尾。QByteArray自动提供此’\0’字节,并且不计入size()中。
QByteArray提供了以下基本函数来修改字节数据:append()、prepend()、insert()、replace()和remove()。例如:

  QByteArray x("and");
  x.prepend("rock ");         // x == "rock and"
  x.append(" roll");          // x == "rock and roll"
  x.replace(5, 3, "&");       // x == "rock & roll"

replace()remove()函数的前两个参数是要开始擦除的位置和应该擦除的字节数。
当将数据追加到非空数组时数组将被重新分配并将新数据复制到其中。您可以通过调用reserve()来避免此行为,它预分配一定数量的内存。您还可以调用capacity()来了解QByteArray实际分配了多少内存。追加到空数组中的数据不会被制。
经常要求从字节数组中删除空白字符(‘\n’、‘\t’、''等)。如果要从QByteArray两端删除空白字符,请使用trimmed()。如果要从QByteArray两端删除空白字符,并且在字节数组中多个连续的空白字符替换为单个空格字符,请使用simplified()。
如果要在QByteArray中查找特定字符或子字符串的所有出现,可以使用indexOf()或lastIndexOf()。前者从给定的索引位置向前搜索,后者向后搜索。如果找到了字符或子字符串,两者都返回其索引位置;否则,它们返回-1。例如,以下是查找特定子字符串的典型循环:

  QByteArray ba("We must be <b>bold</b>, very <b>bold</b>");
  int j = 0;
  while ((j = ba.indexOf("<b>", j)) != -1) {
      cout << "Found <b> tag at index position " << j << endl;
      ++j;
  }

如果只是想检查一个QByteArray是否包含特定字符或子字符串,请使用contains()。如果想要找出特定字符或子字符串在字节数组中出现的次数,请使用count()。如果想要用另一个值替换所有出现的特定值,请使用带有两个参数的replace()重载之一。
可以使用诸如operator<(), operator<=(), operator==(), operator>=()等重载运算符来比较QByteArray。比较仅基于字符的数值值,非常快速,但不符合人的预期。对于排序用户界面字符串,QString::localeAwareCompare()是更好的选择。

由于历史原因,QByteArray区分空字节数组和空字节数组。空字节数组是通过使用QByteArray的默认构造函数或通过将(const char *)0传递给构造函数进行初始化的字节数组。大小为0的任何字节数组都是空字节数组。空字节数组不一定是null字节数组:

  QByteArray().isNull();          // 返回true
  QByteArray().isEmpty();         // 返回true

  QByteArray("").isNull();        // 返回false
  QByteArray("").isEmpty();       // 返回true

  QByteArray("abc").isNull();     // 返回false
  QByteArray("abc").isEmpty();    // 返回false

除了isNull()函数外,所有其他函数都将null字节数组视为与空字节数组相同。例如,对于null字节数组,data()返回指向 ‘\0’ 字符的指针(非空指针),并且QByteArray()与QByteArray(“”)相等。我们建议始终使用isEmpty(),避免使用isNull()。

  1. QDataStream

QDataStream类提供了将二进制数据序列化到QIODevice的功能。
数据流是一个二进制编码信息流,完全独立于主机计算机的操作系统、CPU或字节顺序。例如,由Windows PC编写的数据流可以被运行Solaris系统的Sun SPARC读取。
您还可以使用数据流来读写原始的未编码的二进制数据。如果您需要一个"解析"输入流,请参见QTextStream
QDataStream类实现了C++基本数据类型(如char、short、int、char *等)的序列化。更杂数据的序列化是通过将数据分解为基本单元来完成的。

数据流与QIODevice紧密配合。QIODevice代表可以从中读取数据或向其写入数据的输入/输出介质。QFile类是一个I/O设备的示例。

示例(将二进制数据写入数据流):

  QFile file("file.dat");
  file.open(QIODevice::WriteOnly);
  QDataStream out(&file);   // 将数据序列化到文件中
  out << QString("the answer is");   // 序列化字符串
  out << (qint32)42;        // 序列化整数

示例(从数据流读取二进制数据):

  QFile file("file.dat");
  file.open(QIODevice::ReadOnly);
  QDataStream in(&file);    // 从文件中读取已序列化的数据
  QString str;
  qint32 a;
  in >> str >> a;           // 提取"the answer is"和42

每个写入流中的项目都以预定义的二进制格式写入,该格式取决于项目的类型。支持的Qt类型包括QBrush、QColor、QDateTime、QFont、QPixmap、QString、QVariant等。有关支持数据流的所有Qt类型的完整列表,请参见序列化Qt数据类型。
对于整数,最好始终将其强制转换为Qt整数类型进行写入,并读取回到相同的Qt整数类型中。这样可以确保您获得所需的整数大小,并将您与编译器和平台的差异隔离开来。
举个例子,一个char *字符串被写为一个32位整数,该整数等于字符串长度,包括’\0’字节,后跟字符串的所有字符,包括’\0’字节。当读取一个char *字符串时,首先读取4个字节以创建32位长度值,然后读取相同数量的字符来创建包含’\0’终止符的char *字符串。
初始的I/O设备通常在构造函数中设置,但可以使用setDevice()进行更改。如果已到达数据的末尾(或者没有设置I/O设备),atEnd()将返回true。

版本控制
自Qt 1.0以来,QDataStream的二进制格式已发生了演变,并且可能会继续着Qt的变化而发展。在输入或输出复杂类型,非常重要的一点是确保在读取和写入时使用相同版本的(version())。如果您需要前向和后向兼容性,您可以在应用程序中硬编码版本号:

  stream.setVersion(QDataStream::Qt_4_0);

如果您正在生成一种新的二进制数据格式,例如您的应用程序创建的文档文件格式,您可以使用QDataStream以便于移植的格式写入数据。通常,您会写一个简短的头部,包含一个标识字符串和一个版本号,以便为将来的扩展留出空间。例如:

  QFile file("file.xxx");
  file.open(QIODevice::WriteOnly);
  QDataStream out(&file);

  // 写入包含"魔术数字"和版本号的头部
  out << (quint32)0xA0B0C0D0;
  out << (qint32)123;

  out.setVersion(QDataStream::Qt_4_0);

  //入数据
  out << lots_of_interest_data;

然后使用以下代码进行读取:

  QFile file("file.xxx");
  file.open(QIODevice::ReadOnly);
  QDataStream (&infile);

  // 读取并检查头部
  quint32 magic;
  in >> magic;
  if (magic != 0xA0B0C0D0)
      return XXX_BAD_FILE_FORMAT;

  // 读取版本号
  qint32 version;
  in >> version;
  if (version < 100)
      return XXX_FILE_TOO_OLD;
  if (version > 123)
      return XXX_BAD_FILE_TOO_NEW;

  if (version <= 110)
      in.setVersion(QDataStream::Qt_3_2);
  else
      in.setVersion(QDataStream::Qt_4_0);

  // 取数据
  in >> lots_of_interesting_data;
  if (version >= 120)
      in >> data_new_in_XXX_version_1_2;
  in >> other_interesting_data;

在序列化数据时,您可以选择使用哪种字节顺序。默认设置为big endian(最高位在前)。将其更改为little endian会破坏可移植性(除非读取方也更改为little endian)。我们建议保持此设置,除非有特殊需求。
读取和写入原始二进制数据
您可能希望直接从数据流中读取/写入自己的原始二进制数据。可以使用readRawData()将数据从流中读取到预先分配的char *中。类似地,可以使用writeRawData()将数据写入到流中。注意,数据的任何编码/解码都必须由您自己完成。
另一对类似的函数是readBytes()writeBytes()。与原始函数的不同之处在于:readBytes()读取一个quint32,将其视为要读取的长度,然后将该数量的字节读入预先分配的char *中;writeBytes()写入包含数据长度的quint32,然后是数据本身。请注意,数据的任何编码/解码(除了长度quint32之外)都必须由您自己完成。

  1. QTextStream

QTextStream类提供了一个方便的界面,用于读写文本。
QTextStream可以操作QIODeviceQByteArrayQString。使用QTextStream的流操作符,您可以方便地读写单词、行和数字。对于生成文本,QTextStream支持字段填充和对齐的格式选项,以及数字的格式化。示例:

  QFile data("output.txt");
  if (data.open(QFile::WriteOnly | QFile::Truncate)) {
      QTextStream out(&data);
      out << "Result: " << qSetFieldWidth(10) << left << 3.14 << 2.7;
      // 输出 "Result: 3.14      2.7       "
  }

通常还会使用QTextStream来读取控制台输入和写入控制台输出。QTextStream具有区域设置意识,并将自动使用正确的编解码器对标准输入进行解码。示例:

  QTextStream stream(stdin);
  QString line;
  while (stream.readLineInto(&line)) {
      ...
  }

除了使用QTextStream的构造函数外,还可以通过调用setDevice()或setString()设置QTextStream操作的设备或字符串。可以通过调用seek()来定位到某个位置,当没有数据可以读取时,调用atEnd()将返回true。如果调用flush(),QTextStream将会将其写缓冲区中的所有数据清空到设备,并调用设备上的flush()。
在内部,QTextStream使用基于Unicode的缓冲区,QTextCodec用于自动支持不同的字符集。默认情况下,读取和写入使用QTextCodec::codecForLocale(),但也可以通过调用setCodec()来设置编解码器。还支持自动Unicode检测。当启用此功能时(默认行为),QTextStream将检测UTF-16或UTF-32的BOM(字节顺序标记),并在读取时切换到适当的UTF编解码器。QTextStream默认不会写入BOM,但可以通过调用setGenerateByteOrderMark(true)启用。当QTextStream直接操作QString时,编解码器被禁用。

使用QTextStream读取文本文件有三种常见方式:

  • 按块读取,通过调用readLine()或readAll()。
  • 逐个单词读取。QTextStream支持流式读取到QString、QByteArray和char*缓冲区中。单词由空格分隔,并自动跳过前导空格。
  • 逐个字符读取,通过流式读取到QChar或char类型中。这种方法通常用于方便地处理输入,无论字符编码和行尾的语义如何。要跳过空格,调用skipWhiteSpace()。\

由于文本流使用缓冲区,您不应该使用超类的实现从流中读取。例如,如果您有一个QFile并直接使用QFile::readLine()从中读取,而不使用流,文本流的内部位置将与文件的位置不同步。
默认情况下,当从文本流中读取数字时,QTextStream会自动检测数字的进制表示。例如,如果数字以"0x"开头,它被认为是十六进制形式。如果以1-9的数字开头,它被认为是十进制形式,依此类推。您可以通过调用setIntegerBase()设置整数的进制,从而禁用自动检测。示例:

  QTextStream in("0x50 0x20");
  int firstNumber, secondNumber;

  in >> firstNumber;             // firstNumber == 80
  in >> dec >> secondNumber;     // secondNumber == 0

  char ch;
  in >> ch;                      // ch == 'x'

QTextStream支持许多用于生成文本的格式选项。您可以通过调用setFieldWidth()和setPadChar()设置字段宽度和填充字符。使用setFieldAlignment()设置每个字段内的对齐方式。对于实数,可以调用setRealNumberNotation()和setRealNumberPrecision()设置所生成数字的表示(智能表示、科学表示、定点表示)和精度(数字位数)。还可以通过setNumberFlags()设置其他一些数字格式选项。

与标准C++库中的一样,QTextStream还定义了几个全局操纵函数:

操纵函数描述
bin与setIntegerBase(2)相同。
oct与setIntegerBase(8)相同。
dec与setIntegerBase(10)相同。
hex与setIntegerBase(16)相同。
showbase与setNumberFlags(numberFlags()
forcesign与setNumberFlags(numberFlags()
forcepoint与setNumberFlags(numberFlags()
noshowbase与setNumberFlags(numberFlags() & ~ShowBase)相同。
noforcesign与setNumberFlags(numberFlags() & ~ForceSign)相同。
noforcepoint与setNumberFlags(numberFlags() & ~ForcePoint)相同。
uppercasebase与setNumberFlags(numberFlags()
uppercasedigits与setNumberFlags(numberFlags()
lowercasebase与setNumberFlags(numberFlags() & ~UppercaseBase)相同。
lowercasedigits与setNumberFlags(numberFlags() & ~UppercaseDigits)相同。
fixed与setRealNumberNotation(FixedNotation)相同。
scientific与setRealNumberNotation(ScientificNotation)同。
left与setFieldAlignment(AlignLeft)相同。
right与setFieldAlignment(AlignRight)相同。
center与setFieldAlignment(AlignCenter)相同。
endl与operator<<(‘\n’)和flush()相同。
flush与flush()相同。
reset与reset()相同。
ws与skipWhiteSpace()相同。
bom与setGenerateByteOrderMark(true)相同。

此外,Qt提供了三个带有参数的全局操纵符:qSetFieldWidth()qSetPadChar()qSetRealNumberPrecision()

总结就是,QFile用于文件的读写操作,QByteArray用于处理二进制数据,QDataStream用于二进制数据的序列化和反序列化,而QTextStream用于文本数据的读写和处理。它们各自有不同的功能和适用场景,根据具体需求选择合适的类进行操作。

用法及示例

当使用Qt中的QFile、QByteArray、QDataStream和QTextStream时,可以按照以下示例来使用它们的成员函数:

  1. QFile的使用示例:
#include <QFile>
#include <QDebug>
#include <QApplication>

int main(int argc, char *argv[])
{
	QApplication a(argc, argv);
    // 创建一个QFile对象
    QFile file("data.txt");

 	// 打开文件以进行写入
    if (file.open(QIODevice::WriteOnly))
    {
        // 写入数据到文件
        QString data = "Hello, World!";
        file.write(data.toUtf8());

        // 关闭文件
        file.close();
    }

    // 打开文件以进行读取
    if (file.open(QIODevice::ReadOnly))
    {
        // 读取文件中的数据
        QByteArray data = file.readAll();
        
        // 将字节数组转换为字符串并输出
        QString str(data);
        qDebug() << str;

        // 关闭文件
        file.close();
    }

    return a.exec();
}
  1. QByteArray的使用示例:
#include <QByteArray>
#include <QDebug>
#include <QApplication>

int main(int argc, char *argv[])
{
	QApplication a(argc, argv);
    // 创建一个QByteArray对象
    QByteArray byteArray;

    // 向字节数组添加数据
    byteArray.append("Hello");
    byteArray.append(" ");
    byteArray.append("World!");

    // 输出字节数组中的数据
    qDebug() << byteArray;

    // 清空字节数组
    byteArray.clear();

    return a.exec();
}
  1. QDataStream的使用示例:
#include <QDataStream>
#include <QFile>
#include <QDebug>
#include <QApplication>

int main(int argc, char *argv[])
{
	QApplication a(argc, argv);
    // 创建一个QFile对象
    QFile file("data.bin");

    // 打开文件以进行写入
    if (file.open(QIODevice::WriteOnly))
    {
        // 创建一个QDataStream对象,并传入QFile对象
        QDataStream stream(&file);
        
        // 写入整数到流中
        int value = 42;
        stream << value;

        // 关闭文件
        file.close();
    }

    // 打开文件以进行读取
    if (file.open(QIODevice::ReadOnly))
    {
        // 创建一个QDataStream对象,并传入QFile对象
        QDataStream stream(&file);

        // 从流中读取整数
        int value;
        stream >> value;

        // 输出取到的整数
        qDebug() << value;

        //闭文件
        file.close();
    }

    return a.exec();
}
  1. QTextStream的使用示例:
#include <QTextStream>
#include <QFile>
#include <QDebug>
#include <QApplication>

int main(int argc, char *argv[])
{
	QApplication a(argc, argv);
    // 创建一个QFile对象
    QFile file("data.txt");

    // 打开文件以进行写
    if (file.open(QIODevice::WriteOnly | QIODevice::Text))
    {
        // 创建一个QTextStream对象,并传入QFile对象
        QTextStream stream(&file);

        // 写入文本到流中
        stream << "Hello, World!";

        // 关闭文件
        file.close();
    }

    // 打开文件以进行读取
    if (file.open(QIODevice::ReadOnly | QIODevice::Text))
    {
        // 创建一个QTextStream对象,并传入QFile对象
        QTextStream stream(&file);

        // 从流中读取文本
        QString text = stream.readAll();

        // 输出读取到的文本
        qDebug() << text;

        // 关闭文件
        file.close();
    }

    return a.exec();
}

示例分别使用QFile来读写文件、使用QByteArray处理字节数组、使用QDataStream进行二进制数据的序列化和反序列化、以及使用QTextStream进行文本数据的读写处理。

综合用法

#include <QFile>
#include <QByteArray>
#include <QDataStream>
#include <QTextStream>
#include <QDebug>
#include <QApplication>


int main(int argc, char *argv[])
{
	QApplication a(argc, argv);
    // 创建一个QByteArray对象
    QByteArray byteArray;

    // 向字节数组添加数据
    byteArray.append("Hello, World!");

    // 创建一个QFile对象
    QFile file("data.bin");

     // 打开文件以进行写入
    if (file.open(QIODevice::WriteOnly))
    {
        // 创建一个QDataStream对象,并传入QFile对象
        QDataStream dataStream(&file);

        // 将字节数组写入流
        dataStream << byteArray;

        // 关闭文件
        file.close();
    }

    // 打开文件以进行读取
    if (file.open(QIODevice::ReadOnly))
    {
        // 创建一个QDataStream对象,并传入QFile对象
        QDataStream dataStream(&file);

        // 创建一个新的QByteArray对象
        QByteArray newDataArray;

        // 从流中读取字节数组
        dataStream >> newDataArray;

        // 创建一个QTextStream对象
        QTextStream textStream(&newDataArray);

        // 读取字符串数据
        QString text = textStream.readAll();

        // 输出读取到的字符串
        qDebug() << text;

        // 关闭文件
        file.close();
    }

    return a.exec();
}

输出结果

在这里插入图片描述
示例中,首先创建一个QByteArray对象并向它添加数据。然后,将这个字节数组写入到一个文件中(使用QDataStream),并将文件关闭。接下来,再次打开文件并使用QDataStream从中读取字节数组。最后,使用QTextStream读取字节数组中的字符串数据,并将其输出到控制台。

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

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

相关文章

开关电源检测的技术标准和安全标准是什么?纳米软件为您介绍

开关电源总体技术标准 1. 外观&#xff1a;元器件排列整齐、美观、结构合理&#xff0c;焊点均匀饱满、明亮、光滑、无尖刺&#xff0c;焊接牢固。PCB板铜条无脱落、外露、无毛刺、飞边、变形。 2. 输入电压&#xff1a;110VAC/DC或220VAC/DC或380VAC三相20%;或85~264VAC全范围…

Go学习第六章——系统函数与错误处理

Go系统函数与错误处理 1 字符串相关系统函数2 时间和日期相关的函数2.1 Now函数2.2 日期的格式化 3 内置函数4 错误处理4.1 基本使用4.2 自定义错误 1 字符串相关系统函数 下面给出20个常见的函数&#xff0c;不需要导包&#xff0c;直接可以用 func main() {// 1.统计字符串…

Synchronized同步锁

synchronized 一&#xff0c;介绍 Java中的synchronized关键字用于实现线程同步&#xff0c;可以修饰方法或代码块。 1. 修饰方法&#xff1a;当一个方法被synchronized修饰时&#xff0c;只有获得该方法的锁的线程才能执行该方法。其他线程需要等待锁的释放才能执行该方法。…

Java Netty - Buffer类

Buffer类 当应用程序进行数据传输的时候&#xff0c;往往需要使用缓冲区&#xff0c;常用的缓存区就是JDK NIO类库提供的 java.nio.Buffer&#xff1b; NIO的Buffer本质上是一个内存块&#xff0c;既可以写入数据&#xff0c;也可以从中读取数据&#xff1b; 其中&#xff0c;…

IP地址定位是什么?有哪些优缺点?

IP地址定位是一种用于确定设备或用户地理位置的方法&#xff0c;具有一些明显的优点和缺点。以下是IP地址定位的优缺点&#xff1a; 优点&#xff1a; 广泛适用性&#xff1a; IP地址定位适用于几乎所有与互联网连接的设备&#xff0c;包括计算机、智能手机、平板电脑和物联网…

全链路压测专题---2、全链路压测架构和技术

如何开展全链路压测 业务模型梳理 首先应该将核心业务和非核心业务进行拆分&#xff0c;确认流量高峰针对的是哪些业务场景和模块&#xff0c;针对性的进行扩容准备梳理出对外的接口&#xff1a;使用MOCK&#xff08;模拟&#xff09;方式做挡板千万不要污染正常数据&#xf…

包装类知识.JDK7,JDK8相关时间类练习

包装类 包装类:基本数据类型对应的引用类型 JDK5之前的Integer包装类 如下 了解即可 i5传入的值 第一个是字符串,第二个是进制数.为83是因为再打印的时将123看成8进制了 然后打印转化为10进制 区别对比 public static void main(String[] args) {Integer i1 Integer.valu…

5年经验之谈 —— 手把手教你接口性能测试之工具篇!

本文是我们《手把手教你接口性能测试》系列文章中的中篇&#xff0c;主要介绍软件测试工作中常用的一些接口测试工具。 一、接口都有哪些类型&#xff1f; 1、接口一般分为两种&#xff1a; 1.程序内部的接口 2.系统对外的接口 程序内部的接口 方法与方法之间&#xff0c;模…

最靠谱的nvm安装教程 for mac

nvm是node 版本管理工具 一、前提 1、保证自己的mac电脑上没安装node环境 2、保证自己的电脑上有安装git&#xff0c;不然下载nvm时会报错。 git下载可以参考&#xff1a; 1&#xff09;佛系安装&#xff1a;通过提示下载 XCode 2&#xff09;brem安装 3&#xff09;终极大法&…

Kubernetes - 一键安装部署 K8S(附:Kubernetes Dashboard)

问题描述 不知道大伙是如何安装 K8s&#xff0c;特别还是集群的时候&#xff0c;我上一次安装搭建的时候&#xff0c;那个恶心到我了&#xff0c;真的是一步一个脚印走完整个搭建流程&#xff0c;爬了不少坑。 于是&#xff0c;才有了今天的文章&#xff0c;到底有没有可以一…

在mybatis的xml中使用枚举来做判断条件

1.枚举类 import com.baomidou.mybatisplus.annotation.IEnum; import com.fasterxml.jackson.annotation.JsonCreator; import com.fasterxml.jackson.annotation.JsonValue; import com.shinkeer.common.utils.StringUtils;import java.util.HashMap; import java.util.Map;…

日志logback详解

该篇博客是我参考了很多博客总结出来的&#xff0c;内容很多&#xff0c;你能看完不睡着算你厉害&#xff01; 一、日志框架发展史 二、日志规范 三、日志级别 四、Logback 4.1 Maven引入 4.2 logback-spring.xml是在哪里加载的&#xff1f; 4.3 logback.xml和logback-s…

err -110 while initial SD card failed enmmm否系解决方案

1.情况表述 之前把 sd卡从小容量换成了大容量 且运行正常 过了一两个月 发现启动失败 报错&#xff1a;mmc0: error -110 whilst initialising SD card 给出报错图片 2.相关解决方案 给出链接 讲的很有道理&#xff0c;但是 我在bash中没有找到对应内核的驱动文件 SD卡…

抖音SEO优化怎么做?详细的方法来了,搭配批量剪辑让效果更优

抖音作为一个非常受欢迎的短视频平台&#xff0c;无论是用户还是内容创作者&#xff0c;体量都越来越大&#xff0c;这意味着竞争也越来越大&#xff0c;想要在抖音上获得更多的曝光和关注&#xff0c;做好抖音 SEO 优化是一门必修课。 抖音 SEO 优化是指优化我们发布到抖音平…

AMEYA360:瑞萨电子转矩控制解决方案加强工业自动化

随着科技的不断进步和全球制造业的竞争加剧&#xff0c;越来越多的企业开始投资和采用自动化技术&#xff0c;以提高生产效率、降低成本、改善产品质量并确保过程安全。而为协同工作完成复杂任务&#xff0c;确保高质量的生产结果。工业自动化设备对精度控制要求非常高&#xf…

Redis底层核心数据结构详解

文章目录 一、深入String&#xff08;SDS&#xff09;1. 字符串简介2. SDS存在的意义3. SDS结构设计4. SDS与C字符串的区别4.1 常数复杂度获取字符串长度4.2 杜绝缓冲区溢出4.3 二进制安全4.4 SDS API 5 小结 二、深入List (QuickList)1. 链表节点结构设计2. Redis的链表实现的…

linux驱动开发led绑定亮灯

head.h 应用程序 驱动程序 结果

LabVIEW基于机器视觉的钢轨表面缺陷检测系统

LabVIEW基于机器视觉的钢轨表面缺陷检测系统 机器视觉检测技术和LabVIEW软件程序&#xff0c;可以实现轨道工件的表面质量。CMOS彩色工业相机采集的图像通过图像预处理、图像阈值分割、形态分析、特征定位和图案匹配进行处理和分析。图形显示界面采用LabVIEW软件编程设计&…

【大模型的一些基本结论】

这里写自定义目录标题 LLama的一些基本结论 各个论文中给出一些观察显现&#xff0c;我们比摘要更简略地摘要一些文本大模型大佬地基本结论和观察到的现象 LLama的一些基本结论 由于大模型要作为服务&#xff0c;因而推理时间更重要。一个较小的、训练时间较长的模型最终会在…

GLIP,FLIP论文阅读

Scaling Language-Image Pre-training via Masking&#xff08;FLIP&#xff0c;2023&#xff09;&#x1f44d; 贡献&#xff1a; 1.图像端引入MAE的随机MASK&#xff0c;image encoder只处理未mask的patches&#xff08;和之前的MAE方法一致&#xff09;&#xff0c;减少了输…