Application Path
အရင်ဦးဆုံး ရေးထားတဲ့ အပလီမှာ JAX-RS ကို သုံးလို့ရအောင်လုပ်ပါဦးမယ်။ web.xml မှာ ဖြည့်ရေးတဲ့နည်းရယ်၊ Annotation ကို သုံးရေတဲ့နည်းရယ် ရှိပါတယ်။ ဒီအပလီမှာ အစထဲက web.xml ကို အသုံးမပြုထားပါဘူး။ အဲ့ဒီအတွက် Annotation ကိုပဲသုံးပြီးရေပါမယ်။ ရေးရမှာကတော့ javax.ws.rs.ApplicationPath ကို extends လုပ်ထားတဲ့ Class တစ်ခုကို ရေးရပါမယ်။ ပြီးရင်အဲ့ဒီမှာ @ApplicationPath ကို ဖြည့်ပြီးရေရုံပါပဲ။ အဲ့ဒီ Annotation ရဲ့ Parameter အဖြစ်နဲ့ အသုံးပြုလိုတဲ့ Path ကို ရေးလိုက်တာနဲ့ အဲ့ဒီ Path ဟာ Rest Application ရဲ့ Root End Point အဖြစ်အသုံးပြုနိုင်မှာဖြစ်ပါတယ်။
@ApplicationPath(value = "/") public class ServiceApplication extends Application { }
အထဲမှာ "/" လို့ရေထားတဲ့အတွက် Web Application ရဲ့ Root Path ကို Rest ရဲ့ Root End Point အဖြစ်အသုံးပြုနိုင်မှာဖြစ်ပါတယ်။
Abstract Service Class
RESTFul Web Service ကို ရေးလို့ရပြီဆိုတော့ Entity တွေကို Resources အဖြစ်ဖော်ပြဖို့အတွက် Abstract Class တစ်ခုကိုရေးပါမယ်။
public abstract class AbstractService<T> { @Inject protected Dao<T> dao; protected Class<T> entity; protected abstract void setEntity(); @PostConstruct public void init() { setEntity(); } @GET @Produces(value = MediaType.APPLICATION_JSON) public List<T> getAll() { return dao.findAll(entity); } @GET @Path("{id}") @Produces(value = MediaType.APPLICATION_JSON) public T findById(@PathParam(value = "id") Integer id) { return dao.findById(id, entity); } @POST @Consumes(value = MediaType.APPLICATION_JSON) public T create(T t) { dao.persist(t); return t; } @PUT @Consumes(value =MediaType.APPLICATION_JSON) public T update(T t) { dao.update(t); return t; } @DELETE @Path("{id}") @Produces(MediaType.APPLICATION_JSON) public Response delete(@PathParam("id") int id) { this.dao.delete(dao.findById(id, entity)); return Response.noContent().build(); } }
ကျွှန်တော်တို့ဒီနေရာမှာ သတိထားရမှာက Abstract Class ကို Generics Type အနေနဲ့ရေးထားပါတယ်။ ပြီးတော့ အဲ့ဒီ Class ကို Extends လုပ်တဲ့ Concrete Class မှာ Type Parameter ကို ဖြည့်ပြီး ရေးပါမယ်။ ဥပမာအား ဖြင့် Student Resource အတွက် StudentService Class ကို ရေးမယ်ဆိုရင် StudentService extends AbstractService<Student> လို့ရေးပါမယ်။ Type Parameter အဖြစ်အသုံးပြုထားတဲ့ Student ကို အတွင်းမှာပါတဲ့ Type Parameter နေရာတွေမှာ အစားထိုးအသုံးပြုသွားနိုင်မှာဖြစ်ပါတယ်။
နောက်တစ်ခု သတိထားကြည့်ရမယ့်နေရာက abstract method ဖြစ်တဲ့ setEntity ကိုပါ။ ဒီအထဲမှာ Dao မှာအသုံးပြုမယ့် Entity Class ရဲ့ အမျိုးအစားကို ဖြည့်ပေးဖို့လိုအပ်ပါတယ်။ အဲ့ဒီအတွက် member အနေနဲ့ Class<T> entity လို့ရေသားထားပါတယ်။ entity ရဲ့ တန်ဖိုးကို Concrete Class တွေမှာ အစားထိုးစေလိုတဲ့အတွက် abstract method setEntity ကို ရေးသားထားရခြင်း ဖြစ်ပါတယ်။ အဲ့ဒီ setEntity ကို @PostConstruct Annotation တပ်ထားတဲ့ init method ထဲကနေခေါ်ပါတယ်။ Enterprise Bean တွေကို Container ကနေ လိုအပ်တဲ့အခါမှာ Create လုပ်ပေးမှာဖြစ်တဲ့အတွက် Constructor ကို အသုံးပြုလို့မရပါဘူး။ အဲ့ဒီအတွက် Constructor ထဲမှာ ရေးချင်တဲ့ Logic တွေကို Life Cycle Annotation ဖြစ်တဲ့ PostConstruct Annotation နဲ့ရေးထားတဲ့ Method ထဲမှာ ရေးထားရခြင်း ဖြစ်ပါတယ်။ AbstractService ကို Extends လုပ်ထားတဲ့ Concrete Class ထဲမှာ setEntity ကို Override လုပ်ရေးထားပြီး၊ အတွင်းမှာ entity ကို အသုံးပြုမယ့် Class နဲ့ အစားထိုးထားမယ်ဆိုရင် အဲ့ဒီ Concrete Class ကို Construct လုပ်ပြီးတာနဲ့ entity ရဲ့တန်ဖိုးကို အခြား method တွေကနေ အသုံးပြုနိုင်မှာဖြစ်ပါတယ်။
ကျန်တာတွေကတော့ Resource တွေမှာ အသုံးပြုလိုတဲ့ CRUD Method တွေဖြစ်ပါတယ်။ Reference အတွက် အားလုံးကို ရှာမယ့် getAll method နဲ့ findById ကို ရေးသားထားပါတယ်။
getAll ရဲ့ Return Type ကတော့ List<T> ဖြစ်ပါတယ်။ ပြီးတော့ @GET Annotation ကို အသုံးပြုထားပါတယ်။ သက်ဆိုင်ရာ Resource ရဲ့ URI ကို HTTP Get method နဲ့ Request လုပ်လာရင် အဲ့ဒီ Resource အားလုံးကို ပြန်လည် ရရှိမှာဖြစ်ပါတယ်။ နောက်ပြီး ဒီအပလီမှာ Representation Type အနေနဲ့ JSON ကို အသုံးပြုချင်တဲ့အတွက် @Produces Annotation ရဲ့ Parameter မှာ MediaType.APPLICATION_JSON လို့ ရေးသားထားပါတယ်။
နောက်Method တစ်ခုကတော့ findById ပါ။ Argument အနေနဲ့ Integer id ကို အသုံးပြုထားပြီး၊ Return Type ကတော့ Generices Type ဖြစ်တဲ့ T ပါ။ Parameter အဖြစ် id ရဲ့တန်ဖိုးကို ပေးတာနဲ့ သက်ဆိုင်ရာ Object ကို ရရှိမှာဖြစ်ပါတယ်။ ဒီနေရာမှာ သတိပြုစေလိုတာက အဲ့ဒီ id ကို ဘယ်က ယူမလဲဆိုတဲ့အချက်ပါပဲ။ အဲ့ဒီ id ကို Path Parameter ကနေရယူလိုတဲ့အတွက် @Path("{id}") လို့ Path ကို ရေးထားပါတယ်။ ဥပမာအားဖြင့် Student Resource ကို သုံးတဲ့နေရာမှာ /student/1 လို့ရေထားရင် 1 အား id အဖြစ်ရရှိမှာ ဖြစ်ပါတယ်။ အဲ့ဒီလို ရယူနိုင်ရန်အတွက်လည်း Argument id ရှေ့တွင် @PathParam("id") Integer id လို့ရေသားထားရခြင်းဖြစ်တယ်။ ထိုကဲ့သို့ရေးထားပါက Argument id ၏တန်ဖိုးအား Path Parameter id မှ ရယူသွားမည် ဖြစ်ပါသည်။
ကျန်တဲ့ Method တွေကတော့ create, update နဲ့ delete တို့ပါပဲ။ ၎င်းတို့ကိုလည်း သက်ဆိုင်ရာ HTTP POST, PUT နဲ့ DELETE တို့ကို အသုံးပြုထားပါတယ်။
အသုံးပြုလိုတဲ့ Resource Class တွေမှာ အထက်ပါ AbstractService ကို extends လုပ်ရေးရုံနဲ့ အထက်ပါ CRUD method တွေကို အသုံးပြုနိုင်မှာ ဖြစ်ပါတယ်။နောက်ရက်များမှပဲ Resources တွေအတွက် Concrete Class တွေကို ရေးသားပါတော့မယ်။
လေးစားစွာဖြင့်
မင်းလွင်
နားေတာ့မလည္ဘူး ဒါေပမဲ့ ယခုလိုေစတနာထားတာကိုေလးစားပါတယ္
ReplyDeleteI will write the concrete class for this abstract class. In this post I will explain more details. Anyway thank you for reading my blog.
ReplyDeleteအခုလိုသင်ဦကားေပးတဲဥအတွက်ေကျးဇူးတင်ပါတ,်
ReplyDeleteThanks a lot sir
ReplyDeleteConcrete class ရေးတဲ့အပိုင်းလေးတွေ ဆက်တင်ပေးပါအုံးခင်ဗျာ
ReplyDelete