.Net轻松处理亿级数据--ClickHouse数据操做

该篇内容由我的博客点击跳转同步更新!转载请注明出处!git

我不喜欢拿一堆数据的运行耗时来对比各个解决方案的性能等,有时候看一些测评长篇大论写耗时的一些对比,有时就差个 几百毫秒 我以为也没啥必要,关键是好用就行,一切从简,我写博客也喜欢一切从简。
.Net操做Clickhouse的库比较少,大多数都是基于ClickHouse.ADO的一个封装,下面也主要介绍一下ClickHouse.ADO的使用,以及本身封装的一个库的使用。github

前言

Clickhouse适用于大数据量分析,个人应用场景是每十秒从公交轨迹中取固定时间段数据分析一些状况,电脑配置就是普通的开发配置,整体数据轨迹量在3亿左右,处理的数据时间段在一天之内,取出的数据量在2.3万条左右。你们能够当个借鉴!
耗时服务器

具体操做

1、简单的查询和新增以及批量新增(Clickhouse不推荐数据的编辑和删除此处就再也不举例)微信

public class Demo
{
        private ClickHouseConnection GetConnection(string cstr= "Compress=True;CheckCompressedHash=False;Compressor=lz4;Host=ch-test.flippingbook.com;Port=9000;Database=default;User=andreya;Password=123")
        {
            var settings = new ClickHouseConnectionSettings(cstr);
            var cnn = new ClickHouseConnection(settings);
            cnn.Open();
            return cnn;
        }
        /*查询*/
        public void Select()
        {
            using (var cnn = GetConnection())
            {
                var reader = cnn.CreateCommand("SELECT * FROM test").ExecuteReader()
                ......省略
            }
        }
        /*增长*/
        public void Insert()
        {
            using (var cnn = GetConnection())
            {
                var cmd = cnn.CreateCommand("INSERT INTO test (date,x, arr)values ('2017-01-01',1,['a','b','c'])");
                cmd.ExecuteNonQuery();
            }
        }
        /*批量新增*/
        public void InsertBulk()
        {
            using (var cnn = GetConnection())
            {
                var cmd = cnn.CreateCommand("INSERT INTO test (date,x, values.name,values.value)values @bulk;");
                cmd.Parameters.Add(new ClickHouseParameter
                {
                    DbType = DbType.Object,
                    ParameterName = "bulk",
                    Value = new[]
                    {
                        new object[] {DateTime.Now, 1, new[] {"aaaa@bbb.com", "awdasdas"}, new[] {"dsdsds", "dsfdsds"}},
                        new object[] {DateTime.Now.AddHours(-1), 2, new string[0], new string[0]},
                    }
                });
                cmd.ExecuteNonQuery();
            }
        }
}

2、鉴于使用原始方法读取数据后转换的方式太麻烦,分页等也须要本身实现,因此写了一个帮助类,方便操做Clickhouse,点击跳转post

帮助类
使用方式也很简单,以下:性能

public HistoryModel GetHistories(string busid, string begindt, string enddt)
        {
            using (var helper = new ClickHouseHelper())
            {
                try
                {
                    HistoryModel historyModel = new HistoryModel();
                    historyModel.Histories = helper .ExecuteList<HistoriesModel>($"select mile,speed,lon,lat,direct,termtime from its.gps_MergeTree where termtime >='{begindt}' and termtime<='{enddt}' and busid={busid} order by termtime");
                    historyModel.Inouts = helper .ExecuteList<InoutModel>($"SELECT * FROM its.inout_t WHERE Adtime>='{begindt}' and Adtime<='{enddt}' and Busid={busid} order by Recvtime");
                    //clickhouse中取出来的时间默认会有时区的问题,这里须要手动转下本地的时区
                    historyModel.Histories.ForEach(u => u.termtime = DateTime.Parse(u.termtime).ToLocalTime().ToString("yyyy-MM-dd HH:mm:ss"));
                    historyModel.Inouts.ForEach(u => u.Recvtime = u.Recvtime.ToLocalTime());
                    return historyModel;
                }
                catch (Exception e)
                {
                    ckhelper.Dispose();
                    Console.WriteLine(e);
                    throw;
                }
            }
        }

3、一些小问题记录大数据

  1. 时区问题
    Clickhosue中取出来的时候会多8个小时,以前一度怀疑安装时服务器时区不对,但实际上都是正确的,只能手动将时间经过ToLocalTime转成本地时区
  2. 批量插数据
    批量插数据的时候若是传入一个List的话,对应的类须要增长GetEnumerator方法,就像这样
public class Demo
{
     public string obu { get; set; }
     public int busid { get; set; }
     public string buscode { get; set; }
     public IEnumerator GetEnumerator()
        {
            yield return obu;
            yield return busid;
            yield return buscode;
            .....
        }
}
  1. 类型统一问题
    具体参考个人这篇文章 点击跳转

微信关注我哦!(转载注明出处)关注我哦

相关文章
相关标签/搜索