[英]using C++ with namespace in C
在Linux上使用Eclpse我已经定义了一个名为ABC(ABC.hpp)的C ++类:
#ifndef ABC_HPP_
#define ABC_HPP_
#include <stdio.h>
// namespace my { <---- COMMENTED OUT
class ABC {
private:
int m_number;
public:
ABC(int p_number);
void doSomething();
virtual ~ABC();
};
// } /* namespace my */ <---- COMMENTED OUT
#endif /* ABC_HPP_ */
它的实现是(ABC.cpp):
#include "ABC.hpp"
// using namespace my; <---- COMMENTED OUT
ABC::ABC(int p_number) {
this->m_number = p_number;
}
ABC::~ABC() {
this->m_number = -1;
}
void ABC::doSomething() {
printf("doing something (%d)\n", this->m_number);
}
要在C程序中使用此类,我创建了一个包含这些方法的层(ABCwrapper.h):
typedef void CABC;
#ifdef __cplusplus
extern "C" {
#endif
CABC* create_abc();
void call_abc_methods(const CABC *p_abc);
void destroy_abc(CABC *p_abc);
#ifdef __cplusplus
} // extern "C"
#endif
和
#include "ABC.hpp"
#include "ABCWrapper.h"
extern "C" {
CABC* create_abc() {
ABC* abc = new ABC();
return (ABC*)abc;
}
void call_abc_methods(const CABC *p_abc) {
ABC* abc = (ABC*)p_abc;
abc->doSomething();
}
void destroy_abc(CABC *p_abc) {
ABC* abc = (ABC*)p_abc;
delete abc;
}
}
那很好,我可以使用ABC类实例。 但是如何在名称空间中定义ABC类,让我们说“我的”? 如果我从所有名称空格行中删除注释符号,IDE会抱怨说“类型'ABC'无法解析”。
如果我想扩展我的C ++库,我必须为我的类使用名称空间,但后来我不知道如何在包装器中使用。 拜托,帮我解开这个谜。
谢谢。
SK
在ABCWrapper.cpp
,在extern "C" {
line,上面添加:
using my::ABC;
要么
using namespace my;
约翰的建议(用ABCWrapper.cpp
my::ABC
替换ABC
所有实例)也有效。
关于代码的一些主要是次要的挑剔。 您已经获得了具有活动命名空间的解决方案的详细信息,但是应该解决各种小问题。
你最好引入一个不完整的类型, typedef struct CABC CABC;
(代替typedef void CABC;
),以便在C代码中获得某种类型的安全性。 这会阻止您将随机FILE *
传递给C接口函数,而使用void
函数则不会。
当您使用不完整类型时,您应该在C包装函数中使用reinterpret_cast<ABC *>(abc)
而不是C样式转换。 编译器然后在call_abc_methods()
函数中显示const-ness的问题; 参数不应该是const
(但C风格的转换隐藏了问题)。
此外,您的ABC.hpp
标题显示常见(次要)错误; 它包含一个无关的头文件( #include <stdio.h>
),安全使用头文件不需要它。 该行应仅出现在实现文件ABC.cpp
,其中代码使用<stdio.h>
的服务。 大多数标题应该#include
只包含使标题可单独使用所需的其他标题。 它们不应包含随机的其他标题。
这是一个完整的工作程序 - 它有很多文件。 有3个标题:
ABC.hpp
- 声明class ABC
。 ABCwrapper.h
- 将C接口声明为class ABC
。 ABCprogram.h
- 双语标题声明其他功能。 有1个C文件:
ABCuser.c
- 必须有一些C代码需要使用C接口来class ABC
才能使整个练习变得有价值,就是这样。 有3个C ++文件:
ABC.cpp
- 定义class ABC
。 ABCwrapper.cpp
- 定义class ABC
的C接口。 ABCmain.cpp
- 双语系统中的主程序通常应该用C ++编写。 还有一个makefile
。
#ifndef ABC_HPP_INCLUDED
#define ABC_HPP_INCLUDED
namespace abc_library {
class ABC {
private:
int m_number;
public:
ABC(int p_number);
void doSomething();
virtual ~ABC();
};
} /* namespace abc_library */
#endif /* ABC_HPP_INCLUDED */
#ifndef ABCWRAPPER_H_INCLUDED
#define ABCWRAPPER_H_INCLUDED
typedef struct CABC CABC; // Pointer to this ncomplete type used in C code
#ifdef __cplusplus
extern "C" {
#endif
CABC *create_abc(int val);
void call_abc_methods(CABC *p_abc);
void destroy_abc(CABC *p_abc);
#ifdef __cplusplus
}
#endif
#endif /* ABCWRAPPER_H_INCLUDED */
#ifndef ABCPROGRAM_H_INCLUDED
#define ABCPROGRAM_H_INCLUDED
#if defined(__cplusplus)
extern "C" {
#endif
extern int c_code_function(int init);
#if defined(__cplusplus)
}
#endif
#endif /* ABCPROGRAM_H_INCLUDED */
#include "ABCwrapper.h"
#include "ABCprogram.h"
int c_code_function(int init)
{
CABC *abc = create_abc(init);
call_abc_methods(abc);
destroy_abc(abc);
return 0;
}
#include "ABC.hpp"
#include <stdio.h>
using namespace abc_library;
ABC::ABC(int p_number) {
this->m_number = p_number;
}
ABC::~ABC() {
this->m_number = -1;
}
void ABC::doSomething() {
printf("doing something (%d)\n", this->m_number);
}
#include "ABC.hpp"
#include "ABCwrapper.h"
using namespace abc_library;
extern "C" {
CABC *create_abc(int val) {
ABC* abc = new ABC(val);
return reinterpret_cast<CABC*>(abc);
}
void call_abc_methods(CABC *p_abc) {
ABC *abc = reinterpret_cast<ABC *>(p_abc);
abc->doSomething();
}
void destroy_abc(CABC *p_abc) {
ABC* abc = reinterpret_cast<ABC *>(p_abc);
delete abc;
}
}
#include "ABCprogram.h"
int main()
{
return c_code_function(39);
}
CC = gcc # /usr/bin/gcc
CXX = g++
RM_FR = rm -fr --
WFLAGS = -Wall -Wextra -Wmissing-prototypes -Wstrict-prototypes -Wold-style-definition
SFLAGS = -std=c99
OFLAGS = -g -O3
UFLAGS = # Set on make command line only
OXXFLAGS = -g -O3
SXXFLAGS = -std=c++11
WXXFLAGS = -Wall -Wextra
UXXFLAGS = # Set on make command line only
LDFLAGS =
LDLIBS =
CFLAGS = ${OFLAGS} ${SFLAGS} ${WFLAGS} ${UFLAGS}
CXXFLAGS = ${OXXFLAGS} ${SXXFLAGS} ${WXXFLAGS} ${UXXFLAGS}
PROGRAM = abc
FILES.cpp = \
ABC.cpp \
ABCmain.cpp \
ABCwrapper.cpp
FILES.c = \
ABCuser.c
FILES.h = \
ABCprogram.h \
ABCwrapper.h
FILES.o = ${FILES.cpp:.cpp=.o} ${FILES.c:.c=.o}
all: ${PROGRAM}
${PROGRAM}: ${FILES.o}
${CXX} -o $@ ${CXXFLAGS} ${FILES.o} ${LDFLAGS} ${LDLIBS}
clean:
${RM_FR} *.o *.dSYM core a.out
depend:
mkdep ${FILES.cpp} ${FILES.c}
# DO NOT DELETE THIS LINE or the blank line after it -- make depend uses them.
ABC.o: ABC.cpp
ABC.o: ABC.hpp
ABCmain.o: ABCmain.cpp
ABCmain.o: ABCprogram.h
ABCuser.o: ABCprogram.h
ABCuser.o: ABCuser.c
ABCuser.o: ABCwrapper.h
ABCwrapper.o: ABC.hpp
ABCwrapper.o: ABCwrapper.cpp
ABCwrapper.o: ABCwrapper.h
您必须为ABC类设置范围。 因此,除了类声明之外,将所有ABC类替换为我的:: ABC。
extern "C" {
CABC* create_abc() {
my::ABC* abc = new my::ABC();
return (my::ABC*)abc;
}
void call_abc_methods(const CABC *p_abc) {
my::ABC* abc = (my::ABC*)p_abc;
abc->doSomething();
}
void destroy_abc(CABC *p_abc) {
my::ABC* abc = (my::ABC*)p_abc;
delete abc;
}
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.