HBase的安裝、配置與實踐


 本教程運行環境是在Ubuntu-64位系統下,HBase版本為hbase-1.1.2,這是目前已經發行的已經編譯好的穩定的版本,帶有src的文件是未編譯的版本,這里我們只要下載bin版本hbase-1.1.2-bin.tar.gz就好了。

下載地址:http://archive.apache.org/dist/hbase/

本文參考原文地址:https://dblab.xmu.edu.cn/blog/install-hbase/

一、 HBase安裝及配置

   1. HBase安裝                                     

1.1 解壓安裝包hbase-1.1.2-bin.tar.gz至路徑 /usr/local,命令如下:

1.2 將解壓的文件名hbase-1.1.2改為hbase,以方便使用,命令如下:

1.3 配置環境變量
將hbase下的bin目錄添加到path中,這樣,啟動hbase就無需到/usr/local/hbase目錄下,大大的方便了hbase的使用。教程下面的部分還是切換到了/usr/local/hbase目錄操作,有助於初學者理解運行過程,熟練之后可以不必切換。
編輯~/.bashrc文件

 如果沒有引入過PATH請在~/.bashrc文件尾行添加如下內容:

如果已經引入過PATH請在export PATH這行追加/usr/local/hbase/bin,這里的“:”是分隔符。如下圖:

 編輯完成后,再執行source命令使上述配置在當前終端立即生效,命令如下:

 

1.4 添加HBase權限

 

本句意思是將hbase下的所有文件的所有者改為hadoop,hadoop是當前用戶的用戶名。

 

 1.5 查看HBase版本,確定hbase安裝成功,命令如下:

 

命令執行后,輸出信息截圖如下:

看到以上輸出消息表示HBase已經安裝成功,接下來將分別進行HBase單機模式和偽分布式模式的配置。

 

   2. HBase配置                      

 HBase有三種運行模式,單機模式、偽分布式模式、分布式模式。作為學習,我們重點討論單機模式和偽分布式模式。
以下先決條件很重要,比如沒有配置JAVA_HOME環境變量,就會報錯。
– jdk
– Hadoop( 單機模式不需要,偽分布式模式和分布式模式需要)
– SSH

必須保證這三項均已安裝成功,再繼續下面的操作。

    2.1單機模式配置                    

1. 配置/usr/local/hbase/conf/hbase-env.sh 。配置JAVA環境變量,並添加配置HBASE_MANAGES_ZK為true,用vi命令打開並編輯hbase-env.sh,命令如下:

配置JAVA環境變量,jdk的安裝目錄默認是 /usr/lib/jvm/java-1.7.0-openjdk, 則JAVA _HOME =/usr/lib/jvm/java-7-openjdk-amd64,其中java-1.7.0-openjdk是你的jdk版本;配置HBASE_MANAGES_ZK為true,表示由hbase自己管理zookeeper,不需要單獨的zookeeper。hbase-env.sh中本來就存在這些變量的配置,大家只需要刪除前面的#並修改配置內容即可(#代表注釋):

 

添加完成后保存退出即可。

2. 配置/usr/local/hbase/conf/hbase-site.xml
打開並編輯hbase-site.xml,命令如下:

在啟動HBase前需要設置屬性hbase.rootdir,用於指定HBase數據的存儲位置,

因為如果不設置的話,hbase.rootdir默認為/tmp/hbase-${user.name},這意味着每次重啟系統都會丟失數據。

此處設置為HBase安裝目錄下的hbase-tmp文件夾即(/usr/local/hbase/hbase-tmp),添加配置如下:

 

3. 接下來測試運行。首先切換目錄至HBase安裝目錄/usr/local/hbase;再啟動HBase。命令如下:

 注意先啟動hadoop   , start-dfs.sh

 

上述三條命令中,sudo bin/start-hbase.sh用於啟動HBase,bin/hbase shell用於打開shell命令行模式,用戶可以通過輸入shell命令操作HBase數據庫。

成功啟動HBase,截圖如下:

 

停止HBase運行,命令如下:

注意:如果在操作HBase的過程中發生錯誤,可以通過{HBASE_HOME}目錄(/usr/local/hbase)下的logs子目錄中的日志文件查看錯誤原因。

 

    2.2 偽分布式模式配置                      

1.配置/usr/local/hbase/conf/hbase-env.sh。命令如下:

配置JAVA_HOME,HBASE_CLASSPATH,HBASE_MANAGES_ZK.
HBASE_CLASSPATH設置為本機Hadoop安裝目錄下的conf目錄(即/usr/local/hadoop/conf)

