用注解编写创建表的SQL语句

原创 2016年08月30日 00:27:07

今晚读了think in java 的章节,感觉很不错,我就敲了下来,贴上代码给以后一个回顾:

建议提前读一下think in java 注解 。

说明创建注解我在第一个注解说明下,以后的注解不在说明。‘

DBTable 注解:

/**
* Project Name:myannotation
* File Name:DBTable.java
* Package Name:com.iflytek.db
* Date:2016-8-28下午08:20:54
* Copyright (c) 2016, syzhao@iflytek.com All Rights Reserved.
*
*/

package com.iflytek.db;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
@Target:
   @Target说明了Annotation所修饰的对象范围:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循环变量、catch参数)。在Annotation类型的声明中使用了target可更加明晰其修饰的目标。
  作用:用于描述注解的使用范围(即:被描述的注解可以用在什么地方)
  取值(ElementType)有:
    1.CONSTRUCTOR:用于描述构造器
    2.FIELD:用于描述域
    3.LOCAL_VARIABLE:用于描述局部变量
    4.METHOD:用于描述方法
    5.PACKAGE:用于描述包
    6.PARAMETER:用于描述参数
    7.TYPE:用于描述类、接口(包括注解类型) 或enum声明

 @Retention:
  @Retention定义了该Annotation被保留的时间长短:某些Annotation仅出现在源代码中,而被编译器丢弃;而另一些却被编译在class文件中;编译在class文件中的Annotation可能会被虚拟机忽略,而另一些在class被装载时将被读取(请注意并不影响class的执行,因为Annotation与class在使用上是被分离的)。使用这个meta-Annotation可以对 Annotation的“生命周期”限制。
  作用:表示需要在什么级别保存该注释信息,用于描述注解的生命周期(即:被描述的注解在什么范围内有效)
  取值(RetentionPoicy)有:
    1.SOURCE:在源文件中有效(即源文件保留)
    2.CLASS:在class文件中有效(即class保留)
    3.RUNTIME:在运行时有效(即运行时保留)
  Retention meta-annotation类型有唯一的value作为成员,它的取值来自java.lang.annotation.RetentionPolicy的枚举类型值
 */
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface DBTable
{
    public String name() default "";
}

Constraints 约束注解:

/**
* Project Name:myannotation
* File Name:Constraints.java
* Package Name:com.iflytek.db
* Date:2016-8-28下午08:27:08
* Copyright (c) 2016, syzhao@iflytek.com All Rights Reserved.
*
*/

package com.iflytek.db;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Constraints
{
    boolean primaryKey() default false;
    
    boolean allowNull() default true;
    
    boolean unique() default false;
}

SQLInteger int注解:

/**
* Project Name:myannotation
* File Name:SQLInteger.java
* Package Name:com.iflytek.db
* Date:2016-8-29下午10:24:11
* Copyright (c) 2016, syzhao@iflytek.com All Rights Reserved.
*
*/

package com.iflytek.db;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface SQLInteger
{
    String name() default "";
    
    Constraints constraints() default @Constraints;
}

SQLString 字符注解:

/**
* Project Name:myannotation
* File Name:SQLString.java
* Package Name:com.iflytek.db
* Date:2016-8-29下午10:28:04
* Copyright (c) 2016, syzhao@iflytek.com All Rights Reserved.
*
*/

package com.iflytek.db;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface SQLString
{
    int value() default 0;
    
    String name() default "";
    
    Constraints constraints() default @Constraints;
}


domain类:

/**
* Project Name:myannotation
* File Name:Member.java
* Package Name:com.iflytek.table
* Date:2016-8-29下午10:44:16
* Copyright (c) 2016, syzhao@iflytek.com All Rights Reserved.
*
*/

package com.iflytek.table;

import com.iflytek.db.Constraints;
import com.iflytek.db.DBTable;
import com.iflytek.db.SQLInteger;
import com.iflytek.db.SQLString;

