前言

  接着上篇博客,我们来谈谈java操作cassandra; 上篇博客的环境:jdk1.7 + python2.7.10 + cassandra2.2.8; 由于2.2.8没有对应的驱动文档,那么我们就用3.0的驱动文档,而驱动则用2.1.10.3版本;

  驱动文档:http://docs.datastax.com/en/developer/java-driver/3.0/manual/

  路漫漫其修远兮,吾将上下而求索!

  github:https://github.com/youzhibing

  码云(gitee):https://gitee.com/youzhibing

驱动下载

  老规矩,创建maven工程,让maven来维护我们的jar,maven最重要的pom文件内容如下:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion> <groupId>com.huawei</groupId>
<artifactId>cassandra</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>jar</packaging> <name>cassandra</name>
<url>http://maven.apache.org</url> <properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties> <dependencies>
<dependency>
<groupId>com.datastax.cassandra</groupId>
<artifactId>cassandra-driver-core</artifactId>
<version>2.1.10.3</version>
</dependency>
</dependencies>
</project>

Session获取

  我们先来看看官方给的Quick start案例

Cluster cluster = null;
try {
cluster = Cluster.builder() // (1)
.addContactPoint("127.0.0.1")               // cassandra服务器ip
.withCredentials("admin", "admin") // 若没有启用账号认证,此处可以去掉
.build();
Session session = cluster.connect(); // (2) ResultSet rs = session.execute("select release_version from system.local"); // (3)
Row row = rs.one();
System.out.println(row.getString("release_version")); // (4)
} finally {
if (cluster != null) cluster.close(); // (5)
}

  我们来看看代码中的(1) ~ (5)分别表示或者代表什么

    (1):Cluster对象是驱动程序的主入口点,它保存着真实Cassandra集群的状态(尤其是元数据);Cluster是线程安全的,一个Cassandra集群创建一个Cluster的单例,整个应用用这一个单例即可

    (2):Session用来执行查询的,而且它也是线程安全的,同样也应该重复利用

    (3):利用execute来发送一个查询到Cassandra,execute返回一个Resultset(结果集),这个结果集就是必要的列的行集合(二维表,行是满足条件的记录,列是我们关注的某些字段)

    (4):从row中提取数据

    (5):当任务完成后,关闭cluster,关闭cluster的同时将会关闭它创建的全部session;这一步很重要,它会释放潜在的资源(TCP连接、线程池等),在真实的应用中,我们应该在应用关闭(或应用卸载)的时候关闭cluster

    如若大家有jdbc开发的经验,就会发现,上述代码似曾相识,上述代码中的session就相当于jdbc中的connection,是整个数据库操作的基础,那么我们将session的获取单独抽出来

package com.huawei.cassandra.factory;

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock; import com.datastax.driver.core.Cluster;
import com.datastax.driver.core.Session; public class SessionRepository
{
private static Session instance = null;
private static Cluster cluster = null;
private static Lock lock = new ReentrantLock(); private SessionRepository(){} public static Session getSession()
{
if (null == instance)
{
try
{
lock.lock(); if (null == instance)
{
cluster = Cluster.builder()
.addContactPoint("127.0.0.1")
.withCredentials("admin", "admin")
.build();
instance = cluster.connect();
// 也可以针对一个特定的keyspace获取一个session
// instance = cluster.connect("mycas");
}
}
finally
{
lock.unlock();
}
}
return instance;
} public static void close()
{
if (null == cluster)
{
try
{
lock.lock(); if (null == cluster)
{
cluster.close();
}
}
finally
{
lock.unlock();
}
}
}
}

    拿到session了,那么请随意操作Cassandra吧!

cassandra基本操作

  创建表

    在mycas下创建表student

use mycas;
create table student(
id int,
address text,
name text,
age int,
height int,
primary key(id,address,name)
); insert into student(id,address,name,age,height) values(1,'guangdong','lixiao',32,172);

  session直接执行cql

    和jdbc类似,关键是cql的拼接,下例是插入一条记录,删、改、查和这类似,不一一列举了

    // 字符串注意单引号'