export JAVA_HOME=/usr/lib/jvm/java-7-openjdk-amd64

export HBASE_CLASSPATH=/usr/local/hadoop/conf

export HBASE_MANAGES_ZK=true

 

 2.配置/usr/local/hbase/conf/hbase-site.xml
用命令vi打開並編輯hbase-site.xml,命令如下:

修改hbase.rootdir,指定HBase數據在HDFS上的存儲路徑;將屬性hbase.cluter.distributed設置為true。假設當前Hadoop集群運行在偽分布式模式下,在本機上運行,且NameNode運行在9000端口。

<configuration>
  <property>
    <name>hbase.rootdir</name>
    <value>hdfs://localhost:9000/hbase</value>
  </property>
  <property>
    <name>hbase.cluster.distributed</name>
    <value>true</value>
  </property>
</configuration>

 hbase.rootdir指定HBase的存儲目錄;hbase.cluster.distributed設置集群處於分布式模式.
截圖如下:

3. 接下來測試運行HBase。
第一步:首先登陸ssh,之前設置了無密碼登陸,因此這里不需要密碼;再切換目錄至/usr/local/hadoop ;再啟動hadoop,如果已經啟動hadoop請跳過此步驟。命令如下:

 

輸入命令jps,能看到NameNode,DataNode和SecondaryNameNode都已經成功啟動,表示hadoop啟動成功,截圖如下:

第二步:切換目錄至/usr/local/hbase;再啟動HBase.命令如下:

 

啟動成功,輸入命令jps,看到以下界面說明hbase啟動成功

 

進入shell界面:

4.停止HBase運行,命令如下:

  bin/stop-hbase.sh

 

   二、 編程實踐                                              

   1. 利用Shell命令                                     

   1.1 HBase中創建表                

HBase中用create命令創建表,具體如下:

  create 'student','Sname','Ssex','Sage','Sdept','course'

命令執行截圖如下:

此時,即創建了一個“student”表,屬性有:Sname,Ssex,Sage,Sdept,course。因為HBase的表中會有一個系統默認的屬性作為行鍵,無需自行創建,默認為put命令操作中表名后第一個數據。創建完“student”表后,可通過describe命令查看“student”表的基本信息。命令執行截圖如下:

 

   1.2 HBase數據庫基本操作                              

本小節主要介紹HBase的增、刪、改、查操作。在添加數據時,HBase會自動為添加的數據添加一個時間戳,故在需要修改數據時,只需直接添加數據,HBase即會生成一個新的版本,從而完成“改”操作,舊的版本依舊保留,系統會定時回收垃圾數據,只留下最新的幾個版本,保存的版本數可以在創建表的時候指定。 

  • 添加數據
    HBase中用put命令添加數據,注意:一次只能為一個表的一行數據的一個列,也就是一個單元格添加一個數據,所以直接用shell命令插入數據效率很低,在實際應用中,一般都是利用編程操作數據。
    當運行命令:put 'student','95001','Sname','LiYing'時,即為student表添加了學號為95001,名字為LiYing的一行數據,其行鍵為95001。

運行put 'student','95001','course:math','80'

即為95001行下的course列族的math列添加了一個數據。

 

 

  • 查看數據
    HBase中有兩個用於查看數據的命令:1. get命令,用於查看表的某一行數據;2. scan命令用於查看某個表的全部數據

1. get命令

  get 'student','95001'

 命令執行截圖如下, 返回的是‘student’表‘95001’行的數據。

 

2. scan命令

  scan 'student'

命令執行截圖如下, 返回的是‘student’表的全部數據。

 

  • 刪除數據

在HBase中用delete以及deleteall命令進行刪除數據操作,它們的區別是:1. delete用於刪除一個數據,是put的反向操作;2. deleteall操作用於刪除一行數據。

1. delete命令

  delete 'student','95001','Sname'

 命令執行截圖如下, 即刪除了student表中95001行下的Sname列的所有數據。

 

2. deleteall命令

   deleteall 'student','95001'

 命令執行截圖如下,即刪除了student表中的95001行的全部數據。

 

 

  • 刪除表
    刪除表有兩步,第一步先讓該表不可用,第二步刪除表。

   disable 'student'

  drop 'student'

 命令執行截圖如下:

 

   1.3 查詢表歷史數據                            

 查詢表的歷史版本,需要兩步。
1、在創建表的時候,指定保存的版本數(假設指定為5)

   create 'teacher',{NAME=>'username',VERSIONS=>5}

