Cpp 对象模型探索 / new 对象时加括号和不加括号时的差别

本文详细解析了C++中new操作符的两种使用方式,即CTest*pt1=newCTest();与CTest*pt2=newCTest;的区别。通过实例演示,当类CTest是空类或存在显式声明的默认构造函数时,两者结果相同;但当类未显式声明构造函数时,带括号的new会初始化成员变量,而不带括号的则不会。

一、结论

CTest *pt1 = new CTest();
CTest* pt2 = new CTest;

1、若类 CTest 是空类,则二者 new 的结果没有区别。

2、若类存在显示声明的缺省的构造函数,则二者 new 的结果没有区别。

3、若类没有显示声明的缺省的构造函数,即使编译器为类创建了缺省的构造函数,二者的结果依然不一样。带括号的结果是会将 pt1 的 count_ 初始化为0,而不带括号的结果是不初始化 count_ 。

二、验证

1、对于空类,其对象只占1B,两种申请方式没有区别。

2、对于类存在显示声明的缺省的构造函数,两种 new 的时候都会调用该构造函数,所以对于 new 的结果没有影响。

#include <iostream>

class CTest 
{
public:
	CTest()
	{
		count_ = 0;
	}
public:
	size_t count_;
};

int main()
{
	CTest *pt1 = new CTest();
	CTest* pt2 = new CTest;

	std::cout << "()::count_ = " << pt1->count_ << std::endl;
	std::cout << "count_ = " << pt2->count_ << std::endl;

	std::cin.get();
	return 0;
}

结果

()::count_ = 0
count_ = 0

 3、对于没有显式声明构造函数的类,即使编译器自动生成了一个构造函数,二者 new 的结果也是不同的。带括号的 new 会对类中成员变量进行初始化,而不带括号的 new 则不采取任何操作。

#include <iostream>

class CTest 
{
public:
	size_t count_;
public:
	virtual void func(){}
};

int main()
{
	CTest *pt1 = new CTest();
	CTest* pt2 = new CTest;

	std::cout << "()::count_ = " << pt1->count_ << std::endl;
	std::cout << "count_ = " << pt2->count_ << std::endl;

	std::cin.get();
	return 0;
}

结果

()::count_ = 0
count_ = 3452816845

 

(SAW:Game Over!)

在Qt中,`QTableWidget` 是一个基于项的表格控件,它提供了一个模型-视图架构的便捷封装。要在 `QTableWidget` 的左侧绘制括号线(bracket lines),可以通过自定义委托(`QStyledItemDelegate`)或直接在 `QTableWidget` 的视口(viewport)上进行绘制来实现。 ### 方法一:使用自定义委托绘制括号线 可以继承 `QStyledItemDelegate` 并重写 `paint` 方法,在单元格的左侧绘制括号线。以下是实现代码示例: ```cpp #include <QStyledItemDelegate> #include <QPainter> class BracketDelegate : public QStyledItemDelegate { public: using QStyledItemDelegate::QStyledItemDelegate; void paint(QPainter *painter, const QStyleOptionButton *option, const QModelIndex &index) const override { QStyledItemDelegate::paint(painter, option, index); // 在左侧绘制括号线 QRect rect = option->rect; int bracketLength = 10; int yCenter = rect.top() + rect.height() / 2; painter->save(); painter->setPen(Qt::black); painter->drawLine(rect.left(), yCenter - bracketLength, rect.left(), yCenter + bracketLength); painter->drawLine(rect.left(), yCenter - bracketLength, rect.left() + bracketLength, yCenter - bracketLength); painter->drawLine(rect.left(), yCenter + bracketLength, rect.left() + bracketLength, yCenter + bracketLength); painter->restore(); } }; ``` 然后,将此委托设置到 `QTableWidget` 的特定列或全部列: ```cpp QTableWidget *tableWidget = new QTableWidget(5, 3, this); tableWidget->setItemDelegate(new BracketDelegate(tableWidget)); ``` ### 方法二:通过事件过滤器在视口上绘制 另一种方式是通过在 `QTableWidget` 的视口上安装事件过滤器,并在 `paintEvent` 中绘制括号线。这种方式适用于更复杂的自定义绘制需求。 ```cpp #include <QTableWidget> #include <QPainter> #include <QEvent> class BracketTableWidget : public QTableWidget { public: BracketTableWidget(int rows, int cols, QWidget *parent = nullptr) : QTableWidget(rows, cols, parent) { viewport()->installEventFilter(this); } bool eventFilter(QObject *obj, QEvent *event) override { if (obj == viewport() && event->type() == QEvent::Paint) { QTableWidget::eventFilter(obj, event); // 先调用基类处理 QPainter painter(viewport()); painter.setPen(Qt::black); for (int row = 0; row < rowCount(); ++row) { QRect rect = visualRect(model()->index(row, 0)); int bracketLength = 10; int yCenter = rect.top() + rect.height() / 2; painter.drawLine(rect.left(), yCenter - bracketLength, rect.left(), yCenter + bracketLength); painter.drawLine(rect.left(), yCenter - bracketLength, rect.left() + bracketLength, yCenter - bracketLength); painter.drawLine(rect.left(), yCenter + bracketLength, rect.left() + bracketLength, yCenter + bracketLength); } return true; } return QTableWidget::eventFilter(obj, event); } }; ``` ### 注意事项 - **性能考虑**:如果表格数据量较大,应优化绘制逻辑,避免不必要的重复绘制。 - **样式一致性**:绘制的括号线应与应用程序整体风格一致,必要可使用 `QStyleOption` 获取当前样式信息进行绘制。 - **扩展性**:可以通过引入配置项,允许用户自定义括号线的颜色、长度等属性。 通过上述方法,可以在 `QTableWidget` 的左侧绘制括号线,提升界面的可视化效果用户交互体验。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值