String cql = "insert into mycas.student(id,address,name,age,height) values("
+ student.getId() + ",'" + student.getAddress() + "','" + student.getName()
+ "'," + student.getAge() + "," + student.getHeight() + ");";
System.out.println(cql);
session.execute(cql);

  Querybuilder

    利用Querybuilder可以减轻cql的拼接,sql语句的拼接由驱动完成

    查询一个student:

    @Override
public Student getStudentByKeys(int id, String address, String name)
{
Student student = null;
ResultSet rs = session.execute(
QueryBuilder.select("id", "address", "name", "age", "height")
.from("mycas", "student")
.where(QueryBuilder.eq("id", id))
.and(QueryBuilder.eq("address", address))
.and(QueryBuilder.eq("name", name)));
Iterator<Row> rsIterator = rs.iterator();
if (rsIterator.hasNext())
{
Row row = rsIterator.next();
student = new Student();
student.setAddress(row.getString("address"));
student.setAge(row.getInt("age"));
student.setHeight(row.getInt("height"));
student.setId(row.getInt("id"));
student.setName(row.getString("name"));
}
return student;
}

    保存一个student:

  @Override
public void saveStudent(Student student)
{
session.execute(
QueryBuilder.insertInto("mycas", "student")
.values(new String[]{"id", "address", "name", "age", "height"},
new Object[]{student.getId(), student.getAddress(),
student.getName(), student.getAge(), student.getHeight()}));
}

    修改一个student:

@Override
public void updateStudent(Student student)
{
session.execute(
QueryBuilder.update("mycas", "student")
.with(QueryBuilder.set("age", student.getAge()))
.and(QueryBuilder.set("height", student.getHeight()))
.where(QueryBuilder.eq("id", student.getId()))
.and(QueryBuilder.eq("address", student.getAddress()))
.and(QueryBuilder.eq("name", student.getName())));
}

    删除一个student:

@Override
public void removeStudent(int id, String address, String name)
{
session.execute(QueryBuilder.delete()
.from("mycas", "student")
.where(QueryBuilder.eq("id", id))
.and(QueryBuilder.eq("address", address))
.and(QueryBuilder.eq("name", name)));
}

  注意:驱动版本不同,Querybuilder的用法有些许不同,有些版本的某些方法变成非静态的了!

  占位符

    cassandra也有类似jdbc那样使用预编译占位符

    http://docs.datastax.com/en/developer/java-driver/3.0/manual/statements/prepared/

    预编译的原理是怎样的了,上面的链接是驱动官方的解释,我来谈谈我的理解

    当我们预编译statement的时候,Cassandra会解析query语句,缓存解析的结果并返回一个唯一的标志(PreparedStatement对象保持着这个标志的内部引用,就相当于通过标志可以获取到query语句预编译后的内容):

    当你绑定并且执行预编译statement的时候,驱动只会发送这个标志,那么Cassandra就会跳过解析query语句的过程:

    所以,我们应该保证query语句只应该被预编译一次,缓存PreparedStatement 到我们的应用中(PreparedStatement 是线程安全的);如果我们对同一个query语句预编译了多次,那么驱动会打印警告日志;如果一个query语句只执行一次,那么预编译不会提供性能上的提高,反而会降低性能,因为它是两个来回(结合上面两张图),那么此时可以考虑用 simple statement 来代替

    和jdbc的预编译非常类似,我们来看看实际代码

    静态cql

  private static final String GET_STUDENT = "select id,address,name,age,height from mycas.student where id=? and address=? and name=?;";
  private static final String SAVE_STUDENT = "insert into mycas.student(id,address,name,age,height) values(?,?,?,?,?);";
  private static final String UPDATE_STUDENT = "update mycas.student set age=?, height=? where id=? and address=? and name=?;";
  private static final String REMOVE_STUDENT = "delete from mycas.student where id=? and address=? and name=?;";

    查询一个student

     Student student = null;