@DBTable(name = "MEMBER")
public class Member
{
    @SQLString(30)
    String firstName;
    
    @SQLString(30)
    String lastName;
    
    @SQLInteger
    Integer age;
    
    @SQLString(value = 40, constraints = @Constraints(primaryKey = true))
    String handle;
    
    static int memberCount;
    
    public String getHandle()
    {
        return handle;
    }
    
    public String getFirstName()
    {
        return firstName;
    }
    
    public String getLastName()
    {
        return lastName;
    }
    
    @Override
    public String toString()
    {
        return handle;
    }
    
    public Integer getAge()
    {
        return age;
    }
    
}



创建表的处理器:

/**
* Project Name:myannotation
* File Name:TableCreator.java
* Package Name:com.iflytek.table
* Date:2016-8-29下午10:57:52
* Copyright (c) 2016, syzhao@iflytek.com All Rights Reserved.
*
*/

package com.iflytek.table;

import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.List;

import com.iflytek.db.Constraints;
import com.iflytek.db.DBTable;
import com.iflytek.db.SQLInteger;
import com.iflytek.db.SQLString;

public class TableCreator
{
    public static void main(String[] args)
    {
        createTable(Member.class);
    }
    
    //创建表SQL语句
    private static void createTable(Class<?> cl)
    {
        //获取DBTable注解
        DBTable dbTable = cl.getAnnotation(DBTable.class);
        //判断DBTable注解是否存在
        if (dbTable == null)
        {
            System.out.println("没有找到关于DBTable");
            return;
        }
        
        //如果@DBTable注解存在获取表明  
        String tableName = dbTable.name();
        //判断表名是否存在
        if (tableName.length() < 1)
        {
            //不存在,说明默认就是类名,通过 cl.getSimpleName()获取类名并且大写
            tableName = cl.getSimpleName().toUpperCase();
        }
        
        //定义获取column的容器
        List<String> columnDefs = new ArrayList<String>();
        //循环属性字段
        //说明:getDeclaredFields()获得某个类的所有申明的字段,即包括public、private和proteced,但是不包括父类的申明字段。 
        //getFields()获得某个类的所有的公共(public)的字段,包括父类。 
        for (Field field : cl.getDeclaredFields())
        {
            //定义表字段名称变量
            String columnName = null;
            //获取字段上的注解(现在字段允许多个注解,因此返回的是数组)
            Annotation[] anns = field.getDeclaredAnnotations();
            //判断属性是否存在注解
            if (anns.length < 1)
                continue;
            
            //判断是否是我们定义的数据类型
            if (anns[0] instanceof SQLInteger)
            {
                //获取SQLInteger 注解
                SQLInteger sInt = (SQLInteger)anns[0];
                //判断是否注解的name是否有值
                if (sInt.name().length() < 1)
                {
                    //如果没有值,说明是类的属性字段,获取属性并转换大写
                    columnName = field.getName().toUpperCase();
                }
                else
                { //如果有值,获取设置的name值
                    columnName = sInt.name();
                }
                //放到属性的容器内
                columnDefs.add(columnName + " INT " + getConstraints(sInt.constraints()));
            }
            
            //同上SQLInteger,这里不写注释了
            if (anns[0] instanceof SQLString)
            {
                SQLString sString = (SQLString)anns[0];
                if (sString.name().length() < 1)
                {
                    columnName = field.getName().toUpperCase();
                }
                else
                {
                    columnName = sString.name();
                }
                columnDefs.add(columnName + " VARCHAR(" + sString.value() + ")" + getConstraints(sString.constraints()));
            }
            
            //定义生成创建表的SQL语句
            StringBuilder createCommand = new StringBuilder("CREATE TABLE " + tableName + "(");
            //循环上面属性容器,
            for (String columnDef : columnDefs)
            {
                //把属性添加到sql语句中
                createCommand.append("\n    " + columnDef + ",");
                //去掉最后一个逗号
                String tableCreate = createCommand.substring(0, createCommand.length() - 1) + ");";
                //打印
                System.out.println("Table creation SQL for " + cl.getName() + " is :\n" + tableCreate);
            }
        }
    }
    
