
我最近在 openframework 做作品的时候遇到了一个困难,在实例化一个类的时候我发现如果类本身定义了构造器,那我在实例化的时候就必须带构造器。有没有办法我先声明,然后我再在别的函数里面构造呢?
比如说有这么一个类是这么写的:
class dot { int speed; dot(int i ) } dot::dot(int i ){ sped = i; } 然后比如我要实例化它的时候我发现是需要这么写:
dot dottest(1); 否则好像他是没有办法被初始化的? 但是实际问题是,我在 OF 中它是带 setup 和 draw 函数的。 通常情况下如果我想在 setup 中初始化这个类,然后在 draw 函数中循环运行它的话,我必须要把这个类声明在函数外,但我只想在 setup 中构造它,但目前好像我测试了好几次都不行。 有没有办法先声明然后在定义呢? 我查到了关于<extern>的用法,但是它似乎并不是我想要的。因为我本质上还是在一个 cpp 文件里面去声明它。 还有一种情况是如果在 h 文件定义,在 cpp 文件中也可以调用。但只要是构造器带了参数的,在 h 文件中好像就没有办法很好的定义这个对象。 我之所以有这个疑问是因为,在 processing 中,实例化可以这么写:
FlowField flowfield; void setup() { flowfield = new FlowField(10); } void draw(){ } 我查了一下 C++和 java 的 new 的区别,还有声明和定义的区别,本来用的好好的现在越看越懵。所以如果我想模仿上面 processing 这么的写法,我在 C++中应该如何实现呢? 我知道 C++是有默认构造器的,我以前的写法是自己额外写一个函数去初始化类里面的变量。但是前段时间突发奇想想用一下构造器,结果就傻眼了。关于 C++声明和定义的一些相关的介绍我也看了,但还是处于似懂非懂的状态。求大家指条明路。。
1 InkStone 2020-04-17 18:32:25 +08:00 如果是你自己写的类,你可以写一个轻量级的构造器,加上一个包含了真正初始化逻辑的 init 函数。 如果是别人写的类,考虑下用一个单例类来包装一下这个对象,来实现你想要的延迟加载。 |
2 Waihinchan OP @InkStone 嗯嗯 感谢 其实总体我也明白有很多方法可以规避这个问题。其实是我有点钻牛角尖了,因为我本来不是很系统地学习过 C++,感觉学到后面深入的有一些概念就开始模糊和混淆了。 另外想问一下:您回复的关于这个轻量级的构造器,我是否可以理解为真正发挥作用的还是自己写的 init 函数,而他构造器本身可以直接缺省? |
3 stackexplode 2020-04-17 18:40:11 +08:00 class A { public: A() = default; Initialize(int speed) {speed_ = speed;} private: int speed_; } A a; a.Initialize(1); ======================= class B { public: B(int speed) : speed_(speed) {} private: int speed_; } std::unique_ptr<B> b; b = std::make_unique<B>(1); |
4 Waihinchan OP @stackexplode 感谢回复! classA 的我看懂了 classB 的我再消化消化 非常感谢。 |
5 stackexplode 2020-04-17 18:42:56 +08:00 @Waihinchan B* b; b = new B(1); 也可以,一样 |
6 ysc3839 2020-04-18 05:36:00 +08:00 via Android 用指针吧,setup 里面 new 然后返回到外面,再传递进 draw 。 或者用 std::optional 也行,相比之下更安全 (不会有忘记 delete 的问题),不需要动态分配内存。 https://zh.cppreference.com/w/cpp/utility/optional |
7 Wirbelwind 2020-04-18 06:45:56 +08:00 有用户定义的构造函数时,编译器不会生成其他构造函数。 没有用户定义的构造函数时,编译器看 class 里面成员变量类型来判断是否需要生成构造函数,因为普通内置类型如 char,int,long long 可以通过直接拷贝值来完成。如果有定义了构造函数的 class,比如说 std::string,会生成一个构造函数,这个构造函数里面会递归调用 std::string 的构造函数。 差不多是这样。 dot()=default; dot(int i )前面加一句这个就可以了。 |
8 Wirbelwind 2020-04-18 06:51:07 +08:00 还有一种写法是使用 placement new 来申请一定的内存,但是不进行构造。 在需要这块内存的时候调用相关构造函数,不需要的时候调用析构函数。 不过,不需要这块内存(不再使用 new 的对象)都是需要 delete 调用的 |
9 Waihinchan OP @ysc3839 感谢回复~现在我基本上掌握了指针的用法 其实关于 delete 的问题我也查阅了一下 应该是只要进程结束了内存还是会释放掉的吧?我主要做的是一些展示性的程序 感觉手动释放对于我这个问题并不会是一个很大的阻碍 XD |
10 codyfeng 2020-04-18 21:53:46 +08:00 std::unique_ptr 不需要手动释放。其 dtor 会自动 delete 。 |
11 banxi1988 2020-04-26 10:22:24 +08:00 推荐一下我的总结: 重学 C++ (1) 对象创建的这件小事 https://mp.weixin.qq.com/s/b0PeGYpb-iKfbI_N7QnQqg 重学 C++ (2) 从内存分配角度看对象的创建 https://mp.weixin.qq.com/s/IjMGi2cpQzucYdOufpVpBg 重学 C++ (04) 对 new/delete 说不! https://mp.weixin.qq.com/s/LrYagIpuvJyDqUQRq1xzNg 重学 C++(05) 手写 unique_ptr https://mp.weixin.qq.com/s/Ni8YUlyIyhHzoqvX2hvruw |