我有这2个c文件和一个头文件。所有这些文件都是由Makefile
构建的。
mysql_storage.h
#include <stdio.h>
#include <mysql.h>
#include <glib/gstdio.h>
#if 1
typedef enum {
STORAGE_OK = 0,
STORAGE_NO_SUCH_USER,
STORAGE_INVALID_PASSWORD,
STORAGE_ALREADY_EXISTS,
STORAGE_OTHER_ERROR/* Error that isn't caused by user input, such as
a database that is unreachable. log() will be
used for the exact error message */
} storage_status_t;
typedef enum
{
MYSQL_PASS_CHECK_ONLY = -1,
MYSQL_PASS_UNKNOWN = 0,
MYSQL_PASS_WRONG,
MYSQL_PASS_OK
} mysql_pass_st;
#endif
static void mysql_storage_init( void );
static void mysql_storage_deinit( void );
static storage_status_t mysql_storage_load(const char *password );
static storage_status_t mysql_storage_check_pass( const char *nick, const char *password );
static storage_status_t mysql_storage_save( int overwrite );
static storage_status_t mysql_storage_remove( const char *nick, const char *password );
mysql_storage.c
#include "mysql_storage.h"
MYSQL *mysql;
static void mysql_storage_init( void ) {
if (mysql_init(mysql) == NULL)
fprintf( stderr, "Can not initialize MySQL. Configuration won't be saved.");
if (!mysql_real_connect(mysql, "localhost", "USERNAME", "PASSWORD", NULL, 3306, NULL, 0))
fprintf( stderr, "%s\nConfiguration won't be saved.", mysql_error(mysql));
if (mysql_select_db(mysql, "DATABASENAME"))
fprintf( stderr, "%s\nConfiguration won't be saved.", mysql_error(mysql));
}
static void mysql_storage_deinit( void ) {
if(mysql!=NULL)
mysql_close(mysql);
}
static storage_status_t mysql_storage_check_pass( const char *nick, const char *password ){ return STORAGE_OTHER_ERROR;}
static storage_status_t mysql_storage_load(const char *password ){ return STORAGE_OTHER_ERROR;}
static storage_status_t mysql_storage_save( int overwrite ){ return STORAGE_OTHER_ERROR;}
static storage_status_t mysql_storage_remove( const char *nick, const char *password ){ return STORAGE_OTHER_ERROR;}
的main.c
#include<stdio.h>
#include "mysql_storage.h"
int main(){
// connect mysql
mysql_storage_init();
// check a password
printf("check password: %d\n:", mysql_storage_check_pass("hello", "world"));
// close mysql
mysql_storage_deinit();
}
最后是 Makefile
CFLAGS=-I/usr/include/mysql -DBIG_JOINS=1 -fno-strict-aliasing -g -I/usr/include/glib-2.0 -I/usr/lib/x86_64-linux-gnu/glib-2.0/include -I.
LIBS=-L/usr/lib/x86_64-linux-gnu -lmysqlclient -lpthread -lz -lm -lrt -ldl -lglib-2.0 -L.
CC=gcc
all: main
clean:
rm main *.o
main: mysql_storage.o main.o
$(CC) main.o mysql_storage.o $(LIBS) -o main
.c.o:
$(CC) -c $(CFLAGS) $<
当我做的时候,我收到了这个错误。
gcc main.o mysql_storage.o -L/usr/lib/x86_64-linux-gnu -lmysqlclient -lpthread -lz -lm -lrt -ldl -lglib-2.0 -L. -o main
main.o: In function `main':
/home/shiplu/projects/mysql_interface/main.c:8: undefined reference to `mysql_storage_init'
/home/shiplu/projects/mysql_interface/main.c:11: undefined reference to `mysql_storage_check_pass'
/home/shiplu/projects/mysql_interface/main.c:15: undefined reference to `mysql_storage_deinit'
collect2: ld returned 1 exit status
正如您所看到的,我正在关联main.o
和mysql_storage.o
。所以它不应该抛出未定义的引用错误。这有什么问题?
答案 0 :(得分:4)
您已将自己的职能mysql_storage_init
声明为static
,这意味着他们有内部linkage - 这意味着他们对其他翻译单位不可见。
由于您需要从其他对象文件中调用它们,因此您应该通过在声明和定义中省略static
关键字来声明它们具有外部链接。头文件中的原型可以选择声明它们extern
,但这不是必需的,因为默认情况下链接是外部的。
答案 1 :(得分:3)
您将mysql_storage.c函数定义为static
,这意味着它们在文件外部不可见。删除static
,它应该有效。
答案 2 :(得分:3)
函数的static
关键字表示此函数的范围仅为此.obj文件,这可能不是您所期望的。删除函数定义和声明的static
关键字(.h和.cpp文件)
答案 3 :(得分:1)
在函数之前使用关键字static可确保它们仅在定义它们的编译单元中可见。如果从函数中删除static关键字,则应该没有问题。
答案 4 :(得分:0)
如果您计划在不同的编译单元中使用函数,则不应将它们声明为静态。