最近发现golang社区里出了一个新星的微服务框架,来自好未来,光看这个名字,就很有奔头,之前,也只是玩过go-micro,其实真正的还没有在项目中运用过,只是觉得 微服务,grpc 这些很高大尚,还没有在项目中,真正的玩过,我看了一下官方提供的工具真的很好用,只需要定义好,舒适文件jia结构 都生成了,只需要关心业务,加上最近 有个投票的活动,加上最近这几年中台也比较火,所以决定玩一下,

开源地址: https://github.com/jackluo2012/datacenter

先聊聊中台架构思路吧:

中台的概念大概就是把一个一个的app 统一起来,反正我是这样理解的。

先聊用户服务吧,现在一个公司有很多的公众号,小程序,微信的,支付宝的,还有xxx xxx ,很多的平台,每次开发的时候,我们总是需要做用户登陆的服务,不停的复制代码,然后我们就在思考能不能有一套独立的用户服务,只需要告诉我你需要传个你要登陆的平台(比如微信),微信登陆,需要的是客户端返回给服务端一个code ,然后服务端拿着这个code去微信获取用户信息,反正大家都明白。

我们决定,将所有的信息 弄到 配置公共服务中去,里面在存,微信,支付宝,以及其它平台的 appid ,appkey,还有支付的appid,appkey,这样就写一套。


最后说说实现吧,整个就一个repo:

  • 网关,我们用的是: go-zero的Api服务
  • 其它它的是服务,我们就是用的go-zero的rpc服务

看下目录结构

整个项目完成,我一个人操刀, 写了1个来星期,我就实现了上面的中台系统。

datacenter-api服务

先看官方文档 https://www.yuque.com/tal-tech/go-zero/yaoehb

我们先把网关搭建起来

➜ blogs mkdir datacenter && cd datacenter
➜ datacenter go mod init datacenter
go: creating new go.mod: module datacenter
➜ datacenter

查看book目录:

➜  datacenter tree
.
└── go.mod0 directories, 1 file

创建api文件

➜  datacenter goctl api -o datacenter.api
Done.
➜  datacenter tree
.
├── datacenter.api
└── go.mod

定义api服务

分别包含了上面的 公共服务用户服务投票活动服务