PreparedStatement prepareStatement = session.prepare(GET_STUDENT);
BoundStatement bindStatement = new BoundStatement(prepareStatement).bind(id, address, name);
ResultSet rs = session.execute(bindStatement);
Iterator<Row> rsIterator = rs.iterator();
if (rsIterator.hasNext())
{
Row row = rsIterator.next();
student = new Student();
student.setAddress(row.getString("address"));
student.setAge(row.getInt("age"));
student.setHeight(row.getInt("height"));
student.setId(row.getInt("id"));
student.setName(row.getString("name"));
}
return student;

    保存一个student

        PreparedStatement prepareStatement = session.prepare(SAVE_STUDENT);
BoundStatement bindStatement = new BoundStatement(prepareStatement)
.bind(student.getId(), student.getAddress(), student.getName(), student.getAge(), student.getHeight());
session.execute(bindStatement);

    修改一个student

     PreparedStatement prepareStatement = session.prepare(UPDATE_STUDENT);
BoundStatement bindStatement = new BoundStatement(prepareStatement)
.bind(student.getAge(), student.getHeight(), student.getId(), student.getAddress(), student.getName());
session.execute(bindStatement);

    删除一个student

     PreparedStatement prepareStatement = session.prepare(REMOVE_STUDENT);
BoundStatement bindStatement = new BoundStatement(prepareStatement)
.bind(id, address, name);
session.execute(bindStatement);

  批量batch

public static void batch()
{
Session session = SessionRepository.getSession();
BoundStatement insertBind1 = new BoundStatement(
session.prepare("insert into mycas.student(id,address,name,age,height) values(?,?,?,?,?);"))
.bind(3, "guangxi", "huangfeihong", 67, 175); BoundStatement insertBind2 = new BoundStatement(
session.prepare("insert into mycas.student(id,address,name,age,height) values(?,?,?,?,?);"))
.bind(4, "hunan", "youzhibing", 26, 160); BoundStatement updateBind = new BoundStatement(
session.prepare("update mycas.student set age=?, height=? where id=? and address=? and name=?;"))
.bind(72, 173, 3, "guangxi", "huangfeihong"); BoundStatement deleteBind = new BoundStatement(
session.prepare("delete from mycas.student where id=? and address=? and name=?;"))
.bind(4, "hunan", "youzhibing"); BatchStatement batchStatement = new BatchStatement();
batchStatement.add(insertBind1);
batchStatement.add(insertBind2);
batchStatement.add(updateBind);
batchStatement.add(deleteBind);
session.execute(batchStatement);
}

参考与代码附件

  java实现cassandra的增删改查

  基本操作demo

