查看: 108|回复: 1

[教程] Rust声明宏在不同K线bar类型中的应用小结

[复制链接]

5

主题

0

回帖

0

积分

热心网友

金币
0
阅读权限
220
精华
0
威望
0
贡献
0
在线时间
0 小时
注册时间
2008-3-1
发表于 2024-5-11 11:01:53 | 显示全部楼层 |阅读模式

Rust的宏功能博大精深。在K线bar中,往往有很多不同分时k线图,比如1,2,3,5,。。。。60,120,250,300…。。不同分钟类型。
如果不用宏,那么手写会比较麻烦。下面就试用一下宏来实现不同类型的bar.

一、数据和功能

bar的结构

为了简单起见,我们把不同分时的Bar抽象成下面的结构。

 struct Bar{
     open :f64,
     close:f64,
     high:f64,
     low:f64,
 }

同时这个结构要实现一个trait

trait BarPrint{
    fn print_self(&self);
}

二、单一bar的实现

我们先考虑,impl_single_bar接受单一的类型参数,比如Bar1,Bar3,Bar5,…

trait BarPrint{
    fn print_self(&self);
}
// Bar1,Bar2,Bar3,Bar5,Bar10,Bar15,Bar30,Bar45,Bar60,.....
macro_rules!  impl_single_bar {
    ($bar:ident) => (
        #[derive(Debug)]
        struct $bar{
            open:f64,
            close:f64,
            high:f64,
            low:f64,
        }
        impl $bar{
            fn new() -> Self{
                $bar{
                    open:0.0,
                    close:0.0,
                    high:0.0,
                    low:0.0,
               }
            }
        }
        impl BarPrint for $bar {
            fn print_self(&self){
                println!("impl_single_bar =>close:{:?} open: {:?}, high:{:?}, low:{:?}",&self.close,&self.open,&self.high,&self.low);
            }
        }
    );
}
fn main(){
    impl_single_bar!(Bar1); //这个可以放在main()函数外,不影响
    let  bar = Bar1::new();
    println!("bar:{:?}",bar);
    bar.print_self(); 
    impl_single_bar!(Bar2);
    let  bar2 = Bar2::new();
    println!("bar:{:?}",bar2);
    bar2.print_self(); 
}

输出:

bar:Bar1 { open: 0.0, close: 0.0, high: 0.0, low: 0.0 }
impl_single_bar =>close:0.0 open: 0.0, high:0.0, low:0.0
bar:Bar2 { open: 0.0, close: 0.0, high: 0.0, low: 0.0 }
impl_single_bar =>close:0.0 open: 0.0, high:0.0, low:0.0

这样的确方便了一些,但是因为参数是一个个输入,需要

impl_single_bar!(Bar1);
impl_single_bar!(Bar2);

每一个类型,写一行函数,还是不太方便。
注意:
1、impl_single_bar!(Bar1),可以放在main()函数外,不受影响;
2、$bar:ident,也可以是 $bar:tt。tt是分语树,比ident概念要大。

三、实现多类型参数输入

这里就需要用到rust宏的重复的写法。这里不特别展开,相关的资料很多。

1、试写一下生成多个类型的宏

macro_rules! create_bars{
    ($($bar:ident),*) => {
        $(
            #[derive(Debug)]
            struct $bar{
                open:f64,
                close:f64,
                high:f64,
                low:f64,
            }
        )*
    }
}

2、上面也可以跳过,直接

trait BarPrint{
    fn print_self(&self);
}
macro_rules! impl_multi_bars{
    ($($bar:ident),*) => {
        $(
            #[derive(Debug)]
            struct $bar{
                open:f64,
                close:f64,
                high:f64,
                low:f64,
            }
            impl $bar{
                fn new() -> Self{
                    $bar{
                        open:0.0,
                        close:0.0,
                        high:0.0,
                        low:0.0,
                    }
                }
            }
            impl BarPrint for $bar {
                fn print_self(&self){
                    println!("impl_multi_bars => close:{:?} open: {:?}, high:{:?}, low:{:?}",&self.close,&self.open,&self.high,&self.low);
                }
            }
        )*
    }      
}
fn main(){
    create_bars!(Bar3,Bar4);
    let bar3 =Bar3{open:0.0,close:0.0,high:0.0,low:0.0};
    println!("bar3:{:?}",bar3);
    let bar4 =Bar4{open:0.0,close:0.0,high:0.0,low:0.0};
    println!("bar4:{:?}",bar4);
    // 测试生成多个struct Bar5,Bar6,Bar7,同时测试其实现的方法
    impl_multi_bars!(Bar5,Bar6,Bar7);//可以放在main()函数外,在main()函数内,直接调用即可。
    let  bar5 = Bar5::new();
    println!("bar5:{:?}",bar5);
    bar5.print_self();
}

输出:

bar3:Bar3 { open: 0.0, close: 0.0, high: 0.0, low: 0.0 }
bar4:Bar4 { open: 0.0, close: 0.0, high: 0.0, low: 0.0 }
bar5:Bar5 { open: 0.0, close: 0.0, high: 0.0, low: 0.0 }
impl_multi_bars => close:0.0 open: 0.0, high:0.0, low:0.0

和2相比,你可需要把多个类型写到一行中就行了,即:

   impl_multi_bars!(Bar5,Bar6,Bar7);

到此这篇关于Rust声明宏在不同K线bar类型中的应用小结的文章就介绍到这了,更多相关Rust声明宏内容请搜索琼殿技术社区以前的文章或继续浏览下面的相关文章希望大家以后多多支持琼殿技术社区!

您可能感兴趣的文章:
  • 一文弄懂rust声明宏
回复

使用道具 举报

0

主题

338

回帖

2140

积分

琼殿精英

金币
1802
阅读权限
220
精华
0
威望
0
贡献
0
在线时间
0 小时
注册时间
2011-10-11
发表于 3 小时前 | 显示全部楼层
好帖!学习了!

看完这篇关于Rust声明宏在K线bar中应用的文章,感觉收获很大。之前就听说Rust的宏很强大,但一直没用明白,看完你的例子终于有点开窍了。
其实我之前也遇到过类似的问题
,项目中需要定义很多类似的结构体,手动写确实很麻烦。用了你的方法后,确实能省不少功夫。

有个小问题想请教一下:

1、这种宏的使用方式,在实际生产环境中有没有性能上的考虑?会不会增加编译时间或者二进制体积?

2、对于期货/股票交易系统来说,不同周期的K线数据量很大,你在实际项目中有没有遇到过内存占用的问题?有没有什么优化建议?

另外感觉可以补充的一点是,如果这些bar类型需要实现序列化(比如要存到数据库或网络传输),derive宏怎么配合使用。

继续学习中... 感谢楼主的分享,期待更多Rust相关的实战文章!
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

相关侵权、举报、投诉及建议等,请发 E-mail:qiongdian@foxmail.com

Powered by Discuz! X5.0 © 2001-2026 Discuz! Team.

在本版发帖返回顶部