info(title: "中台系统"desc: "中台系统"author: "jackluo"email: "net.webjoy@gmail.com"
)// 获取 应用信息
type Beid struct {Beid int64 `json:"beid"`
}
type Token struct{Token string `json:"token"`
}
type WxTicket struct{Ticket string `json:"ticket"`
}
type Application struct {Sname string `json:"Sname"` //名称Logo string `json:"logo"` // loginIsclose int64 `json:"isclose"` //是否关闭Fullwebsite string `json:"fullwebsite"` // 全站名称
}
type SnsReq struct{BeidPtyid int64  `json:"ptyid"` //对应平台BackUrl string `json:"back_url"` //登陆返回的地址
}
type SnsResp struct{BeidPtyid int64  `json:"ptyid"` //对应平台Appid string  `json:"appid"` //sns 平台的idTitle string  `json:"title"` //名称LoginUrl string `json:"login_url"` //微信登陆的地址
}type WxShareResp struct {Appid string `json:"appid"`Timestamp int64 `json:"timestamp"`Noncestr string `json:"noncestr"`Signature string `json:"signature"`
}@server(group: common
)
service datacenter-api {@doc(summary: "获取站点的信息")@handler votesVerificationget /MP_verify_NT04cqknJe0em3mT.txt (SnsReq) returns (SnsResp)@handler appInfoget /common/appinfo (Beid) returns (Application)@doc(summary: "获取站点的社交属性信息")@handler snsInfopost /common/snsinfo (SnsReq) returns (SnsResp)// 获取分享的@handler wxTicketpost /common/wx/ticket (SnsReq) returns (WxShareResp)}// 上传需要登陆
@server(jwt: Authgroup: common
)
service datacenter-api {@doc(summary: "七牛上传凭证")@handler qiuniuTokenpost /common/qiuniu/token (Beid) returns (Token)
}// 注册请求
type RegisterReq struct {// TODO: add members here and delete this commentMobile   string `json:"mobile"` // 基本一个手机号码就完事Password string `json:"password"`Smscode    string `json:"smscode"` // 短信码
}
// 登陆请求
type LoginReq struct{Mobile   string `json:"mobile"`Type int64 `json:"type"`    // 1.密码登陆,2.短信登陆Password string `json:"password"`
}
// 微信登陆
type WxLoginReq struct {Beid      int64  `json:"beid"` // 应用idCode string `json:"code"` // 微信登陆密钥Ptyid      int64  `json:"ptyid"` // 对应平台
}//返回用户信息
type UserReply struct {Auid       int64  `json:"auid"`Uid       int64  `json:"uid"`Beid      int64  `json:"beid"` // 应用idPtyid      int64  `json:"ptyid"` // 对应平台Username string `json:"username"`Mobile   string `json:"mobile"`Nickname string `json:"nickname"`Openid string `json:"openid"`Avator string `json:"avator"`JwtToken
}
// 返回APPUser
type AppUser struct{Uid       int64  `json:"uid"`Auid       int64  `json:"auid"`Beid      int64  `json:"beid"` // 应用idPtyid      int64  `json:"ptyid"` // 对应平台Nickname string `json:"nickname"`Openid string `json:"openid"`Avator string `json:"avator"`
}type LoginAppUser struct{Uid       int64  `json:"uid"`Auid       int64  `json:"auid"`Beid      int64  `json:"beid"` // 应用idPtyid      int64  `json:"ptyid"` // 对应平台Nickname string `json:"nickname"`Openid string `json:"openid"`Avator string `json:"avator"`JwtToken
}type JwtToken struct {AccessToken  string `json:"access_token,omitempty"`AccessExpire int64  `json:"access_expire,omitempty"`RefreshAfter int64  `json:"refresh_after,omitempty"`
}type UserReq struct{Auid       int64  `json:"auid"`Uid       int64  `json:"uid"`Beid      int64  `json:"beid"` // 应用idPtyid      int64  `json:"ptyid"` // 对应平台
}type Request {Name string `path:"name,options=you|me"`
}
type Response {Message string `json:"message"`
}@server(group: user
)
service user-api {@handler pingpost /user/ping ()@handler registerpost /user/register (RegisterReq) returns (UserReply)@handler loginpost /user/login (LoginReq) returns (UserReply)@handler wxloginpost /user/wx/login (WxLoginReq) returns (LoginAppUser)@handler code2Sessionget /user/wx/login () returns (LoginAppUser)
}
@server(jwt: Authgroup: usermiddleware: Usercheck
)
service user-api {@handler userInfoget /user/dc/info (UserReq) returns (UserReply)
}// 投票活动api
type Actid struct {Actid       int64  `json:"actid"` //活动id
}type VoteReq struct {Aeid       int64  `json:"aeid"` // 作品idActid
}
type VoteResp struct {VoteReqVotecount       int64  `json:"votecount"` //投票票数Viewcount       int64  `json:"viewcount"` //浏览数
}// 活动返回的参数
type ActivityResp struct {Actid           int64  `json:"actid"`Title           string  `json:"title"` //活动名称Descr           string  `json:"descr"` //活动描述StartDate       int64  `json:"start_date"` //活动时间EnrollDate      int64  `json:"enroll_date"` //投票时间EndDate           int64  `json:"end_date"` //活动结束时间Votecount       int64  `json:"votecount"` //当前活动的总票数Viewcount       int64  `json:"viewcount"` //当前活动的总浏览数Type            int64 `json:"type"` //投票方式Num                int64 `json:"num"` //投票几票
}//报名
type EnrollReq struct {ActidName           string  `json:"name"` // 名称Address           string  `json:"address"` //地址Images           []string  `json:"images"` //作品图片Descr           string  `json:"descr"` // 作品描述
}// 作品返回
type EnrollResp struct {ActidAeid        int64 `json:"aeid"` // 作品idName           string  `json:"name"` // 名称Address           string  `json:"address"` //地址Images           []string  `json:"images"` //作品图片Descr           string  `json:"descr"` // 作品描述Votecount       int64  `json:"votecount"` //当前活动的总票数Viewcount       int64  `json:"viewcount"` //当前活动的总浏览数}@server(group: votes
)
service votes-api {@doc(summary: "获取活动的信息")@handler activityInfoget /votes/activity/info (Actid) returns (ActivityResp)@doc(summary: "活动访问+1")@handler activityIcrViewget /votes/activity/view (Actid) returns (ActivityResp)@doc(summary: "获取报名的投票作品信息")@handler enrollInfoget /votes/enroll/info (VoteReq) returns (EnrollResp)@doc(summary: "获取报名的投票作品列表")@handler enrollListsget /votes/enroll/lists (Actid)    returns(EnrollResp)
}@server(jwt: Authgroup: votesmiddleware: Usercheck
)
service votes-api {@doc(summary: "投票")@handler votepost /votes/vote (VoteReq) returns (VoteResp)@handler enrollpost /votes/enroll (EnrollReq) returns (EnrollResp)
}

上面基本上写就写的API及文档的思路

生成datacenter api服务

➜  datacenter goctl api go -api datacenter.api -dir .
Done.
➜  datacenter tree
.
├── datacenter.api
├── etc
│   └── datacenter-api.yaml
├── go.mod
├── internal
│   ├── config
│   │   └── config.go
│   ├── handler
│   │   ├── common
│   │   │   ├── appinfohandler.go
│   │   │   ├── qiuniutokenhandler.go
│   │   │   ├── snsinfohandler.go
│   │   │   ├── votesverificationhandler.go
│   │   │   └── wxtickethandler.go
│   │   ├── routes.go
│   │   ├── user
│   │   │   ├── code2sessionhandler.go
│   │   │   ├── loginhandler.go
│   │   │   ├── pinghandler.go
│   │   │   ├── registerhandler.go
│   │   │   ├── userinfohandler.go
│   │   │   └── wxloginhandler.go
│   │   └── votes
│   │       ├── activityicrviewhandler.go
│   │       ├── activityinfohandler.go
│   │       ├── enrollhandler.go
│   │       ├── enrollinfohandler.go
│   │       ├── enrolllistshandler.go
│   │       └── votehandler.go
│   ├── logic
│   │   ├── common
│   │   │   ├── appinfologic.go
│   │   │   ├── qiuniutokenlogic.go
│   │   │   ├── snsinfologic.go
│   │   │   ├── votesverificationlogic.go
│   │   │   └── wxticketlogic.go
│   │   ├── user
│   │   │   ├── code2sessionlogic.go
│   │   │   ├── loginlogic.go
│   │   │   ├── pinglogic.go
│   │   │   ├── registerlogic.go
│   │   │   ├── userinfologic.go
│   │   │   └── wxloginlogic.go
│   │   └── votes
│   │       ├── activityicrviewlogic.go
│   │       ├── activityinfologic.go
│   │       ├── enrollinfologic.go
│   │       ├── enrolllistslogic.go
│   │       ├── enrolllogic.go
│   │       └── votelogic.go
│   ├── middleware
│   │   └── usercheckmiddleware.go
│   ├── svc
│   │   └── servicecontext.go
│   └── types
│       └── types.go
└── datacenter.go14 directories, 43 files

我们打开 etc/datacenter-api.yaml 把必要的配置信息加上

Name: datacenter-api
Log:Mode: console
Host: 0.0.0.0
Port: 8857
Auth:AccessSecret: 你的jwtwon SecretAccessExpire: 86400
CacheRedis:
- Host: 127.0.0.1:6379Pass: 密码Type: node
UserRpc:Etcd:Hosts:- 127.0.0.1:2379Key: user.rpc
CommonRpc:Etcd:Hosts:- 127.0.0.1:2379Key: common.rpc
VotesRpc:Etcd:Hosts:- 127.0.0.1:2379Key: votes.rpc

上面的 UserRpcCommonRpc ,还有 VotesRpc 这些我先写上,后面再来慢慢加。

我们先来写 CommonRpc 服务。

CommonRpc服务

新建项目目录

➜  datacenter mkdir -p common/rpc && cd common/rpc

直接就新建在了,datacenter目录中,因为common 里面,可能以后会不只会提供rpc服务,可能还有api的服务,所以又加了rpc目录

goctl创建模板

➜  rpc goctl rpc template -o=common.proto
➜  rpc ls
common.proto

往里面填入内容:

➜  rpc cat common.proto
syntax = "proto3";package common;message BaseAppReq{int64 beid=1;
}message BaseAppResp{int64 beid=1;string logo=2;string sname=3;int64 isclose=4;string fullwebsite=5;
}// 请求的api
message AppConfigReq {int64 beid=1;int64 ptyid=2;
}// 返回的值
message AppConfigResp {int64 id=1;int64 beid=2;int64 ptyid=3;string appid=4;string appsecret=5;string title=6;
}service Common {rpc GetAppConfig(AppConfigReq) returns(AppConfigResp);rpc GetBaseApp(BaseAppReq) returns(BaseAppResp);
}

gotcl生成rpc服务

➜  rpc goctl rpc proto -src common.proto -dir .
protoc  -I=/Users/jackluo/works/blogs/datacenter/common/rpc common.proto --go_out=plugins=grpc:/Users/jackluo/works/blogs/datacenter/common/rpc/common
Done.
➜ rpc tree
.
├── common
│  └── common.pb.go
├── common.go
├── common.proto
├── commonclient
│  └── common.go
├── etc
│  └── common.yaml
└── internal
├── config
│  └── config.go
├── logic
│  ├── getappconfiglogic.go
│  └── getbaseapplogic.go
├── server
│  └── commonserver.go
└── svc
└── servicecontext.go8 directories, 10 files

基本上,就把所有的目录规范和结构的东西都生成了,就不用纠结项目目录了,怎么放了,怎么组织了。

看一下,配置信息,里面可以写入mysql和其它redis的信息:

Name: common.rpc
ListenOn: 127.0.0.1:8081
Mysql:DataSource: root:admin@tcp(127.0.0.1:3306)/datacenter?charset=utf8&parseTime=true&loc=Asia%2FShanghai
CacheRedis:
- Host: 127.0.0.1:6379Pass:Type: node
Etcd:Hosts:- 127.0.0.1:2379Key: common.rpc

我们再来加上数据库服务:

➜  rpc cd ..
➜  common ls
rpc
➜  common pwd
/Users/jackluo/works/blogs/datacenter/common
➜  common goctl model mysql datasource -url="root:admin@tcp(127.0.0.1:3306)/datacenter" -table="base_app" -dir ./model -c
Done.
➜  common tree
.
├── model
│   ├── baseappmodel.go
│   └── vars.go
└── rpc├── common│   └── common.pb.go├── common.go├── common.proto├── commonclient│   └── common.go├── etc│   └── common.yaml└── internal├── config│   └── config.go├── logic│   ├── getappconfiglogic.go│   └── getbaseapplogic.go├── server│   └── commonserver.go└── svc└── servicecontext.go10 directories, 12 files

这样基本的一个 rpc 就写完了,然后我们将rpc 和model 还有api串连起来,这个官方的文档已经很详细了,这里就只是贴一下代码:

➜  common cat rpc/internal/config/config.go
package configimport ("github.com/tal-tech/go-zero/core/stores/cache""github.com/tal-tech/go-zero/zrpc"
)type Config struct {zrpc.RpcServerConfMysql struct {DataSource string}CacheRedis cache.ClusterConf
}

再在svc中修改:

➜  common cat rpc/internal/svc/servicecontext.go
package svcimport ("datacenter/common/model""datacenter/common/rpc/internal/config""github.com/tal-tech/go-zero/core/stores/sqlx"
)type ServiceContext struct {c              config.ConfigAppConfigModel model.AppConfigModelBaseAppModel   model.BaseAppModel
}func NewServiceContext(c config.Config) *ServiceContext {conn := sqlx.NewMysql(c.Mysql.DataSource)apm := model.NewAppConfigModel(conn, c.CacheRedis)bam := model.NewBaseAppModel(conn, c.CacheRedis)return &ServiceContext{c:              c,AppConfigModel: apm,BaseAppModel:   bam,}
}

上面的代码已经将 rpcmodel 数据库关联起来了,我们现在再将 rpcapi 关联起来:

➜  datacenter cat internal/config/config.gopackage configimport ("github.com/tal-tech/go-zero/core/stores/cache""github.com/tal-tech/go-zero/rest""github.com/tal-tech/go-zero/zrpc"
)type Config struct {rest.RestConfAuth struct {AccessSecret stringAccessExpire int64}UserRpc   zrpc.RpcClientConfCommonRpc zrpc.RpcClientConfVotesRpc  zrpc.RpcClientConfCacheRedis cache.ClusterConf
}

加入 svc 服务中:

➜  datacenter cat internal/svc/servicecontext.go
package svcimport ("context""datacenter/common/rpc/commonclient""datacenter/internal/config""datacenter/internal/middleware""datacenter/shared""datacenter/user/rpc/userclient""datacenter/votes/rpc/votesclient""fmt""net/http""time""github.com/tal-tech/go-zero/core/logx""github.com/tal-tech/go-zero/core/stores/cache""github.com/tal-tech/go-zero/core/stores/redis""github.com/tal-tech/go-zero/core/syncx""github.com/tal-tech/go-zero/rest""github.com/tal-tech/go-zero/zrpc""google.golang.org/grpc"
)type ServiceContext struct {Config           config.ConfigGreetMiddleware1 rest.MiddlewareGreetMiddleware2 rest.MiddlewareUsercheck        rest.MiddlewareUserRpc          userclient.User //用户CommonRpc        commonclient.CommonVotesRpc         votesclient.VotesCache            cache.CacheRedisConn        *redis.Redis
}func timeInterceptor(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error {stime := time.Now()err := invoker(ctx, method, req, reply, cc, opts...)if err != nil {return err}fmt.Printf("调用 %s 方法 耗时: %v\n", method, time.Now().Sub(stime))return nil
}
func NewServiceContext(c config.Config) *ServiceContext {ur := userclient.NewUser(zrpc.MustNewClient(c.UserRpc, zrpc.WithUnaryClientInterceptor(timeInterceptor)))cr := commonclient.NewCommon(zrpc.MustNewClient(c.CommonRpc, zrpc.WithUnaryClientInterceptor(timeInterceptor)))vr := votesclient.NewVotes(zrpc.MustNewClient(c.VotesRpc, zrpc.WithUnaryClientInterceptor(timeInterceptor)))//缓存ca := cache.NewCache(c.CacheRedis, syncx.NewSharedCalls(), cache.NewCacheStat("dc"), shared.ErrNotFound)rcon := redis.NewRedis(c.CacheRedis[0].Host, c.CacheRedis[0].Type, c.CacheRedis[0].Pass)return &ServiceContext{Config:           c,GreetMiddleware1: greetMiddleware1,GreetMiddleware2: greetMiddleware2,Usercheck:        middleware.NewUserCheckMiddleware().Handle,UserRpc:          ur,CommonRpc:        cr,VotesRpc:         vr,Cache:            ca,RedisConn:        rcon,}
}

这样基本上,我们就可以在 logic 的文件目录中调用了:

cat internal/logic/common/appinfologic.gopackage logicimport ("context""datacenter/internal/svc""datacenter/internal/types""datacenter/shared""datacenter/common/model""datacenter/common/rpc/common""github.com/tal-tech/go-zero/core/logx"
)type AppInfoLogic struct {logx.Loggerctx    context.ContextsvcCtx *svc.ServiceContext
}func NewAppInfoLogic(ctx context.Context, svcCtx *svc.ServiceContext) AppInfoLogic {return AppInfoLogic{Logger: logx.WithContext(ctx),ctx:    ctx,svcCtx: svcCtx,}
}func (l *AppInfoLogic) AppInfo(req types.Beid) (appconfig *common.BaseAppResp, err error) {//检查 缓存中是否有值err = l.svcCtx.Cache.GetCache(model.GetcacheBaseAppIdPrefix(req.Beid), appconfig)if err != nil && err == shared.ErrNotFound {appconfig, err = l.svcCtx.CommonRpc.GetBaseApp(l.ctx, &common.BaseAppReq{Beid: req.Beid,})if err != nil {return}err = l.svcCtx.Cache.SetCache(model.GetcacheBaseAppIdPrefix(req.Beid), appconfig)}return
}

这样,基本就连接起来了,其它基本上就不用改了,UserRPCVotesRPC 类似,这里就不在写了。

使用心得

go-zero 的确香,因为它有一个 goctl 的工具,他可以自动的把代码结构全部的生成好,我们就不再去纠结,目录结构 ,怎么组织,没有个好几年的架构能力是不好实现的,有什么规范那些,并发,熔断,完全不用,考滤其它的,专心的实现业务就好,像微服务,还要有服务发现,一系列的东西,都不用关心,因为 go-zero 内部已经实现了。

我写代码也写了有10多年了,之前一直用的 php,比较出名的就 laravel,thinkphp,基本上就是模块化的,像微服那些实现直来真的有成本,但是你用上go-zero,你就像调api接口一样简单的开发,其它什么服务发现,那些根本就不用关注了,只需要关注业务。

一个好的语言,框架,他们的底层思维,永远都是效率高,不加班的思想,我相信go-zero会提高你和你团队或是公司的效率。go-zero的作者说,他们有个团队专门整理go-zero框架,目的也应该很明显,那就是提高,他们自己的开发效率,流程化,标准化,是提高工作效率的准则,像我们平时遇到了问题,或是遇到了bug,我第一个想到的不是怎么去解决我的bug,而是在想我的流程是不是有问题,我的哪个流程会导致bug,最后我相信 go-zero 能成为 微服务开发 的首选框架。

最后说说遇到的坑吧:

  • grpc

grpc 本人第一次用,然后就遇到了,有些字符为空时,字段值不显示的问题:

通过 grpc 官方库中的 jsonpb 来实现,官方在它的设定中有一个结构体用来实现 protoc buffer 转换为JSON结构,并可以根据字段来配置转换的要求。

  • 跨域问题

go-zero 中设置了,感觉没有效果,大佬说通过nginx 设置,后面发现还是不行,最近强行弄到了一个域名下,后面有时间再解决。

  • sqlx

go-zerosqlx 问题,这个真的费了很长的时间:

time.Time` 这个数据结构,数据库中用的是 timestamp 这个 比如我的字段 是delete_at 默认数库设置的是null ,结果插入的时候,就报了 `Incorrect datetime value: '0000-00-00' for column 'deleted_at' at row 1"}` 这个错,查询的时候报 `deleted_at\": unsupported Scan, storing driver.Value type \u003cnil\u003e into type *time.Time"

后面果断去掉了这个字段,字段上面加上 .omitempty 这个标签,好像也有用,db:".omitempty"

其次就是这个 Conversion from collation utf8_general_ci into utf8mb4_unicode_ci,这个导致的大概原因是,现在都喜欢用emj表情了,mysql数据识别不了。

  • 数据连接

mysql 这边照样按照原始的方式,将配置文件修改编码格式,重新创建数据库,并且设置数据库编码为utf8mb4,排序规则为 utf8mb4_unicode_ci

这样的话,所有的表还有string字段都是这个编码格式,如果不想所有的都是,可以单独设置,这个不是重点.因为在navicat上都好设置,手动点一下就行了

重点来了:golang中使用的是 github.com/go-sql-driver/mysql 驱动,将连接 mysqldsn(因为我这使用的是gorm,所以dsn可能跟原生的格式不太一样,不过没关系, 只需要关注 charsetcollation 就行了)

root:password@/name?parseTime=True&loc=Local&charset=utf8` 修改为:
`root:password@/name?parseTime=True&loc=Local&charset=utf8mb4&collation=utf8mb4_unicode_ci

go-zero 项目地址

https://github.com/tal-tech/go-zero

欢迎点赞

我用 go-zero 一周实现了一个中台系统,已开源!相关推荐

  1. 项目经理怎么在两周内熟悉一个项目的业务?

    项目经理空降到一个进行中的项目,怎么在两周内熟悉一个项目的业务? 四步帮你解决:明确项目业务目标,了解系统功能模块,弄清系统核心业务流程,多使用系统. 一.明确项目业务目标 明确项目业务目标,也就是了 ...

  2. 笔记:猎头如何在一周之内“摸清”一个行业

    文章目录 猎头告诉你:如何在一周内摸清一个行业? 1. 圈定对应行业,或拓展至相关行业 2.确定行业内 TOP5 或 TOP10 企业名单,并关注名单排序的标准,及近几年排名变化. 3.锁定所列企业组 ...

  3. 程序员一周内了解一个行业的方法

    感觉很适合程序员阅读,毕竟搞技术的都很宅,对应一些非技术的事物,都是抱着事不关己高高挂起的态度. 原文:程序员一周内了解一个行业的方法 写得有点像做FBI的,所以看看不会有什么坏处的. 我们都是有理想 ...

  4. C语言编程>第十六周 ⑧ S是一个由数字和字母字符串组成的字符串,由变量len传入字符串长度。请补充fun函数,该函数的功能是把字符串s中的数字字符转换成数字并存放到整型数组a中

    例题:S是一个由数字和字母字符串组成的字符串,由变量len传入字符串长度.请补充fun函数,该函数的功能是把字符串s中的数字字符转换成数字并存放到整型数组a中,函数返回数组a的长度. 例如,s=&qu ...

  5. C语言编程>第十一周 ② N名学生的成绩已在主函数中放入一个带头结点的链表结构中,a指向链表的头结点。请编写函数fun,它的功能是:找出学生的最高分,由函数返回。

    例题:N名学生的成绩已在主函数中放入一个带头结点的链表结构中,a指向链表的头结点.请编写函数fun,它的功能是:找出学生的最高分,由函数返回. 请勿改动主函数main与其它函数中的任何内容,仅在函数f ...

  6. C语言编程>第十一周 ⑤ 请编写一个函数,用来删除字符串中的所有空格。

    例题:请编写一个函数,用来删除字符串中的所有空格. 例如,输入we l come,则输出为welcome. 请勿改动主函数main与其它函数中的任何内容,仅在函数fun的花括号中填入所编写的若干语句. ...

  7. C语言编程>第七周 ⑧ 请编一个函数void fun(int a[M][N],int b[N]),c指向一个M行N列的二维数组,求出二维数组每列中最大元素,并依次放入b所指一维数组中。

    例题:请编一个函数void fun(int a[M][N],int b[N]),c指向一个M行N列的二维数组,求出二维数组每列中最大元素,并依次放入b所指一维数组中.二维数组中的数己在主函数中赋予. ...

  8. sql查询近七天,近两周,近一个月的数据

    sql查询近一周,近两周,近一个月数据: ***注意:***DATE(此处是你需要查询的表中关于时间的字段名) ,INTERVAL后的数字可以改变,想查询几天数据就可以查询近几天数据.其他部分固定. ...

  9. 话说Delphi XE7编写安卓APP还是不错的,一周写一个APP系统

     话说Delphi XE7编写安卓APP还是不错的,花了一周写一个APP系统,熟悉delphi开发的,很快就能开发出一个功能完善的APP系统,该系统的简单介绍, 为了配合我们的酒店系统的应用扩展, ...

最新文章

  1. Nature Microbiology:全球污水处理厂细菌群落揭秘!
  2. OpenAI的GPT-3花费了1200万美元,现在放出商用API,人人皆可用
  3. Linux(Centos)快速搭建SVN服务器
  4. 使用Nginx实现服务器反向代理和负载均衡
  5. Service Cloud: Quick Look
  6. 图的m着色问题pascal程序
  7. 机器学习/深度学习 问题总结及解答
  8. 详解MQ消息队列及四大主流MQ的优缺点
  9. Problem G. Pandaria(线段树合并 + Kruskal 重构树)
  10. JS对象拷贝:深拷贝和浅拷贝
  11. Hive数据据类型 DDL DML
  12. matlab void函数,MATLAB基本函数
  13. python实现原始字典数据中键值互换得到新的字典数据
  14. Oracle10g BIGFILE表空间带来的好处
  15. python获取读取文件的文件名_python文件名获取文件路径
  16. CCNA 中文教材 (pdf)
  17. 设计模式(一) 六大原则
  18. SQL Server 2008 R2的完全卸载
  19. 单片机驱动为什么要设置为低电平有效?
  20. san分布式共享文件系统_SAN存储区域网络共享软件全攻略

热门文章

  1. 区块链共享数据架构设计
  2. autocad2014 第二次打开闪退_解决win10下autocad2014闪退的修复方法
  3. vue-devtools 各版本安装
  4. Java修炼 之 基础篇(二)Java语言构成
  5. 【vue】实现超过两行或多行显示展开收起 (单个展开收起和数组多个展开收起)
  6. 漫画绘制技法大放送(上)
  7. JavaSE——StringBuffer与StringBuilder拼接字符串详细解释
  8. GUI小工具-网盘搜索器
  9. 用C语言编辑一光年相当于多少米,天文学里一光年是多长的距离?
  10. request.setattribute详解