微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

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

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

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

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

DBTable 注解:

<div class="jb51code">
<pre class="brush:java;">
/**

  • 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 约束注解:

<div class="jb51code">
<pre class="brush:java;">
/**

  • 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注解:

<div class="jb51code">
<pre class="brush:java;">
/**

  • 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 字符注解:

rush:java;"> /** * 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;
}

创建表的处理器:

<div class="jb51code">
<pre class="brush:java;">
/**

  • 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注解存在<a href="https://www.jb51.cc/tag/huoqu/" target="_blank" class="keywords">获取</a>表明 
String tableName = dbTable.name();
//判断表名是否存在
if (tableName.length() < 1)
{
  //不存在,说明<a href="https://www.jb51.cc/tag/mo/" target="_blank" class="keywords">默</a>认就是类名,通过 cl.getSimpleName()<a href="https://www.jb51.cc/tag/huoqu/" target="_blank" class="keywords">获取</a>类名并且大写
  tableName = cl.getSimpleName().<a href="https://www.jb51.cc/tag/toupper/" target="_blank" class="keywords">toupper</a>Case();
}

//定义<a href="https://www.jb51.cc/tag/huoqu/" target="_blank" class="keywords">获取</a>column的容器
List<String> columnDefs = new ArrayList<String>();
//循环<a href="https://www.jb51.cc/tag/shuxing/" target="_blank" class="keywords">属性</a>字段
//说明:getDeclaredFields()获得某个类的所有申明的字段,即<a href="https://www.jb51.cc/tag/baokuo/" target="_blank" class="keywords">包括</a>public、private和proteced,但是不<a href="https://www.jb51.cc/tag/baokuo/" target="_blank" class="keywords">包括</a><a href="https://www.jb51.cc/tag/fulei/" target="_blank" class="keywords">父类</a>的申明字段。 
//getFields()获得某个类的所有的公共(public)的字段,<a href="https://www.jb51.cc/tag/baokuo/" target="_blank" class="keywords">包括</a><a href="https://www.jb51.cc/tag/fulei/" target="_blank" class="keywords">父类</a>。 
for (Field field : cl.getDeclaredFields())
{
  //定义表字段<a href="https://www.jb51.cc/tag/mingcheng/" target="_blank" class="keywords">名称</a>变量
  String columnName = null;
  //<a href="https://www.jb51.cc/tag/huoquziduan/" target="_blank" class="keywords">获取字段</a>上的注解(现在字段允许多个注解,因此返回的是数组)
  Annotation[] anns = field.getDeclaredAnnotations();
  //判断<a href="https://www.jb51.cc/tag/shuxing/" target="_blank" class="keywords">属性</a>是否存在注解
  if (anns.length < 1)
    continue;

  //判断是否是我们定义的数据类型
  if (anns[0] instanceof <a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>Integer)
  {
    //<a href="https://www.jb51.cc/tag/huoqu/" target="_blank" class="keywords">获取</a><a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>Integer 注解
    <a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>Integer sInt = (<a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>Integer)anns[0];
    //判断是否注解的name是否有值
    if (sInt.name().length() < 1)
    {
      //如果没有值,说明是类的<a href="https://www.jb51.cc/tag/shuxing/" target="_blank" class="keywords">属性</a>字段,<a href="https://www.jb51.cc/tag/huoqu/" target="_blank" class="keywords">获取</a><a href="https://www.jb51.cc/tag/shuxing/" target="_blank" class="keywords">属性</a>并转换大写
      columnName = field.getName().<a href="https://www.jb51.cc/tag/toupper/" target="_blank" class="keywords">toupper</a>Case();
    }
    else
    { //如果有值,<a href="https://www.jb51.cc/tag/huoqu/" target="_blank" class="keywords">获取</a>设置的name值
      columnName = sInt.name();
    }
    //放到<a href="https://www.jb51.cc/tag/shuxing/" target="_blank" class="keywords">属性</a>的容器内
    columnDefs.add(columnName + " INT " + getConstraints(sInt.constraints()));
  }

  //同上<a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>Integer,这里不写注释了
  if (anns[0] instanceof <a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>String)
  {
    <a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>String <a href="https://www.jb51.cc/tag/sst/" target="_blank" class="keywords">sst</a>ring = (<a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>String)anns[0];
    if (<a href="https://www.jb51.cc/tag/sst/" target="_blank" class="keywords">sst</a>ring.name().length() < 1)
    {
      columnName = field.getName().<a href="https://www.jb51.cc/tag/toupper/" target="_blank" class="keywords">toupper</a>Case();
    }
    else
    {
      columnName = <a href="https://www.jb51.cc/tag/sst/" target="_blank" class="keywords">sst</a>ring.name();
    }
    columnDefs.add(columnName + " VARCHAR(" + <a href="https://www.jb51.cc/tag/sst/" target="_blank" class="keywords">sst</a>ring.value() + ")" + getConstraints(<a href="https://www.jb51.cc/tag/sst/" target="_blank" class="keywords">sst</a>ring.constraints()));
  }

  //定义<a href="https://www.jb51.cc/tag/shengcheng/" target="_blank" class="keywords">生成</a>创建表的<a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>语句
  StringBuilder createCommand = new StringBuilder("CREATE TABLE " + tableName + "(");
  //循环上面<a href="https://www.jb51.cc/tag/shuxing/" target="_blank" class="keywords">属性</a>容器,
  for (String columnDef : columnDefs)
  {
    //把<a href="https://www.jb51.cc/tag/shuxing/" target="_blank" class="keywords">属性</a><a href="https://www.jb51.cc/tag/tianjia/" target="_blank" class="keywords">添加</a>到<a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a>语句中
    createCommand.append("\n  " + columnDef + ",");
    //去掉最后<a href="https://www.jb51.cc/tag/yige/" target="_blank" class="keywords">一个</a>逗号
    String tableCreate = createCommand.substring(0,createCommand.length() - 1) + ");";
    //打印
    Sy<a href="https://www.jb51.cc/tag/stem/" target="_blank" class="keywords">stem</a>.out.println("Table creation <a href="https://www.jb51.cc/tag/sql/" target="_blank" class="keywords">sql</a> 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 。

收获:

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

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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程之家。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