Cassandra-java操作——基本操作的更多相关文章

  1. cassandra高级操作之分页的java实现(有项目具体需求)

    接着上篇博客,我们来谈谈java操作cassandra分页,需要注意的是这个分页与我们平时所做的页面分页是不同的,具体有啥不同,大家耐着性子往下看. 上篇博客讲到了cassandra的分页,相信大家会 ...

  2. Java链表基本操作和Java.util.ArrayList

    Java链表基本操作和Java.util.ArrayList 今天做了一道<剑指offer>上的一道编程题“从尾到头打印链表”,具体要求如下:输入一个链表,按链表值从尾到头的顺序返回一个A ...

  3. java操作elasticsearch实现基本的增删改查操作

    一.在进行java操作elasticsearch之前,请确认好集群的名称及对应的ES节点ip和端口 1.查看ES的集群名称 #进入elasticsearch.yml配置文件/opt/elasticse ...

  4. java操作redis学习(一):安装及连接

    文章参考自:http://www.cnblogs.com/edisonfeng/p/3571870.html,在此基础上进行了修改. 一.下载安装Redis redis官网显示暂时不支持Windows ...

  5. java操作Excel、PDF文件

    java操作Excel.PDF文件 分享者:Vashon 分享来源:CSDN博客 下面这些是在开发中用到的一些东西,有的代码贴的不是完整的,只是贴出了关于操作EXCEL的代码: jxl是一个*国人写的 ...

  6. Redis-基本概念、java操作redis、springboot整合redis,分布式缓存,分布式session管理等

    NoSQL的引言 Redis数据库相关指令 Redis持久化相关机制 SpringBoot操作Redis Redis分布式缓存实现 Resis中主从复制架构和哨兵机制 Redis集群搭建 Redis实 ...

  7. Elasticsearch之java的基本操作一

    摘要   接触ElasticSearch已经有一段了.在这期间,遇到很多问题,但在最后自己的不断探索下解决了这些问题.看到网上或多或少的都有一些介绍ElasticSearch相关知识的文档,但个人觉得 ...

  8. Java操作Sqlite数据库-jdbc连接

    Java操作Sqlite数据库步骤: 1. 导入Sqlite jdbc 本文使用sqlite-jdbc-3.7.2.jar,下载地址 http://pan.baidu.com/s/1kVHAGdD 2 ...

  9. 【MongoDB for Java】Java操作MongoDB

    上一篇文章: http://www.cnblogs.com/hoojo/archive/2011/06/01/2066426.html介绍到了在MongoDB的控制台完成MongoDB的数据操作,通过 ...

随机推荐

  1. shell 快速入门

    1: 脚本开始行 #!/bin/bash 这一行表明,不管用户选择的是那种交互式shell,该脚本需要使用bash shell来运行. 由于每种shell的语法大不相同,所以这句非常重要. 2:变量 ...

  2. W3Cschool学习笔记——HTML基础教程

    什么是 HTML? HTML 是用来描述网页的一种语言. HTML 指的是超文本标记语言 (Hyper Text Markup Language) HTML 不是一种编程语言,而是一种标记语言 (ma ...

  3. css3 过渡和2d变换——回顾

    1.transition 语法:transition: property duration timing-function delay; transition-property 设置过渡效果的css ...

  4. Quartz_理解1

    一.引言 quratz是目前最为成熟,使用最广泛的java任务调度框架,功能强大配置灵活.在企业应用中占重要地位.quratz在集群环境中的使用方式是每个企业级系统都要考虑的问题.早在2006年,在I ...

  5. Use Prerender to improve AngularJS SEO

    Use Prerender to improve AngularJS SEO Nuget Package of ASP.NET MVC HttpModule for prerender.io: Ins ...

  6. 最新升级的火狐38.0.6识别ajax调用返回的""空值可能有异常。

    自已在调用一段ajax开发中,返回的是空值 string result = string.Empty;return result; 但在页面进行$.ajax调用 时 输出alert(result);应 ...

  7. jvm垃圾收集小记

    垃圾收集是java与c/c++的最大不同.有了jvm的自动垃圾收集机制,就可以让程序员专注于程序逻辑开发, 而不是花费大量的时间是考虑一个变量应该在什么时候去释放. 下面我们就来简单说一下java的垃 ...

  8. java多线程四种实现模板

    假设一个项目拥有三块独立代码块,需要执行,什么时候用多线程? 这些代码块某些时候需要同时运行,彼此独立,那么需要用到多线程操作更快... 这里把模板放在这里,需要用的时候寻找合适的来选用. 总体分为两 ...

  9. OOP面向对象三大特点

    OOP面向对象三大特点 (一)封装:将现实中一个事物的属性和功能集中定义在一个对象中.(创建对象) 创建对象的3种方式: 1.直接量方式:(创建一个单独的对象) var obj={ 属性名:值,    ...

  10. intellij IDEA mybatis插件破解方法

    1>安装mybatis插件,找到mybatis_plus.jar包的位置,在C:\Users\LZHL\.IntelliJIdea2016.3\config\plugins\mybatis_pl ...