2、插入數據然后更新數據,使其產生歷史版本數據,注意:這里插入數據和更新數據都是用put命令

put 'teacher','91001','username','Mary'
put 'teacher','91001','username','Mary1'
put 'teacher','91001','username','Mary2'
put 'teacher','91001','username','Mary3'
put 'teacher','91001','username','Mary4'
put 'teacher','91001','username','Mary5'

 

 

3、查詢時,指定查詢的歷史版本數。默認會查詢出最新的數據。(有效取值為1到5)

   get 'teacher','91001',{COLUMN=>'username',VERSIONS=>5}

 查詢結果截圖如下:

 

1.4 退出HBase數據庫表操作

 最后退出數據庫操作,輸入exit命令即可退出,注意:這里退出HBase數據庫是退出對數據庫表的操作,而不是停止啟動HBase數據庫后台運行。

 

 

   2. Java API編程實例                   

本實例使用Eclipse編寫java程序,來對HBase數據庫進行增刪改查等操作,Eclipse可以在Ubuntu軟件中心搜索下載並安裝。
第一步:啟動hadoop,啟動hbase

cd /usr/local/hadoop

./sbin/start-dfs.sh

cd /usr/local/hbase

./bin/start-hbase.sh

第二步,新建Java Project——>新建Class

第三步:在工程中導入外部jar包:
這里只需要導入hbase安裝目錄中的lib文件中的所有jar包。
新版的Hbase 1.1.2的java api已經發生變化,舊版的部分api已經停止使用

 

這里給出一個編程實例,,以下是源代碼:

 

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.hbase.*;
import org.apache.hadoop.hbase.client.*;
import java.io.IOException;

public class ExampleForHbase {
    public static Configuration configuration;
    public static Connection connection;
    public static Admin admin;
    
    //主函數中的語句請逐句執行,只需刪除其前的//即可,如:執行insertRow時請將其他語句注釋
    public static void main(String[] args)throws IOException{
        //創建一個表,表名為Score,列族為sname,course
        //createTable("Score",new String[]{"sname","course"});
 
        //在Score表中插入一條數據,其行鍵為95001,sname為Mary(因為sname列族下沒有子列所以第四個參數為空)
        //等價命令:put 'Score','95001','sname','Mary'
        insertRow("Score", "95001", "sname", "", "Mary");
        //在Score表中插入一條數據,其行鍵為95001,course:Math為88(course為列族,Math為course下的子列)
        //等價命令:put 'Score','95001','score:Math','88'
        //insertRow("Score", "95001", "course", "Math", "88");
        //在Score表中插入一條數據,其行鍵為95001,course:English為85(course為列族,English為course下的子列)
        //等價命令:put 'Score','95001','score:English','85'
        //insertRow("Score", "95001", "course", "English", "85");
 
        //1、刪除Score表中指定列數據,其行鍵為95001,列族為course,列為Math
        //執行這句代碼前請deleteRow方法的定義中,將刪除指定列數據的代碼取消注釋注釋,將刪除制定列族的代碼注釋
        //等價命令:delete 'Score','95001','score:Math'
        //deleteRow("Score", "95001", "course", "Math");
 
        //2、刪除Score表中指定列族數據,其行鍵為95001,列族為course(95001的Math和English的值都會被刪除)
        //執行這句代碼前請deleteRow方法的定義中,將刪除指定列數據的代碼注釋,將刪除制定列族的代碼取消注釋
        //等價命令:delete 'Score','95001','score'
        //deleteRow("Score", "95001", "course", "");
 
        //3、刪除Score表中指定行數據,其行鍵為95001
        //執行這句代碼前請deleteRow方法的定義中,將刪除指定列數據的代碼注釋,以及將刪除制定列族的代碼注釋
        //等價命令:deleteall 'Score','95001'
        //deleteRow("Score", "95001", "", "");
 
        //查詢Score表中,行鍵為95001,列族為course,列為Math的值
        //getData("Score", "95001", "course", "Math");
        //查詢Score表中,行鍵為95001,列族為sname的值(因為sname列族下沒有子列所以第四個參數為空)
        //getData("Score", "95001", "sname", "");
 
        //刪除Score表
        //deleteTable("Score");
    }
 
    //建立連接
    public static void init(){
        configuration  = HBaseConfiguration.create();
        configuration.set("hbase.rootdir","hdfs://localhost:9000/hbase");
        try{
            connection = ConnectionFactory.createConnection(configuration);
            admin = connection.getAdmin();
        }catch (IOException e){
            e.printStackTrace();
        }
    }
    //關閉連接
    public static void close(){
        try{
            if(admin != null){
                admin.close();
            }
            if(null != connection){
                connection.close();
            }
        }catch (IOException e){
            e.printStackTrace();
        }
    }
 
