- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我将下面的代码拼凑在一起,没有做任何复杂的事情——只是创建一个 byte[] 变量,将它写入 Cassandra 中的一个 blob 字段(v1.2,通过新的 Datastax CQL 库),然后再次读回。
当我放入它时它有 3 个元素长,而当我读回它时它有 84 个元素长...!这意味着我实际尝试做的事情(序列化 Java 对象)在尝试时失败并出现 org.apache.commons.lang.SerializationException: java.io.StreamCorruptedException: invalid stream header: 81000008
错误再次反序列化。
下面是一些演示我的问题的示例代码:
import java.nio.ByteBuffer;
import org.apache.commons.lang.SerializationUtils;
import com.datastax.driver.core.BoundStatement;
import com.datastax.driver.core.Cluster;
import com.datastax.driver.core.Host;
import com.datastax.driver.core.Metadata;
import com.datastax.driver.core.PreparedStatement;
import com.datastax.driver.core.ResultSet;
import com.datastax.driver.core.Row;
import com.datastax.driver.core.Session;
public class TestCassandraSerialization {
private Cluster cluster;
private Session session;
public TestCassandraSerialization(String node) {
connect(node);
}
private void connect(String node) {
cluster = Cluster.builder().addContactPoint(node).build();
Metadata metadata = cluster.getMetadata();
System.out.printf("Connected to %s\n", metadata.getClusterName());
for (Host host: metadata.getAllHosts()) {
System.out.printf("Datacenter: %s; Host: %s; Rack: %s\n",
host.getDatacenter(), host.getAddress(), host.getRack());
}
session = cluster.connect();
}
public void setUp() {
session.execute("CREATE KEYSPACE test_serialization WITH replication = {'class':'SimpleStrategy', 'replication_factor':1};");
session.execute("CREATE TABLE test_serialization.test_table (id text PRIMARY KEY, data blob)");
}
public void tearDown() {
session.execute("DROP KEYSPACE test_serialization");
}
public void insertIntoTable(String key, byte[] data) {
PreparedStatement statement = session.prepare("INSERT INTO test_serialization.test_table (id,data) VALUES (?, ?)");
BoundStatement boundStatement = new BoundStatement(statement);
session.execute(boundStatement.bind(key,ByteBuffer.wrap(data)));
}
public byte[] readFromTable(String key) {
String q1 = "SELECT * FROM test_serialization.test_table WHERE id = '"+key+"';";
ResultSet results = session.execute(q1);
for (Row row : results) {
ByteBuffer data = row.getBytes("data");
return data.array();
}
return null;
}
public static boolean compareByteArrays(byte[] one, byte[] two) {
if (one.length > two.length) {
byte[] foo = one;
one = two;
two = foo;
}
// so now two is definitely the longer array
for (int i=0; i<one.length; i++) {
//System.out.printf("%d: %s\t%s\n", i, one[i], two[i]);
if (one[i] != two[i]) {
return false;
}
}
return true;
}
public static void main(String[] args) {
TestCassandraSerialization tester = new TestCassandraSerialization("localhost");
try {
tester.setUp();
byte[] dataIn = new byte[]{1,2,3};
tester.insertIntoTable("123", dataIn);
byte[] dataOut = tester.readFromTable("123");
System.out.println(dataIn);
System.out.println(dataOut);
System.out.println(dataIn.length); // prints "3"
System.out.println(dataOut.length); // prints "84"
System.out.println(compareByteArrays(dataIn, dataOut)); // prints false
String toSave = "Hello, world!";
dataIn = SerializationUtils.serialize(toSave);
tester.insertIntoTable("toSave", dataIn);
dataOut = tester.readFromTable("toSave");
System.out.println(dataIn.length); // prints "20"
System.out.println(dataOut.length); // prints "104"
// The below throws org.apache.commons.lang.SerializationException: java.io.StreamCorruptedException: invalid stream header: 81000008
String hasLoaded = (String) SerializationUtils.deserialize(dataOut);
System.out.println(hasLoaded);
} finally {
tester.tearDown();
}
}
}
看起来正确的东西进入了数据库:
cqlsh:flight_cache> select * from test_serialization.test_table;
id | data
--------+--------------------------------------------
123 | 0x010203
toSave | 0xaced000574000d48656c6c6f2c20776f726c6421
cqlsh:flight_cache>
所以它看起来像是读取而不是写入二进制数据时的错误。谁能给我任何关于我做错了什么的指示?
最佳答案
问题几乎可以肯定是因为 ByteBuffer.array() 返回的数组是完整的后备数组,但数据可能只包含在其中的一部分。
返回的有效数据从 ByteBuffer.arrayOffset() 开始,长度为 ByteBuffer.remaining()。要获取仅包含有效数据的字节数组,请在 readFromTable 中使用以下代码:
byte[] result = new byte[data.remaining()];
data.get(result);
然后您的数据就在结果中,您可以返回它。
关于java - 通过 ByteBuffer 和 CQL 3 将 Java 对象序列化到 Cassandra 1.2,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17282361/
我有一个空的 byteBuffer 分配为 data = ByteBuffer.allocateDirect(layerSize(0, faces - 1, 0, levels - 1) * laye
虽然存在 ByteBuffer.put(ByteBuffer) 方法,但 ByteBuffer.get(ByteBuffer) 似乎丢失了?我应该如何从较大的 ByteBuffer 读取较小的 Byt
我在 Dart Web 应用程序中加载二进制文件,其中包含压缩音频数据作为二进制数据的一部分。当数据通过 http 请求加载时,它作为一个 ByteBuffer 出现,我理解它是 Dart 对 JS
假设我有一个列表/数组列表或字节缓冲区数组(List 或 ByteBuffer[]) 是否可以由此直接从上述数组中获取字节,而无需遍历所有项或计算它们的总大小?我正在寻找这样的东西: ByteBuff
我有一个或多个 ByteBuffer,其中包含一条消息的部分内容。现在我想阅读此消息,但我不想将 N ByteBuffer 复制到一个中。我的解析器需要一个包含完整消息的 ByteBuffer,但我的
如果已知它是其他缓冲区? 我知道这可以通过非直接数组支持的 ByteBuffer 使用 arrayOffset() 方法来完成,如下所示: int getRelativeBufferOffset(By
如何在 Dart 中将 ByteBuffer 的字节快速复制到另一个更大的 ByteBuffer(以非零偏移量)? 有一些缓慢的方法可以做到这一点。一种是将每个都转换为 Uint8List 并一次复制
嘿嘿, ByteBuffers 以及 netty 的 ByteBuff 使用索引来存储它们当前“所在”的位置。在我的应用程序开始时,我加载 ByteBuffers/ByteBuffs 中的多个文件以便
我知道 flip() 将当前缓冲区位置设置为 0,并将限制设置为上一个缓冲区位置,而 rewind() 只是将当前缓冲区位置设置为 0。 在下面的代码中,我使用 rewind() 或 flip() 得
我有一个数组 ByteBuffer s(实际上代表整数)。我想在数组中分离唯一和非唯一的 ByteBuffer(即整数)。因此我使用这种类型的 HashSet: HashSet columnsSet
我的代码中有一个 java.nio.ByteBuffer: ByteBuffer bb = ByteBuffer.allocateDirect(1024); ... 我希望能够将其替换为 ByteBu
我需要将代码从 Java 移植到 C#。在Java代码中,使用了方法“ByteBuffer.flip()”和“ByteBuffer.slice”,不知道怎么翻译。 我读过这个问题 ( An equiv
以下 Java 代码可以编译,但在运行时会出错: # javac ByteBufTest.java # java ByteBufTest Exception in thread "main" java
我有一个方法如下,它已经正常运行了很长时间: private String loadFromFile(){ RandomAccessFile inFile = null; FileCh
用户在测试中遇到了此崩溃。我的第一个猜测是这与内存有关,但除此之外我没有什么可做的。更深入地研究代码,我认为这可能是主线程问题,但看起来监听器在后台线程上被删除,所以我怀疑这就是原因。 我认为在应用程
以下代码(此为简化版)以前在jdk1.6中运行良好,现在在jdk 1.7下断言失败。 ByteBuffer buffer = ...; buffer.mark(); char c = (char) b
我有一个 10MB 的二进制文件。我需要以不同大小的 block (例如 300、273 字节)读取它。为了阅读,我使用 FileChannel 和 ByteBuffer。现在,对于每次读取迭代,我都
我想要做的是获取一个十进制整数,将其转换为十六进制,然后分隔字节。 据我了解,ByteBuffer 是实现此目的的最佳方法。整数不会超过65535,因此十六进制数保证为2个字节。例如,我有一个整数 4
我正在写一些使用 ByteBuffer 的东西s。在 docs of the API它说 There is no way to free a buffer explicitly (without JV
我正在尝试读取一个短数组,转换其元素并将它们放入字节数组中。我使用 ByteBuffer 并收回此异常: 线程“main”中的异常 java.lang.IndexOutOfBoundsExceptio
我是一名优秀的程序员,十分优秀!