    private static String getConstraints(Constraints con)
    {
        String constraints = "";
        //判断是否为null
        if (!con.allowNull())
        {
            constraints += " NOT NULL ";
        }
        //判断是否是主键
        if (con.primaryKey())
        {
            constraints += " PRIMARY KEY ";
        }
        //是否唯一
        if (con.unique())
        {
            constraints += " UNIQUE ";
        }
        
        return constraints;
    }
}

以上代码拷贝出来,就可以运行了!

上面虽然是简单的创建表语句,但我们可以蔓延到hibernate的domain类里的注解,各种CURD ,何尝不是这样处理的呢,只是hibernate有很多东西,但是万变不离其宗,以后有机会研究一下hibernate 。

收获:

读了以后,对于注解知道为什么要这么用了,其实顾名思义就是一个注解,只是有一个处理器来处理这个注解,这对我以后用到注解方面应该有帮助的,

时间不早了,就写到这里!


结果:


版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

SQL 注释语句 ("--"与"/*...*/")

(1)--:表示单行注释(2)/*…*/:用于多行(块)注释例:   use pubs   --选择数据库      --first  line of a multiple-line comment....
  • txl816
  • txl816
  • 2008年07月09日 09:48
  • 34654

Java注解(Annotation)详解(四)——注解反射生成SQL语句

(四)注解反射生成SQL语句 1.要求         ①使用注解来代替Hibernate,实现持久层数据访问。         ②使用注解来对每个字段或字段的组合条件进行检索,并打印出SQL。   ...
  • Zen99T
  • Zen99T
  • 2015年12月18日 14:31
  • 4470

Delphi7高级应用开发随书源码

  • 2003年04月30日 00:00
  • 676KB
  • 下载

MyBatis 3(中文版) 第四章 使用注解配置SQL映射器

本章将涵盖以下话题: l 在映射器Mapper接口上使用注解 l 映射语句            @Insert,@Update,@Delete,@SeelctStateme...

MyBatis注解应用之动态SQL语句

有时候,我们需要在输入的标准下,创建动态的查的语言。MyBatis提供了多个注解如:@InsertProvider,@UpdateProvider,@DeleteProvider和@SelectPro...

使用自定义注解生成简单查询sql语句

在代码中注解可以给我们带来很多方便,而且在很多框架中都有注解的身影,那注解是怎么创建和使用的呢。这里大概分以下几个步骤: 1.创建自定义的注解 2.编写注解解析,(及定义这个注解的目的) 3.按照文明...

sql语句,sql文件加注释

单行注释通用-- 如:sql server、oracle、db2等; 多行注释/**/ 如:sql server,其他不知道 access 好象不可以有注释 注释后面好像需要加“;”...

给定A, B两个整数,不使用除法和取模运算,求A/B的商和余数

给定A, B两个整数,不使用除法和取模运算,求A/B的商和余数。 1.   最基本的算法是,从小到大遍历: for (i = 2 to A -1)          if (i * B > A)...

利用K-means聚类算法根据经纬度坐标对中国省市进行聚类

K-means聚类算法是一种非层次聚类算法,在最小误差的基础上将数据划分了特定的类,类间利用距离作为相似度指标,两个向量之间的距离越小,其相似度就越高。程序读取全国省市经纬度坐标,然后根据经纬度坐标进...

Radon变换理论介绍与matlab实现--经验交流

本人最近在研究Radon变换,在查阅了各种资料之后在此写下个人的理解,希望与各位牛牛进行交流共同进步,也使得理解更加深刻些。 Radon变换的本质是将原来的函数做了一个空间转换,即,将原来的XY平...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:用注解编写创建表的SQL语句
举报原因:
原因补充:

(最多只允许输入30个字)