- 浏览: 306298 次
文章分类
最新评论
-
shoru:
最新的netbeans scala插件无法安装到6.9中呀,求 ...
Scala-对Java的修正和超越-Presentation -
alanwu:
night_stalker 写道讲一次多少钱啊,能透露下么…… ...
Scala-对Java的修正和超越-Presentation -
wzjin:
暂时没有考虑学习,等有一定市场再看!
Scala-对Java的修正和超越-Presentation -
dcaoyuan:
jasspier 写道对scala有点兴趣,有没有入门的推荐教 ...
Scala-对Java的修正和超越-Presentation -
jasspier:
草原兄,听了你的课,对scala有点兴趣,有没有入门的推荐教程 ...
Scala-对Java的修正和超越-Presentation
3. The Magic Behind Erlyweb
With dynamic typing, hot code swapping, built-in parsing and compilation tools, Erlang is also suitable for dynamic meta-programming. Erlyweb uses a small convenient tool smerl to generate db record CRUD code automatically.
The music example on erlyweb.org shows the magic:
Define a simple musician.erl module (just one line here):
-module(musician).
Then, you will get a long list functions for musician module after erlyweb:compile("apps/music"). Sounds similar to Rails.
I like to watch magic show, but I can't stand that I do not know the things behind magic in programming. For Rails, the magic behind it always makes me headache, it's too difficult to follow who, where, some code are injected into a class or module. But in Erlang, it's super easy.
I add a simple function to erlyweb_app.erl (please see my previous post):
-export([decompile/2]). decompile(AppName, Beam) when is_atom(AppName) -> decompile(atom_to_list(AppName), Beam); decompile(AppName, Beam) when is_list(AppName) -> BinFilename = "./apps/" ++ AppName ++ "/ebin/" ++ atom_to_list(Beam), io:format("Beam file: ~s~n", [BinFilename]), {ok, {_, [{abstract_code, {_, AC}}]}} = beam_lib:chunks(BinFilename, [abstract_code]), SrcFilename = "./apps/" ++ AppName ++ "_" ++ atom_to_list(Beam), {ok, S} = file:open(SrcFilename ++ ".erl", write), io:fwrite(S, "~s~n", [erl_prettypr:format(erl_syntax:form_list(AC))]).
Now type erlyweb_decompile(music, musician) under the Erlang shell, I get a file: music_musician.erl under folder myproject/apps/ (I put these decompiled source files under /myproject/apps/ to avoid that they are auto-compiled to beams by erlyweb again ):
-file("musician", 1). -module(musician). -export([relations/0, fields/0, table/0, type_field/0, db_table/0, db_field/1, is_new/1, is_new/2, get_module/1, to_iolist/1, to_iolist/2, field_to_iolist/2, new/0, new_with/1, new_with/2, new_from_strings/1, set_fields/2, set_fields/3, set_fields_from_strs/2, field_from_string/2, save/1, insert/1, update/1, update/2, delete/1, delete_id/1, delete_where/1, delete_all/0, transaction/1, before_save/1, after_save/1, before_delete/1, after_delete/1, after_fetch/1, find/2, find_first/2, find_max/3, find_range/4, find_id/1, aggregate/4, count/0, get/2, set_related_one_to_many/2, find_related_one_to_many/2, find_related_many_to_one/4, aggregate_related_many_to_one/6, add_related_many_to_many/3, remove_related_many_to_many/3, remove_related_many_to_many_all/5, find_related_many_to_many/5, aggregate_related_many_to_many/7, find_related_many_first/4, find_related_many_max/5, find_related_many_range/6, aggregate_related_many/6, do_save/1, do_delete/1, field_names_for_query/0, field_names_for_query/1, field_to_iolist/1, set/3, db_pk_fields/0, get_pk_fk_fields/0, get_pk_fk_fields2/0, db_fields/0, db_field_names/0, db_field_names_str/0, db_field_names_bin/0, db_num_fields/0, id/1, id/2, name/1, name/2, birth_date/1, birth_date/2, instrument/1, instrument/2, bio/1, bio/2, new/4, driver/0, count/3, count/1, count/2, count_with/2, avg/3, avg/1, avg/2, avg_with/2, min/3, min/1, min/2, min_with/2, max/3, max/1, max/2, max_with/2, sum/3, sum/1, sum/2, sum_with/2, stddev/3, stddev/1, stddev/2, stddev_with/2, find/0, find/1, find_with/1, find_first/0, find_first/1, find_first_with/1, find_max/1, find_max/2, find_max_with/2, find_range/2, find_range/3, find_range_with/3]). relations() -> erlydb_base:relations(). fields() -> erlydb_base:fields(). table() -> erlydb_base:table(). type_field() -> erlydb_base:type_field(). db_table() -> erlydb_base:db_table(musician). db_field(FieldName) -> erlydb_base:db_field(musician, FieldName). is_new(Rec) -> erlydb_base:is_new(Rec). is_new(Rec, Val) -> erlydb_base:is_new(Rec, Val). get_module(Rec) -> erlydb_base:get_module(Rec). to_iolist(Recs) -> erlydb_base:to_iolist(musician, Recs). to_iolist(Recs, ToIolistFun) -> erlydb_base:to_iolist(musician, Recs, ToIolistFun). field_to_iolist(Val, _Field) -> erlydb_base:field_to_iolist(Val, _Field). new() -> erlydb_base:new(musician). new_with(Fields) -> erlydb_base:new_with(musician, Fields). new_with(Fields, ToFieldFun) -> erlydb_base:new_with(musician, Fields, ToFieldFun). new_from_strings(Fields) -> erlydb_base:new_from_strings(musician, Fields). set_fields(Record, Fields) -> erlydb_base:set_fields(musician, Record, Fields). set_fields(Record, Fields, ToFieldFun) -> erlydb_base:set_fields(musician, Record, Fields, ToFieldFun). set_fields_from_strs(Record, Fields) -> erlydb_base:set_fields_from_strs(musician, Record, Fields). field_from_string(ErlyDbField, undefined) -> erlydb_base:field_from_string(ErlyDbField, undefined). save(Rec) -> erlydb_base:save(Rec). insert(Recs) -> erlydb_base:insert(Recs). update(Props) -> erlydb_base:update(musician, Props). update(Props, Where) -> erlydb_base:update(musician, Props, Where). delete(Rec) -> erlydb_base:delete(Rec). delete_id(Id) -> erlydb_base:delete_id(musician, Id). delete_where(Where) -> erlydb_base:delete_where(musician, Where). delete_all() -> erlydb_base:delete_all(musician). transaction(Fun) -> erlydb_base:transaction(musician, Fun). before_save(Rec) -> erlydb_base:before_save(Rec). after_save(Rec) -> erlydb_base:after_save(Rec). before_delete(Rec) -> erlydb_base:before_delete(Rec). after_delete({_Rec, Num}) -> erlydb_base:after_delete({_Rec, Num}). after_fetch(Rec) -> erlydb_base:after_fetch(Rec). find(Where, Extras) -> erlydb_base:find(musician, Where, Extras). find_first(Where, Extras) -> erlydb_base:find_first(musician, Where, Extras). find_max(Max, Where, Extras) -> erlydb_base:find_max(musician, Max, Where, Extras). find_range(First, Max, Where, Extras) -> erlydb_base:find_range(musician, First, Max, Where, Extras). find_id(Id) -> erlydb_base:find_id(musician, Id). aggregate(AggFunc, Field, Where, Extras) -> erlydb_base:aggregate(musician, AggFunc, Field, Where, Extras). count() -> erlydb_base:count(musician). get(Idx, Rec) -> erlydb_base:get(Idx, Rec). set_related_one_to_many(Rec, Other) -> erlydb_base:set_related_one_to_many(Rec, Other). find_related_one_to_many(OtherModule, Rec) -> erlydb_base:find_related_one_to_many(OtherModule, Rec). find_related_many_to_one(OtherModule, Rec, Where, Extras) -> erlydb_base:find_related_many_to_one(OtherModule, Rec, Where, Extras). aggregate_related_many_to_one(OtherModule, AggFunc, Rec, Field, Where, Extras) -> erlydb_base:aggregate_related_many_to_one(OtherModule, AggFunc, Rec, Field, Where, Extras). add_related_many_to_many(JoinTable, Rec, OtherRec) -> erlydb_base:add_related_many_to_many(JoinTable, Rec, OtherRec). remove_related_many_to_many(JoinTable, Rec, OtherRec) -> erlydb_base:remove_related_many_to_many(JoinTable, Rec, OtherRec). remove_related_many_to_many_all(JoinTable, OtherTable, Rec, Where, Extras) -> erlydb_base:remove_related_many_to_many_all(JoinTable, OtherTable, Rec, Where, Extras). find_related_many_to_many(OtherModule, JoinTable, Rec, Where, Extras) -> erlydb_base:find_related_many_to_many(OtherModule, JoinTable, Rec, Where, Extras). aggregate_related_many_to_many(OtherModule, JoinTable, AggFunc, Rec, Field, Where, Extras) -> erlydb_base:aggregate_related_many_to_many(OtherModule, JoinTable, AggFunc, Rec, Field, Where, Extras). find_related_many_first(Func, Rec, Where, Extras) -> erlydb_base:find_related_many_first(Func, Rec, Where, Extras). find_related_many_max(Func, Rec, Num, Where, Extras) -> erlydb_base:find_related_many_max(Func, Rec, Num, Where, Extras). find_related_many_range(Func, Rec, First, Last, Where, Extras) -> erlydb_base:find_related_many_range(Func, Rec, First, Last, Where, Extras). aggregate_related_many(Func, AggFunc, Rec, Field, Where, Extras) -> erlydb_base:aggregate_related_many(Func, AggFunc, Rec, Field, Where, Extras). do_save(Rec) -> erlydb_base:do_save(Rec). do_delete(Rec) -> erlydb_base:do_delete(Rec). field_names_for_query() -> erlydb_base:field_names_for_query(musician). field_names_for_query(UseStar) -> erlydb_base:field_names_for_query(musician, UseStar). field_to_iolist(Val) -> erlydb_base:field_to_iolist(Val). set(Idx, Rec, Val) -> setelement(Idx, Rec, Val). db_pk_fields() -> erlydb_base:db_pk_fields([{erlydb_field, id, "id", <<105, 100>>, int, 11, integer, text_field, false, primary, undefined, identity}]). get_pk_fk_fields() -> erlydb_base:get_pk_fk_fields([{id, musician_id}]). get_pk_fk_fields2() -> erlydb_base:get_pk_fk_fields2([{id, musician_id1, musician_id2}]). db_fields() -> erlydb_base:db_fields([{erlydb_field, id, "id", <<105, 100>>, int, 11, integer, text_field, false, primary, undefined, identity}, {erlydb_field, name, "name", <<110, 97, 109, 101>>, varchar, 20, binary, text_field, true, undefined, undefined, undefined}, {erlydb_field, birth_date, "birth_date", <<98, 105, 114, 116, 104, 95, 100, 97, 116, 101>>, date, undefined, date, text_field, true, undefined, undefined, undefined}, {erlydb_field, instrument, "instrument", <<105, 110, 115, 116, 114, 117, 109, 101, 110, 116>>, enum, [<<103, 117, 105, 116, 97, 114>>, <<112, 105, 97, 110, 111>>, <<100, 114, 117, 109, 115>>, <<118, 111, 99, 97, 108, 115>>], binary, select, true, undefined, undefined, undefined}, {erlydb_field, bio, "bio", <<98, 105, 111>>, text, undefined, binary, text_area, true, undefined, undefined, undefined}]). db_field_names() -> erlydb_base:db_field_names([id, name, birth_date, instrument, bio]). db_field_names_str() -> erlydb_base:db_field_names_str(["id", "name", "birth_date", "instrument", "bio"]). db_field_names_bin() -> erlydb_base:db_field_names_bin([<<105, 100>>, <<110, 97, 109, 101>>, <<98, 105, 114, 116, 104, 95, 100, 97, 116, 101>>, <<105, 110, 115, 116, 114, 117, 109, 101, 110, 116>>, <<98, 105, 111>>]). db_num_fields() -> erlydb_base:db_num_fields(5). id(Rec) -> erlydb_base:get(3, Rec). id(Rec, Val) -> setelement(3, Rec, Val). name(Rec) -> erlydb_base:get(4, Rec). name(Rec, Val) -> setelement(4, Rec, Val). birth_date(Rec) -> erlydb_base:get(5, Rec). birth_date(Rec, Val) -> setelement(5, Rec, Val). instrument(Rec) -> erlydb_base:get(6, Rec). instrument(Rec, Val) -> setelement(6, Rec, Val). bio(Rec) -> erlydb_base:get(7, Rec). bio(Rec, Val) -> setelement(7, Rec, Val). new(name, birth_date, instrument, bio) -> {musician, true, undefined, name, birth_date, instrument, bio}. driver() -> erlydb_base:driver({erlydb_mysql, [{last_compile_time, {{1980, 1, 1}, {0, 0, 0}}}, {outdir, "apps/music/ebin"}, debug_info, report_errors, report_warnings, {erlydb_driver, mysql}]}). count(Field, Where, Extras) -> erlydb_base:aggregate(musician, count, Field, Where, Extras). count(Field) -> erlydb_base:aggregate(musician, count, Field, undefined, undefined). count(Field, Where) -> erlydb_base:aggregate(musician, count, Field, Where, undefined). count_with(Field, Extras) -> erlydb_base:aggregate(musician, count, Field, undefined, Extras). avg(Field, Where, Extras) -> erlydb_base:aggregate(musician, avg, Field, Where, Extras). avg(Field) -> erlydb_base:aggregate(musician, avg, Field, undefined, undefined). avg(Field, Where) -> erlydb_base:aggregate(musician, avg, Field, Where, undefined). avg_with(Field, Extras) -> erlydb_base:aggregate(musician, avg, Field, undefined, Extras). min(Field, Where, Extras) -> erlydb_base:aggregate(musician, min, Field, Where, Extras). min(Field) -> erlydb_base:aggregate(musician, min, Field, undefined, undefined). min(Field, Where) -> erlydb_base:aggregate(musician, min, Field, Where, undefined). min_with(Field, Extras) -> erlydb_base:aggregate(musician, min, Field, undefined, Extras). max(Field, Where, Extras) -> erlydb_base:aggregate(musician, max, Field, Where, Extras). max(Field) -> erlydb_base:aggregate(musician, max, Field, undefined, undefined). max(Field, Where) -> erlydb_base:aggregate(musician, max, Field, Where, undefined). max_with(Field, Extras) -> erlydb_base:aggregate(musician, max, Field, undefined, Extras). sum(Field, Where, Extras) -> erlydb_base:aggregate(musician, sum, Field, Where, Extras). sum(Field) -> erlydb_base:aggregate(musician, sum, Field, undefined, undefined). sum(Field, Where) -> erlydb_base:aggregate(musician, sum, Field, Where, undefined). sum_with(Field, Extras) -> erlydb_base:aggregate(musician, sum, Field, undefined, Extras). stddev(Field, Where, Extras) -> erlydb_base:aggregate(musician, stddev, Field, Where, Extras). stddev(Field) -> erlydb_base:aggregate(musician, stddev, Field, undefined, undefined). stddev(Field, Where) -> erlydb_base:aggregate(musician, stddev, Field, Where, undefined). stddev_with(Field, Extras) -> erlydb_base:aggregate(musician, stddev, Field, undefined, Extras). find() -> erlydb_base:find(musician, undefined, undefined). find(Where) -> erlydb_base:find(musician, Where, undefined). find_with(Extras) -> erlydb_base:find(musician, undefined, Extras). find_first() -> erlydb_base:find_first(musician, undefined, undefined). find_first(Where) -> erlydb_base:find_first(musician, Where, undefined). find_first_with(Extras) -> erlydb_base:find_first(musician, undefined, Extras). find_max(Max) -> erlydb_base:find_max(musician, Max, undefined, undefined). find_max(Max, Where) -> erlydb_base:find_max(musician, Max, Where, undefined). find_max_with(Max, Extras) -> erlydb_base:find_max(musician, Max, undefined, Extras). find_range(First, Max) -> erlydb_base:find_range(musician, First, Max, undefined, undefined). find_range(First, Max, Where) -> erlydb_base:find_range(musician, First, Max, Where, undefined). find_range_with(First, Max, Extras) -> erlydb_base:find_range(musician, First, Max, undefined, Extras).
With this decompiled file, you get all things clearly behind the magic, such as, you have pair getter/setter functions for each field, for example:
Musician = musician:find({name, 'like' "Caoyuan Mus"}), %% get the 'name' field of record Musician Name = musician:name(Musician), %% set the 'name' field to "new name" and bind to a new record Musician1. Musician1 = musician:name(Musician, "new name"), %% Or, Musician2 = musician:set_fields(Musician, {name, "new name"}, {birth_day, "1940/10/9"}), %% then save one of them musician:save(Musician2).
Finally, some notices for new comer to Erlang and Erlyweb:
- In Erlang, the Variable can only be bound(set value) once , so, only Musician1 and Musician2 have the "new name", Musician will keep the original value
- For efficiency reason, if the field is varchar/text type, the getter will return a binary rather than string, which can be printed on browser directly in Erlyweb, but, if you want to use it as a string, you can apply binary_to_list(Name) on it.
发表评论
-
Updated "From Rails to Erlyweb" Part I and Part II
2007-07-21 08:29 895I've updated "From Rails t ... -
Updated "From Rails to Erlyweb" Part I and Part II
2007-07-21 08:29 1089I've updated "From Rails t ... -
Updated "From Rails to Erlyweb" Part I and Part II
2007-07-21 08:29 881I've updated "From Rails t ... -
AIOTrade Is Open for Submissions
2007-07-06 10:34 1029Here's the announcement from M ... -
From Rails to Erlyweb - Part IV
2007-07-10 11:43 984IV. Support Mysql Spatial Exten ... -
From Rails to Erlyweb - Part IV
2007-07-10 11:43 1043IV. Support Mysql Spatial Exten ... -
From Rails to Erlyweb - Part IV
2007-07-10 11:43 1069IV. Support Mysql Spatial Exten ... -
HTML Entity Refs and xmerl
2007-07-05 09:33 1240According to [erlang-bugs] xme ... -
AIOTrade Is Open for Submissions
2007-07-06 10:34 1049Here's the announcement from M ... -
HTML Entity Refs and xmerl
2007-07-05 09:33 1081According to [erlang-bugs] xme ... -
AIOTrade Is Open for Submissions
2007-07-06 10:34 965Here's the announcement from M ... -
HTML Entity Refs and xmerl
2007-07-05 09:33 1283According to [erlang-bugs] xme ... -
AIOTrade Goes to BSD License Again
2007-06-21 13:19 1157Updated June 21: I've updated t ... -
AIOTrade Goes to BSD License Again
2007-06-21 13:19 998Updated June 21: I've updated t ... -
AIOTrade Goes to BSD License Again
2007-06-21 13:19 1057Updated June 21: I've updated t ... -
Mac + Virtual Windows + CDMA上网
2007-06-17 10:54 2733换了Macbook,但手上的CDMA上网套餐只能在Window ... -
Vi Module Meets ErlyBird
2007-05-27 08:20 1251There have been several Erlang ... -
ErlyBird 0.11.2 released - An Erlang IDE based on NetBeans
2007-05-18 07:57 1308I'm please to announce ErlyBird ... -
Vi Module Meets ErlyBird
2007-05-27 08:20 1155There have been several Erlang ... -
Vi Module Meets ErlyBird
2007-05-27 08:20 1097There have been several Erlang ...
相关推荐
虚拟机vmware workstatiions pro 17.6.1个人使用免费,不用证书直接安装使用
scipy-1.2.0-cp35-cp35m-linux_armv7l.whl
基于springboot爬虫高考志愿智能推荐系统源码数据库文档.zip
bimdata_api_client-4.3.0-py3-none-any.whl
Pillow-7.0.0-cp37-cp37m-linux_armv7l.whl
onnxruntime-1.16.2-cp38-cp38-win_amd64.whl
opencv_python-4.4.0.46-cp39-cp39-linux_armv7l.whl
论文描述:该论文研究了某一特定领域的问题,并提出了新的解决方案。论文首先对问题进行了详细的分析和理解,并对已有的研究成果进行了综述。然后,论文提出了一种全新的解决方案,包括算法、模型或方法。在整个研究过程中,论文使用了合适的实验设计和数据集,并进行了充分的实验验证。最后,论文对解决方案的性能进行了全面的评估和分析,并提出了进一步的研究方向。 源码内容描述:该源码实现了论文中提出的新的解决方案。源码中包含了算法、模型或方法的具体实现代码,以及相关的数据预处理、实验设计和性能评估代码。源码中还包括了合适的注释和文档,以方便其他研究者理解和使用。源码的实现应该具有可读性、可维护性和高效性,并能够复现论文中的实验结果。此外,源码还应该尽可能具有通用性,以便在其他类似问题上进行进一步的应用和扩展。
ta_lib-0.5.1-cp313-cp313-win32.whl
STM32开发相关软件ISP 程序下载STM32开发相关软件ISP 程序下载提取方式是百度网盘分享地址
java源码资源基于JAVA的UDP服务器模型源代码提取方式是百度网盘分享地址
基于Python+Django的电影票房数据分析系统源码数据库文档.zip
onnxruntime-1.19.0-cp311-cp311-win_amd64.whl
Java固定资产管理系统源码 固定资产管理系统是对高校固定资产的一个信息化管理系统,基本功能包括:对固定资产的购进、接触、销毁,对物品的使用状态、借出状态、库存状态等进行标识,对各类物品进行编号,根据编号进行查询,根据名称进行查询等。本系统结构如下: (1)系统登录:用户登录模块:登录功能 重置 (2)系统用户管理:对系统用户的增加 系统用户的权限修改 系统用户的删除 分配系统用户的权限 修改本身登录密码 资产的相关维护 (3)员工信息管理:教工的增加、修改、删除、查询 (4)资产入库管理:资产的属性修改 资产的报废删除 资产的属性查询 (5)资产维护管理:物资的维修、维护物资的信息修改 (6)资产借还管理:增加借出资产 查询借出资产 归还已借出资产(7)打印报表
matplotlib-3.7.0-cp311-cp311-linux_armv7l.whl
基于springboot的房屋租赁管理系统源码数据库文档.zip
opencv_contrib_python-4.1.0.25-cp35-cp35m-linux_armv7l.whl
bimdata_api_client-3.1.0-py3-none-any.whl
本23级物流01班谭敏20234970114.docx
opencv_contrib_python-4.5.2.52-cp39-cp39-linux_armv7l.whl