    /**
     * 建表。HBase的表中會有一個系統默認的屬性作為主鍵,主鍵無需自行創建,默認為put命令操作中表名后第一個數據,因此此處無需創建id列
     * @param myTableName 表名
     * @param colFamily 列族名
     * @throws IOException
     */
    public static void createTable(String myTableName,String[] colFamily) throws IOException {
 
        init();
        TableName tableName = TableName.valueOf(myTableName);
 
        if(admin.tableExists(tableName)){
            System.out.println("talbe is exists!");
        }else {
            HTableDescriptor hTableDescriptor = new HTableDescriptor(tableName);
            for(String str:colFamily){
                HColumnDescriptor hColumnDescriptor = new HColumnDescriptor(str);
                hTableDescriptor.addFamily(hColumnDescriptor);
            }
            admin.createTable(hTableDescriptor);
            System.out.println("create table success");
        }
        close();
    }
    /**
     * 刪除指定表
     * @param tableName 表名
     * @throws IOException
     */
    public static void deleteTable(String tableName) throws IOException {
        init();
        TableName tn = TableName.valueOf(tableName);
        if (admin.tableExists(tn)) {
            admin.disableTable(tn);
            admin.deleteTable(tn);
        }
        close();
    }
 
    /**
     * 查看已有表
     * @throws IOException
     */
    public static void listTables() throws IOException {
        init();
        HTableDescriptor hTableDescriptors[] = admin.listTables();
        for(HTableDescriptor hTableDescriptor :hTableDescriptors){
            System.out.println(hTableDescriptor.getNameAsString());
        }
        close();
    }
    /**
     * 向某一行的某一列插入數據
     * @param tableName 表名
     * @param rowKey 行鍵
     * @param colFamily 列族名
     * @param col 列名(如果其列族下沒有子列,此參數可為空)
     * @param val 值
     * @throws IOException
     */
    public static void insertRow(String tableName,String rowKey,String colFamily,String col,String val) throws IOException {
        init();
        Table table = connection.getTable(TableName.valueOf(tableName));
        Put put = new Put(rowKey.getBytes());
        put.addColumn(colFamily.getBytes(), col.getBytes(), val.getBytes());
        table.put(put);
        table.close();
        close();
    }
 
    /**
     * 刪除數據
     * @param tableName 表名
     * @param rowKey 行鍵
     * @param colFamily 列族名
     * @param col 列名
     * @throws IOException
     */
    public static void deleteRow(String tableName,String rowKey,String colFamily,String col) throws IOException {
        init();
        Table table = connection.getTable(TableName.valueOf(tableName));
        Delete delete = new Delete(rowKey.getBytes());
        //刪除指定列族的所有數據
        //delete.addFamily(colFamily.getBytes());
        //刪除指定列的數據
        //delete.addColumn(colFamily.getBytes(), col.getBytes());
 
        table.delete(delete);
        table.close();
        close();
    }
    /**
     * 根據行鍵rowkey查找數據
     * @param tableName 表名
     * @param rowKey 行鍵
     * @param colFamily 列族名
     * @param col 列名
     * @throws IOException
     */
    public static void getData(String tableName,String rowKey,String colFamily,String col)throws  IOException{
        init();
        Table table = connection.getTable(TableName.valueOf(tableName));
        Get get = new Get(rowKey.getBytes());
        get.addColumn(colFamily.getBytes(),col.getBytes());
        Result result = table.get(get);
        showCell(result);
        table.close();
        close();
    }
    /**
     * 格式化輸出
     * @param result
     */
    public static void showCell(Result result){
        Cell[] cells = result.rawCells();
        for(Cell cell:cells){
            System.out.println("RowName:"+new String(CellUtil.cloneRow(cell))+" ");
            System.out.println("Timetamp:"+cell.getTimestamp()+" ");
            System.out.println("column Family:"+new String(CellUtil.cloneFamily(cell))+" ");
            System.out.println("row Name:"+new String(CellUtil.cloneQualifier(cell))+" ");
            System.out.println("value:"+new String(CellUtil.cloneValue(cell))+" ");
        }
    }
}

 

 

 

每次執行完,都可以回到shell界面查看是否執行成功,如:執行完插入數據后,在shell界面中執行scan 'Score'。截圖如下:

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM